aboutsummaryrefslogtreecommitdiffstats
path: root/arch
diff options
context:
space:
mode:
authorIsaku Yamahata <yamahata@valinux.co.jp>2008-05-19 09:13:38 -0400
committerTony Luck <tony.luck@intel.com>2008-05-27 18:03:29 -0400
commit498c5170472ff0c03a29d22dbd33225a0be038f4 (patch)
treee1f972fcbf3dc96219736723a1ff78452c5747e1 /arch
parent02e32e36f42f8ea7ee6060d02f2d69ad5bad6d50 (diff)
[IA64] pvops: paravirtualize ivt.S
paravirtualize ivt.S which implements fault handler in hand written assembly code. They includes sensitive or performance critical privileged instructions. So they need paravirtualization. Cc: Keith Owens <kaos@ocs.com.au> Cc: tgingold@free.fr Cc: Akio Takebe <takebe_akio@jp.fujitsu.com> Signed-off-by: Yaozu (Eddie) Dong <eddie.dong@intel.com> Signed-off-by: Isaku Yamahata <yamahata@valinux.co.jp> Signed-off-by: Tony Luck <tony.luck@intel.com>
Diffstat (limited to 'arch')
-rw-r--r--arch/ia64/kernel/ivt.S249
1 files changed, 122 insertions, 127 deletions
diff --git a/arch/ia64/kernel/ivt.S b/arch/ia64/kernel/ivt.S
index 80b44ea052d7..23749ed3cf08 100644
--- a/arch/ia64/kernel/ivt.S
+++ b/arch/ia64/kernel/ivt.S
@@ -12,6 +12,14 @@
12 * 12 *
13 * 00/08/23 Asit Mallick <asit.k.mallick@intel.com> TLB handling for SMP 13 * 00/08/23 Asit Mallick <asit.k.mallick@intel.com> TLB handling for SMP
14 * 00/12/20 David Mosberger-Tang <davidm@hpl.hp.com> DTLB/ITLB handler now uses virtual PT. 14 * 00/12/20 David Mosberger-Tang <davidm@hpl.hp.com> DTLB/ITLB handler now uses virtual PT.
15 *
16 * Copyright (C) 2005 Hewlett-Packard Co
17 * Dan Magenheimer <dan.magenheimer@hp.com>
18 * Xen paravirtualization
19 * Copyright (c) 2008 Isaku Yamahata <yamahata at valinux co jp>
20 * VA Linux Systems Japan K.K.
21 * pv_ops.
22 * Yaozu (Eddie) Dong <eddie.dong@intel.com>
15 */ 23 */
16/* 24/*
17 * This file defines the interruption vector table used by the CPU. 25 * This file defines the interruption vector table used by the CPU.
@@ -102,13 +110,13 @@ ENTRY(vhpt_miss)
102 * - the faulting virtual address uses unimplemented address bits 110 * - the faulting virtual address uses unimplemented address bits
103 * - the faulting virtual address has no valid page table mapping 111 * - the faulting virtual address has no valid page table mapping
104 */ 112 */
105 mov r16=cr.ifa // get address that caused the TLB miss 113 MOV_FROM_IFA(r16) // get address that caused the TLB miss
106#ifdef CONFIG_HUGETLB_PAGE 114#ifdef CONFIG_HUGETLB_PAGE
107 movl r18=PAGE_SHIFT 115 movl r18=PAGE_SHIFT
108 mov r25=cr.itir 116 MOV_FROM_ITIR(r25)
109#endif 117#endif
110 ;; 118 ;;
111 rsm psr.dt // use physical addressing for data 119 RSM_PSR_DT // use physical addressing for data
112 mov r31=pr // save the predicate registers 120 mov r31=pr // save the predicate registers
113 mov r19=IA64_KR(PT_BASE) // get page table base address 121 mov r19=IA64_KR(PT_BASE) // get page table base address
114 shl r21=r16,3 // shift bit 60 into sign bit 122 shl r21=r16,3 // shift bit 60 into sign bit
@@ -168,21 +176,21 @@ ENTRY(vhpt_miss)
168 dep r21=r19,r20,3,(PAGE_SHIFT-3) // r21=pte_offset(pmd,addr) 176 dep r21=r19,r20,3,(PAGE_SHIFT-3) // r21=pte_offset(pmd,addr)
169 ;; 177 ;;
170(p7) ld8 r18=[r21] // read *pte 178(p7) ld8 r18=[r21] // read *pte
171 mov r19=cr.isr // cr.isr bit 32 tells us if this is an insn miss 179 MOV_FROM_ISR(r19) // cr.isr bit 32 tells us if this is an insn miss
172 ;; 180 ;;
173(p7) tbit.z p6,p7=r18,_PAGE_P_BIT // page present bit cleared? 181(p7) tbit.z p6,p7=r18,_PAGE_P_BIT // page present bit cleared?
174 mov r22=cr.iha // get the VHPT address that caused the TLB miss 182 MOV_FROM_IHA(r22) // get the VHPT address that caused the TLB miss
175 ;; // avoid RAW on p7 183 ;; // avoid RAW on p7
176(p7) tbit.nz.unc p10,p11=r19,32 // is it an instruction TLB miss? 184(p7) tbit.nz.unc p10,p11=r19,32 // is it an instruction TLB miss?
177 dep r23=0,r20,0,PAGE_SHIFT // clear low bits to get page address 185 dep r23=0,r20,0,PAGE_SHIFT // clear low bits to get page address
178 ;; 186 ;;
179(p10) itc.i r18 // insert the instruction TLB entry 187 ITC_I_AND_D(p10, p11, r18, r24) // insert the instruction TLB entry and
180(p11) itc.d r18 // insert the data TLB entry 188 // insert the data TLB entry
181(p6) br.cond.spnt.many page_fault // handle bad address/page not present (page fault) 189(p6) br.cond.spnt.many page_fault // handle bad address/page not present (page fault)
182 mov cr.ifa=r22 190 MOV_TO_IFA(r22, r24)
183 191
184#ifdef CONFIG_HUGETLB_PAGE 192#ifdef CONFIG_HUGETLB_PAGE
185(p8) mov cr.itir=r25 // change to default page-size for VHPT 193 MOV_TO_ITIR(p8, r25, r24) // change to default page-size for VHPT
186#endif 194#endif
187 195
188 /* 196 /*
@@ -192,7 +200,7 @@ ENTRY(vhpt_miss)
192 */ 200 */
193 adds r24=__DIRTY_BITS_NO_ED|_PAGE_PL_0|_PAGE_AR_RW,r23 201 adds r24=__DIRTY_BITS_NO_ED|_PAGE_PL_0|_PAGE_AR_RW,r23
194 ;; 202 ;;
195(p7) itc.d r24 203 ITC_D(p7, r24, r25)
196 ;; 204 ;;
197#ifdef CONFIG_SMP 205#ifdef CONFIG_SMP
198 /* 206 /*
@@ -234,7 +242,7 @@ ENTRY(vhpt_miss)
234#endif 242#endif
235 243
236 mov pr=r31,-1 // restore predicate registers 244 mov pr=r31,-1 // restore predicate registers
237 rfi 245 RFI
238END(vhpt_miss) 246END(vhpt_miss)
239 247
240 .org ia64_ivt+0x400 248 .org ia64_ivt+0x400
@@ -248,11 +256,11 @@ ENTRY(itlb_miss)
248 * mode, walk the page table, and then re-execute the PTE read and 256 * mode, walk the page table, and then re-execute the PTE read and
249 * go on normally after that. 257 * go on normally after that.
250 */ 258 */
251 mov r16=cr.ifa // get virtual address 259 MOV_FROM_IFA(r16) // get virtual address
252 mov r29=b0 // save b0 260 mov r29=b0 // save b0
253 mov r31=pr // save predicates 261 mov r31=pr // save predicates
254.itlb_fault: 262.itlb_fault:
255 mov r17=cr.iha // get virtual address of PTE 263 MOV_FROM_IHA(r17) // get virtual address of PTE
256 movl r30=1f // load nested fault continuation point 264 movl r30=1f // load nested fault continuation point
257 ;; 265 ;;
2581: ld8 r18=[r17] // read *pte 2661: ld8 r18=[r17] // read *pte
@@ -261,7 +269,7 @@ ENTRY(itlb_miss)
261 tbit.z p6,p0=r18,_PAGE_P_BIT // page present bit cleared? 269 tbit.z p6,p0=r18,_PAGE_P_BIT // page present bit cleared?
262(p6) br.cond.spnt page_fault 270(p6) br.cond.spnt page_fault
263 ;; 271 ;;
264 itc.i r18 272 ITC_I(p0, r18, r19)
265 ;; 273 ;;
266#ifdef CONFIG_SMP 274#ifdef CONFIG_SMP
267 /* 275 /*
@@ -278,7 +286,7 @@ ENTRY(itlb_miss)
278(p7) ptc.l r16,r20 286(p7) ptc.l r16,r20
279#endif 287#endif
280 mov pr=r31,-1 288 mov pr=r31,-1
281 rfi 289 RFI
282END(itlb_miss) 290END(itlb_miss)
283 291
284 .org ia64_ivt+0x0800 292 .org ia64_ivt+0x0800
@@ -292,11 +300,11 @@ ENTRY(dtlb_miss)
292 * mode, walk the page table, and then re-execute the PTE read and 300 * mode, walk the page table, and then re-execute the PTE read and
293 * go on normally after that. 301 * go on normally after that.
294 */ 302 */
295 mov r16=cr.ifa // get virtual address 303 MOV_FROM_IFA(r16) // get virtual address
296 mov r29=b0 // save b0 304 mov r29=b0 // save b0
297 mov r31=pr // save predicates 305 mov r31=pr // save predicates
298dtlb_fault: 306dtlb_fault:
299 mov r17=cr.iha // get virtual address of PTE 307 MOV_FROM_IHA(r17) // get virtual address of PTE
300 movl r30=1f // load nested fault continuation point 308 movl r30=1f // load nested fault continuation point
301 ;; 309 ;;
3021: ld8 r18=[r17] // read *pte 3101: ld8 r18=[r17] // read *pte
@@ -305,7 +313,7 @@ dtlb_fault:
305 tbit.z p6,p0=r18,_PAGE_P_BIT // page present bit cleared? 313 tbit.z p6,p0=r18,_PAGE_P_BIT // page present bit cleared?
306(p6) br.cond.spnt page_fault 314(p6) br.cond.spnt page_fault
307 ;; 315 ;;
308 itc.d r18 316 ITC_D(p0, r18, r19)
309 ;; 317 ;;
310#ifdef CONFIG_SMP 318#ifdef CONFIG_SMP
311 /* 319 /*
@@ -322,7 +330,7 @@ dtlb_fault:
322(p7) ptc.l r16,r20 330(p7) ptc.l r16,r20
323#endif 331#endif
324 mov pr=r31,-1 332 mov pr=r31,-1
325 rfi 333 RFI
326END(dtlb_miss) 334END(dtlb_miss)
327 335
328 .org ia64_ivt+0x0c00 336 .org ia64_ivt+0x0c00
@@ -330,9 +338,9 @@ END(dtlb_miss)
330// 0x0c00 Entry 3 (size 64 bundles) Alt ITLB (19) 338// 0x0c00 Entry 3 (size 64 bundles) Alt ITLB (19)
331ENTRY(alt_itlb_miss) 339ENTRY(alt_itlb_miss)
332 DBG_FAULT(3) 340 DBG_FAULT(3)
333 mov r16=cr.ifa // get address that caused the TLB miss 341 MOV_FROM_IFA(r16) // get address that caused the TLB miss
334 movl r17=PAGE_KERNEL 342 movl r17=PAGE_KERNEL
335 mov r21=cr.ipsr 343 MOV_FROM_IPSR(p0, r21)
336 movl r19=(((1 << IA64_MAX_PHYS_BITS) - 1) & ~0xfff) 344 movl r19=(((1 << IA64_MAX_PHYS_BITS) - 1) & ~0xfff)
337 mov r31=pr 345 mov r31=pr
338 ;; 346 ;;
@@ -341,9 +349,9 @@ ENTRY(alt_itlb_miss)
341 ;; 349 ;;
342 cmp.gt p8,p0=6,r22 // user mode 350 cmp.gt p8,p0=6,r22 // user mode
343 ;; 351 ;;
344(p8) thash r17=r16 352 THASH(p8, r17, r16, r23)
345 ;; 353 ;;
346(p8) mov cr.iha=r17 354 MOV_TO_IHA(p8, r17, r23)
347(p8) mov r29=b0 // save b0 355(p8) mov r29=b0 // save b0
348(p8) br.cond.dptk .itlb_fault 356(p8) br.cond.dptk .itlb_fault
349#endif 357#endif
@@ -358,9 +366,9 @@ ENTRY(alt_itlb_miss)
358 or r19=r19,r18 // set bit 4 (uncached) if the access was to region 6 366 or r19=r19,r18 // set bit 4 (uncached) if the access was to region 6
359(p8) br.cond.spnt page_fault 367(p8) br.cond.spnt page_fault
360 ;; 368 ;;
361 itc.i r19 // insert the TLB entry 369 ITC_I(p0, r19, r18) // insert the TLB entry
362 mov pr=r31,-1 370 mov pr=r31,-1
363 rfi 371 RFI
364END(alt_itlb_miss) 372END(alt_itlb_miss)
365 373
366 .org ia64_ivt+0x1000 374 .org ia64_ivt+0x1000
@@ -368,11 +376,11 @@ END(alt_itlb_miss)
368// 0x1000 Entry 4 (size 64 bundles) Alt DTLB (7,46) 376// 0x1000 Entry 4 (size 64 bundles) Alt DTLB (7,46)
369ENTRY(alt_dtlb_miss) 377ENTRY(alt_dtlb_miss)
370 DBG_FAULT(4) 378 DBG_FAULT(4)
371 mov r16=cr.ifa // get address that caused the TLB miss 379 MOV_FROM_IFA(r16) // get address that caused the TLB miss
372 movl r17=PAGE_KERNEL 380 movl r17=PAGE_KERNEL
373 mov r20=cr.isr 381 MOV_FROM_ISR(r20)
374 movl r19=(((1 << IA64_MAX_PHYS_BITS) - 1) & ~0xfff) 382 movl r19=(((1 << IA64_MAX_PHYS_BITS) - 1) & ~0xfff)
375 mov r21=cr.ipsr 383 MOV_FROM_IPSR(p0, r21)
376 mov r31=pr 384 mov r31=pr
377 mov r24=PERCPU_ADDR 385 mov r24=PERCPU_ADDR
378 ;; 386 ;;
@@ -381,9 +389,9 @@ ENTRY(alt_dtlb_miss)
381 ;; 389 ;;
382 cmp.gt p8,p0=6,r22 // access to region 0-5 390 cmp.gt p8,p0=6,r22 // access to region 0-5
383 ;; 391 ;;
384(p8) thash r17=r16 392 THASH(p8, r17, r16, r25)
385 ;; 393 ;;
386(p8) mov cr.iha=r17 394 MOV_TO_IHA(p8, r17, r25)
387(p8) mov r29=b0 // save b0 395(p8) mov r29=b0 // save b0
388(p8) br.cond.dptk dtlb_fault 396(p8) br.cond.dptk dtlb_fault
389#endif 397#endif
@@ -402,7 +410,7 @@ ENTRY(alt_dtlb_miss)
402 tbit.nz p9,p0=r20,IA64_ISR_NA_BIT // is non-access bit on? 410 tbit.nz p9,p0=r20,IA64_ISR_NA_BIT // is non-access bit on?
403 ;; 411 ;;
404(p10) sub r19=r19,r26 412(p10) sub r19=r19,r26
405(p10) mov cr.itir=r25 413 MOV_TO_ITIR(p10, r25, r24)
406 cmp.ne p8,p0=r0,r23 414 cmp.ne p8,p0=r0,r23
407(p9) cmp.eq.or.andcm p6,p7=IA64_ISR_CODE_LFETCH,r22 // check isr.code field 415(p9) cmp.eq.or.andcm p6,p7=IA64_ISR_CODE_LFETCH,r22 // check isr.code field
408(p12) dep r17=-1,r17,4,1 // set ma=UC for region 6 addr 416(p12) dep r17=-1,r17,4,1 // set ma=UC for region 6 addr
@@ -411,11 +419,11 @@ ENTRY(alt_dtlb_miss)
411 dep r21=-1,r21,IA64_PSR_ED_BIT,1 419 dep r21=-1,r21,IA64_PSR_ED_BIT,1
412 ;; 420 ;;
413 or r19=r19,r17 // insert PTE control bits into r19 421 or r19=r19,r17 // insert PTE control bits into r19
414(p6) mov cr.ipsr=r21 422 MOV_TO_IPSR(p6, r21, r24)
415 ;; 423 ;;
416(p7) itc.d r19 // insert the TLB entry 424 ITC_D(p7, r19, r18) // insert the TLB entry
417 mov pr=r31,-1 425 mov pr=r31,-1
418 rfi 426 RFI
419END(alt_dtlb_miss) 427END(alt_dtlb_miss)
420 428
421 .org ia64_ivt+0x1400 429 .org ia64_ivt+0x1400
@@ -444,10 +452,10 @@ ENTRY(nested_dtlb_miss)
444 * 452 *
445 * Clobbered: b0, r18, r19, r21, r22, psr.dt (cleared) 453 * Clobbered: b0, r18, r19, r21, r22, psr.dt (cleared)
446 */ 454 */
447 rsm psr.dt // switch to using physical data addressing 455 RSM_PSR_DT // switch to using physical data addressing
448 mov r19=IA64_KR(PT_BASE) // get the page table base address 456 mov r19=IA64_KR(PT_BASE) // get the page table base address
449 shl r21=r16,3 // shift bit 60 into sign bit 457 shl r21=r16,3 // shift bit 60 into sign bit
450 mov r18=cr.itir 458 MOV_FROM_ITIR(r18)
451 ;; 459 ;;
452 shr.u r17=r16,61 // get the region number into r17 460 shr.u r17=r16,61 // get the region number into r17
453 extr.u r18=r18,2,6 // get the faulting page size 461 extr.u r18=r18,2,6 // get the faulting page size
@@ -510,21 +518,15 @@ END(ikey_miss)
510 //----------------------------------------------------------------------------------- 518 //-----------------------------------------------------------------------------------
511 // call do_page_fault (predicates are in r31, psr.dt may be off, r16 is faulting address) 519 // call do_page_fault (predicates are in r31, psr.dt may be off, r16 is faulting address)
512ENTRY(page_fault) 520ENTRY(page_fault)
513 ssm psr.dt 521 SSM_PSR_DT_AND_SRLZ_I
514 ;;
515 srlz.i
516 ;; 522 ;;
517 SAVE_MIN_WITH_COVER 523 SAVE_MIN_WITH_COVER
518 alloc r15=ar.pfs,0,0,3,0 524 alloc r15=ar.pfs,0,0,3,0
519 mov out0=cr.ifa 525 MOV_FROM_IFA(out0)
520 mov out1=cr.isr 526 MOV_FROM_ISR(out1)
527 SSM_PSR_IC_AND_DEFAULT_BITS_AND_SRLZ_I(r14, r3)
521 adds r3=8,r2 // set up second base pointer 528 adds r3=8,r2 // set up second base pointer
522 ;; 529 SSM_PSR_I(p15, p15, r14) // restore psr.i
523 ssm psr.ic | PSR_DEFAULT_BITS
524 ;;
525 srlz.i // guarantee that interruption collectin is on
526 ;;
527(p15) ssm psr.i // restore psr.i
528 movl r14=ia64_leave_kernel 530 movl r14=ia64_leave_kernel
529 ;; 531 ;;
530 SAVE_REST 532 SAVE_REST
@@ -556,10 +558,10 @@ ENTRY(dirty_bit)
556 * page table TLB entry isn't present, we take a nested TLB miss hit where we look 558 * page table TLB entry isn't present, we take a nested TLB miss hit where we look
557 * up the physical address of the L3 PTE and then continue at label 1 below. 559 * up the physical address of the L3 PTE and then continue at label 1 below.
558 */ 560 */
559 mov r16=cr.ifa // get the address that caused the fault 561 MOV_FROM_IFA(r16) // get the address that caused the fault
560 movl r30=1f // load continuation point in case of nested fault 562 movl r30=1f // load continuation point in case of nested fault
561 ;; 563 ;;
562 thash r17=r16 // compute virtual address of L3 PTE 564 THASH(p0, r17, r16, r18) // compute virtual address of L3 PTE
563 mov r29=b0 // save b0 in case of nested fault 565 mov r29=b0 // save b0 in case of nested fault
564 mov r31=pr // save pr 566 mov r31=pr // save pr
565#ifdef CONFIG_SMP 567#ifdef CONFIG_SMP
@@ -576,7 +578,7 @@ ENTRY(dirty_bit)
576 ;; 578 ;;
577(p6) cmp.eq p6,p7=r26,r18 // Only compare if page is present 579(p6) cmp.eq p6,p7=r26,r18 // Only compare if page is present
578 ;; 580 ;;
579(p6) itc.d r25 // install updated PTE 581 ITC_D(p6, r25, r18) // install updated PTE
580 ;; 582 ;;
581 /* 583 /*
582 * Tell the assemblers dependency-violation checker that the above "itc" instructions 584 * Tell the assemblers dependency-violation checker that the above "itc" instructions
@@ -602,7 +604,7 @@ ENTRY(dirty_bit)
602 itc.d r18 // install updated PTE 604 itc.d r18 // install updated PTE
603#endif 605#endif
604 mov pr=r31,-1 // restore pr 606 mov pr=r31,-1 // restore pr
605 rfi 607 RFI
606END(dirty_bit) 608END(dirty_bit)
607 609
608 .org ia64_ivt+0x2400 610 .org ia64_ivt+0x2400
@@ -611,22 +613,22 @@ END(dirty_bit)
611ENTRY(iaccess_bit) 613ENTRY(iaccess_bit)
612 DBG_FAULT(9) 614 DBG_FAULT(9)
613 // Like Entry 8, except for instruction access 615 // Like Entry 8, except for instruction access
614 mov r16=cr.ifa // get the address that caused the fault 616 MOV_FROM_IFA(r16) // get the address that caused the fault
615 movl r30=1f // load continuation point in case of nested fault 617 movl r30=1f // load continuation point in case of nested fault
616 mov r31=pr // save predicates 618 mov r31=pr // save predicates
617#ifdef CONFIG_ITANIUM 619#ifdef CONFIG_ITANIUM
618 /* 620 /*
619 * Erratum 10 (IFA may contain incorrect address) has "NoFix" status. 621 * Erratum 10 (IFA may contain incorrect address) has "NoFix" status.
620 */ 622 */
621 mov r17=cr.ipsr 623 MOV_FROM_IPSR(p0, r17)
622 ;; 624 ;;
623 mov r18=cr.iip 625 MOV_FROM_IIP(r18)
624 tbit.z p6,p0=r17,IA64_PSR_IS_BIT // IA64 instruction set? 626 tbit.z p6,p0=r17,IA64_PSR_IS_BIT // IA64 instruction set?
625 ;; 627 ;;
626(p6) mov r16=r18 // if so, use cr.iip instead of cr.ifa 628(p6) mov r16=r18 // if so, use cr.iip instead of cr.ifa
627#endif /* CONFIG_ITANIUM */ 629#endif /* CONFIG_ITANIUM */
628 ;; 630 ;;
629 thash r17=r16 // compute virtual address of L3 PTE 631 THASH(p0, r17, r16, r18) // compute virtual address of L3 PTE
630 mov r29=b0 // save b0 in case of nested fault) 632 mov r29=b0 // save b0 in case of nested fault)
631#ifdef CONFIG_SMP 633#ifdef CONFIG_SMP
632 mov r28=ar.ccv // save ar.ccv 634 mov r28=ar.ccv // save ar.ccv
@@ -642,7 +644,7 @@ ENTRY(iaccess_bit)
642 ;; 644 ;;
643(p6) cmp.eq p6,p7=r26,r18 // Only if page present 645(p6) cmp.eq p6,p7=r26,r18 // Only if page present
644 ;; 646 ;;
645(p6) itc.i r25 // install updated PTE 647 ITC_I(p6, r25, r26) // install updated PTE
646 ;; 648 ;;
647 /* 649 /*
648 * Tell the assemblers dependency-violation checker that the above "itc" instructions 650 * Tell the assemblers dependency-violation checker that the above "itc" instructions
@@ -668,7 +670,7 @@ ENTRY(iaccess_bit)
668 itc.i r18 // install updated PTE 670 itc.i r18 // install updated PTE
669#endif /* !CONFIG_SMP */ 671#endif /* !CONFIG_SMP */
670 mov pr=r31,-1 672 mov pr=r31,-1
671 rfi 673 RFI
672END(iaccess_bit) 674END(iaccess_bit)
673 675
674 .org ia64_ivt+0x2800 676 .org ia64_ivt+0x2800
@@ -677,10 +679,10 @@ END(iaccess_bit)
677ENTRY(daccess_bit) 679ENTRY(daccess_bit)
678 DBG_FAULT(10) 680 DBG_FAULT(10)
679 // Like Entry 8, except for data access 681 // Like Entry 8, except for data access
680 mov r16=cr.ifa // get the address that caused the fault 682 MOV_FROM_IFA(r16) // get the address that caused the fault
681 movl r30=1f // load continuation point in case of nested fault 683 movl r30=1f // load continuation point in case of nested fault
682 ;; 684 ;;
683 thash r17=r16 // compute virtual address of L3 PTE 685 THASH(p0, r17, r16, r18) // compute virtual address of L3 PTE
684 mov r31=pr 686 mov r31=pr
685 mov r29=b0 // save b0 in case of nested fault) 687 mov r29=b0 // save b0 in case of nested fault)
686#ifdef CONFIG_SMP 688#ifdef CONFIG_SMP
@@ -697,7 +699,7 @@ ENTRY(daccess_bit)
697 ;; 699 ;;
698(p6) cmp.eq p6,p7=r26,r18 // Only if page is present 700(p6) cmp.eq p6,p7=r26,r18 // Only if page is present
699 ;; 701 ;;
700(p6) itc.d r25 // install updated PTE 702 ITC_D(p6, r25, r26) // install updated PTE
701 /* 703 /*
702 * Tell the assemblers dependency-violation checker that the above "itc" instructions 704 * Tell the assemblers dependency-violation checker that the above "itc" instructions
703 * cannot possibly affect the following loads: 705 * cannot possibly affect the following loads:
@@ -721,7 +723,7 @@ ENTRY(daccess_bit)
721#endif 723#endif
722 mov b0=r29 // restore b0 724 mov b0=r29 // restore b0
723 mov pr=r31,-1 725 mov pr=r31,-1
724 rfi 726 RFI
725END(daccess_bit) 727END(daccess_bit)
726 728
727 .org ia64_ivt+0x2c00 729 .org ia64_ivt+0x2c00
@@ -745,10 +747,10 @@ ENTRY(break_fault)
745 */ 747 */
746 DBG_FAULT(11) 748 DBG_FAULT(11)
747 mov.m r16=IA64_KR(CURRENT) // M2 r16 <- current task (12 cyc) 749 mov.m r16=IA64_KR(CURRENT) // M2 r16 <- current task (12 cyc)
748 mov r29=cr.ipsr // M2 (12 cyc) 750 MOV_FROM_IPSR(p0, r29) // M2 (12 cyc)
749 mov r31=pr // I0 (2 cyc) 751 mov r31=pr // I0 (2 cyc)
750 752
751 mov r17=cr.iim // M2 (2 cyc) 753 MOV_FROM_IIM(r17) // M2 (2 cyc)
752 mov.m r27=ar.rsc // M2 (12 cyc) 754 mov.m r27=ar.rsc // M2 (12 cyc)
753 mov r18=__IA64_BREAK_SYSCALL // A 755 mov r18=__IA64_BREAK_SYSCALL // A
754 756
@@ -767,7 +769,7 @@ ENTRY(break_fault)
767 nop.m 0 769 nop.m 0
768 movl r30=sys_call_table // X 770 movl r30=sys_call_table // X
769 771
770 mov r28=cr.iip // M2 (2 cyc) 772 MOV_FROM_IIP(r28) // M2 (2 cyc)
771 cmp.eq p0,p7=r18,r17 // I0 is this a system call? 773 cmp.eq p0,p7=r18,r17 // I0 is this a system call?
772(p7) br.cond.spnt non_syscall // B no -> 774(p7) br.cond.spnt non_syscall // B no ->
773 // 775 //
@@ -864,18 +866,17 @@ ENTRY(break_fault)
864#endif 866#endif
865 mov ar.rsc=0x3 // M2 set eager mode, pl 0, LE, loadrs=0 867 mov ar.rsc=0x3 // M2 set eager mode, pl 0, LE, loadrs=0
866 nop 0 868 nop 0
867 bsw.1 // B (6 cyc) regs are saved, switch to bank 1 869 BSW_1(r2, r14) // B (6 cyc) regs are saved, switch to bank 1
868 ;; 870 ;;
869 871
870 ssm psr.ic | PSR_DEFAULT_BITS // M2 now it's safe to re-enable intr.-collection 872 SSM_PSR_IC_AND_DEFAULT_BITS_AND_SRLZ_I(r3, r16) // M2 now it's safe to re-enable intr.-collection
873 // M0 ensure interruption collection is on
871 movl r3=ia64_ret_from_syscall // X 874 movl r3=ia64_ret_from_syscall // X
872 ;; 875 ;;
873
874 srlz.i // M0 ensure interruption collection is on
875 mov rp=r3 // I0 set the real return addr 876 mov rp=r3 // I0 set the real return addr
876(p10) br.cond.spnt.many ia64_ret_from_syscall // B return if bad call-frame or r15 is a NaT 877(p10) br.cond.spnt.many ia64_ret_from_syscall // B return if bad call-frame or r15 is a NaT
877 878
878(p15) ssm psr.i // M2 restore psr.i 879 SSM_PSR_I(p15, p15, r16) // M2 restore psr.i
879(p14) br.call.sptk.many b6=b6 // B invoke syscall-handker (ignore return addr) 880(p14) br.call.sptk.many b6=b6 // B invoke syscall-handker (ignore return addr)
880 br.cond.spnt.many ia64_trace_syscall // B do syscall-tracing thingamagic 881 br.cond.spnt.many ia64_trace_syscall // B do syscall-tracing thingamagic
881 // NOT REACHED 882 // NOT REACHED
@@ -899,16 +900,15 @@ ENTRY(interrupt)
899 mov r31=pr // prepare to save predicates 900 mov r31=pr // prepare to save predicates
900 ;; 901 ;;
901 SAVE_MIN_WITH_COVER // uses r31; defines r2 and r3 902 SAVE_MIN_WITH_COVER // uses r31; defines r2 and r3
902 ssm psr.ic | PSR_DEFAULT_BITS 903 SSM_PSR_IC_AND_DEFAULT_BITS_AND_SRLZ_I(r3, r14)
903 ;; 904 // ensure everybody knows psr.ic is back on
904 adds r3=8,r2 // set up second base pointer for SAVE_REST 905 adds r3=8,r2 // set up second base pointer for SAVE_REST
905 srlz.i // ensure everybody knows psr.ic is back on
906 ;; 906 ;;
907 SAVE_REST 907 SAVE_REST
908 ;; 908 ;;
909 MCA_RECOVER_RANGE(interrupt) 909 MCA_RECOVER_RANGE(interrupt)
910 alloc r14=ar.pfs,0,0,2,0 // must be first in an insn group 910 alloc r14=ar.pfs,0,0,2,0 // must be first in an insn group
911 mov out0=cr.ivr // pass cr.ivr as first arg 911 MOV_FROM_IVR(out0, r8) // pass cr.ivr as first arg
912 add out1=16,sp // pass pointer to pt_regs as second arg 912 add out1=16,sp // pass pointer to pt_regs as second arg
913 ;; 913 ;;
914 srlz.d // make sure we see the effect of cr.ivr 914 srlz.d // make sure we see the effect of cr.ivr
@@ -978,6 +978,7 @@ END(interrupt)
978 * - ar.fpsr: set to kernel settings 978 * - ar.fpsr: set to kernel settings
979 * - b6: preserved (same as on entry) 979 * - b6: preserved (same as on entry)
980 */ 980 */
981#ifdef __IA64_ASM_PARAVIRTUALIZED_NATIVE
981GLOBAL_ENTRY(ia64_syscall_setup) 982GLOBAL_ENTRY(ia64_syscall_setup)
982#if PT(B6) != 0 983#if PT(B6) != 0
983# error This code assumes that b6 is the first field in pt_regs. 984# error This code assumes that b6 is the first field in pt_regs.
@@ -1069,6 +1070,7 @@ GLOBAL_ENTRY(ia64_syscall_setup)
1069(p10) mov r8=-EINVAL 1070(p10) mov r8=-EINVAL
1070 br.ret.sptk.many b7 1071 br.ret.sptk.many b7
1071END(ia64_syscall_setup) 1072END(ia64_syscall_setup)
1073#endif /* __IA64_ASM_PARAVIRTUALIZED_NATIVE */
1072 1074
1073 .org ia64_ivt+0x3c00 1075 .org ia64_ivt+0x3c00
1074///////////////////////////////////////////////////////////////////////////////////////// 1076/////////////////////////////////////////////////////////////////////////////////////////
@@ -1082,7 +1084,7 @@ END(ia64_syscall_setup)
1082 DBG_FAULT(16) 1084 DBG_FAULT(16)
1083 FAULT(16) 1085 FAULT(16)
1084 1086
1085#ifdef CONFIG_VIRT_CPU_ACCOUNTING 1087#if defined(CONFIG_VIRT_CPU_ACCOUNTING) && defined(__IA64_ASM_PARAVIRTUALIZED_NATIVE)
1086 /* 1088 /*
1087 * There is no particular reason for this code to be here, other than 1089 * There is no particular reason for this code to be here, other than
1088 * that there happens to be space here that would go unused otherwise. 1090 * that there happens to be space here that would go unused otherwise.
@@ -1092,7 +1094,7 @@ END(ia64_syscall_setup)
1092 * account_sys_enter is called from SAVE_MIN* macros if accounting is 1094 * account_sys_enter is called from SAVE_MIN* macros if accounting is
1093 * enabled and if the macro is entered from user mode. 1095 * enabled and if the macro is entered from user mode.
1094 */ 1096 */
1095ENTRY(account_sys_enter) 1097GLOBAL_ENTRY(account_sys_enter)
1096 // mov.m r20=ar.itc is called in advance, and r13 is current 1098 // mov.m r20=ar.itc is called in advance, and r13 is current
1097 add r16=TI_AC_STAMP+IA64_TASK_SIZE,r13 1099 add r16=TI_AC_STAMP+IA64_TASK_SIZE,r13
1098 add r17=TI_AC_LEAVE+IA64_TASK_SIZE,r13 1100 add r17=TI_AC_LEAVE+IA64_TASK_SIZE,r13
@@ -1134,15 +1136,13 @@ ENTRY(non_syscall)
1134 // suitable spot... 1136 // suitable spot...
1135 1137
1136 alloc r14=ar.pfs,0,0,2,0 1138 alloc r14=ar.pfs,0,0,2,0
1137 mov out0=cr.iim 1139 MOV_FROM_IIM(out0)
1138 add out1=16,sp 1140 add out1=16,sp
1139 adds r3=8,r2 // set up second base pointer for SAVE_REST 1141 adds r3=8,r2 // set up second base pointer for SAVE_REST
1140 1142
1141 ssm psr.ic | PSR_DEFAULT_BITS 1143 SSM_PSR_IC_AND_DEFAULT_BITS_AND_SRLZ_I(r15, r24)
1142 ;; 1144 // guarantee that interruption collection is on
1143 srlz.i // guarantee that interruption collection is on 1145 SSM_PSR_I(p15, p15, r15) // restore psr.i
1144 ;;
1145(p15) ssm psr.i // restore psr.i
1146 movl r15=ia64_leave_kernel 1146 movl r15=ia64_leave_kernel
1147 ;; 1147 ;;
1148 SAVE_REST 1148 SAVE_REST
@@ -1168,14 +1168,12 @@ ENTRY(dispatch_unaligned_handler)
1168 SAVE_MIN_WITH_COVER 1168 SAVE_MIN_WITH_COVER
1169 ;; 1169 ;;
1170 alloc r14=ar.pfs,0,0,2,0 // now it's safe (must be first in insn group!) 1170 alloc r14=ar.pfs,0,0,2,0 // now it's safe (must be first in insn group!)
1171 mov out0=cr.ifa 1171 MOV_FROM_IFA(out0)
1172 adds out1=16,sp 1172 adds out1=16,sp
1173 1173
1174 ssm psr.ic | PSR_DEFAULT_BITS 1174 SSM_PSR_IC_AND_DEFAULT_BITS_AND_SRLZ_I(r3, r24)
1175 ;; 1175 // guarantee that interruption collection is on
1176 srlz.i // guarantee that interruption collection is on 1176 SSM_PSR_I(p15, p15, r3) // restore psr.i
1177 ;;
1178(p15) ssm psr.i // restore psr.i
1179 adds r3=8,r2 // set up second base pointer 1177 adds r3=8,r2 // set up second base pointer
1180 ;; 1178 ;;
1181 SAVE_REST 1179 SAVE_REST
@@ -1207,17 +1205,16 @@ ENTRY(dispatch_to_fault_handler)
1207 */ 1205 */
1208 SAVE_MIN_WITH_COVER_R19 1206 SAVE_MIN_WITH_COVER_R19
1209 alloc r14=ar.pfs,0,0,5,0 1207 alloc r14=ar.pfs,0,0,5,0
1210 mov out0=r15 1208 MOV_FROM_ISR(out1)
1211 mov out1=cr.isr 1209 MOV_FROM_IFA(out2)
1212 mov out2=cr.ifa 1210 MOV_FROM_IIM(out3)
1213 mov out3=cr.iim 1211 MOV_FROM_ITIR(out4)
1214 mov out4=cr.itir
1215 ;; 1212 ;;
1216 ssm psr.ic | PSR_DEFAULT_BITS 1213 SSM_PSR_IC_AND_DEFAULT_BITS_AND_SRLZ_I(r3, out0)
1217 ;; 1214 // guarantee that interruption collection is on
1218 srlz.i // guarantee that interruption collection is on 1215 mov out0=r15
1219 ;; 1216 ;;
1220(p15) ssm psr.i // restore psr.i 1217 SSM_PSR_I(p15, p15, r3) // restore psr.i
1221 adds r3=8,r2 // set up second base pointer for SAVE_REST 1218 adds r3=8,r2 // set up second base pointer for SAVE_REST
1222 ;; 1219 ;;
1223 SAVE_REST 1220 SAVE_REST
@@ -1236,8 +1233,8 @@ END(dispatch_to_fault_handler)
1236// 0x5000 Entry 20 (size 16 bundles) Page Not Present (10,22,49) 1233// 0x5000 Entry 20 (size 16 bundles) Page Not Present (10,22,49)
1237ENTRY(page_not_present) 1234ENTRY(page_not_present)
1238 DBG_FAULT(20) 1235 DBG_FAULT(20)
1239 mov r16=cr.ifa 1236 MOV_FROM_IFA(r16)
1240 rsm psr.dt 1237 RSM_PSR_DT
1241 /* 1238 /*
1242 * The Linux page fault handler doesn't expect non-present pages to be in 1239 * The Linux page fault handler doesn't expect non-present pages to be in
1243 * the TLB. Flush the existing entry now, so we meet that expectation. 1240 * the TLB. Flush the existing entry now, so we meet that expectation.
@@ -1256,8 +1253,8 @@ END(page_not_present)
1256// 0x5100 Entry 21 (size 16 bundles) Key Permission (13,25,52) 1253// 0x5100 Entry 21 (size 16 bundles) Key Permission (13,25,52)
1257ENTRY(key_permission) 1254ENTRY(key_permission)
1258 DBG_FAULT(21) 1255 DBG_FAULT(21)
1259 mov r16=cr.ifa 1256 MOV_FROM_IFA(r16)
1260 rsm psr.dt 1257 RSM_PSR_DT
1261 mov r31=pr 1258 mov r31=pr
1262 ;; 1259 ;;
1263 srlz.d 1260 srlz.d
@@ -1269,8 +1266,8 @@ END(key_permission)
1269// 0x5200 Entry 22 (size 16 bundles) Instruction Access Rights (26) 1266// 0x5200 Entry 22 (size 16 bundles) Instruction Access Rights (26)
1270ENTRY(iaccess_rights) 1267ENTRY(iaccess_rights)
1271 DBG_FAULT(22) 1268 DBG_FAULT(22)
1272 mov r16=cr.ifa 1269 MOV_FROM_IFA(r16)
1273 rsm psr.dt 1270 RSM_PSR_DT
1274 mov r31=pr 1271 mov r31=pr
1275 ;; 1272 ;;
1276 srlz.d 1273 srlz.d
@@ -1282,8 +1279,8 @@ END(iaccess_rights)
1282// 0x5300 Entry 23 (size 16 bundles) Data Access Rights (14,53) 1279// 0x5300 Entry 23 (size 16 bundles) Data Access Rights (14,53)
1283ENTRY(daccess_rights) 1280ENTRY(daccess_rights)
1284 DBG_FAULT(23) 1281 DBG_FAULT(23)
1285 mov r16=cr.ifa 1282 MOV_FROM_IFA(r16)
1286 rsm psr.dt 1283 RSM_PSR_DT
1287 mov r31=pr 1284 mov r31=pr
1288 ;; 1285 ;;
1289 srlz.d 1286 srlz.d
@@ -1295,7 +1292,7 @@ END(daccess_rights)
1295// 0x5400 Entry 24 (size 16 bundles) General Exception (5,32,34,36,38,39) 1292// 0x5400 Entry 24 (size 16 bundles) General Exception (5,32,34,36,38,39)
1296ENTRY(general_exception) 1293ENTRY(general_exception)
1297 DBG_FAULT(24) 1294 DBG_FAULT(24)
1298 mov r16=cr.isr 1295 MOV_FROM_ISR(r16)
1299 mov r31=pr 1296 mov r31=pr
1300 ;; 1297 ;;
1301 cmp4.eq p6,p0=0,r16 1298 cmp4.eq p6,p0=0,r16
@@ -1324,8 +1321,8 @@ END(disabled_fp_reg)
1324ENTRY(nat_consumption) 1321ENTRY(nat_consumption)
1325 DBG_FAULT(26) 1322 DBG_FAULT(26)
1326 1323
1327 mov r16=cr.ipsr 1324 MOV_FROM_IPSR(p0, r16)
1328 mov r17=cr.isr 1325 MOV_FROM_ISR(r17)
1329 mov r31=pr // save PR 1326 mov r31=pr // save PR
1330 ;; 1327 ;;
1331 and r18=0xf,r17 // r18 = cr.ipsr.code{3:0} 1328 and r18=0xf,r17 // r18 = cr.ipsr.code{3:0}
@@ -1335,10 +1332,10 @@ ENTRY(nat_consumption)
1335 dep r16=-1,r16,IA64_PSR_ED_BIT,1 1332 dep r16=-1,r16,IA64_PSR_ED_BIT,1
1336(p6) br.cond.spnt 1f // branch if (cr.ispr.na == 0 || cr.ipsr.code{3:0} != LFETCH) 1333(p6) br.cond.spnt 1f // branch if (cr.ispr.na == 0 || cr.ipsr.code{3:0} != LFETCH)
1337 ;; 1334 ;;
1338 mov cr.ipsr=r16 // set cr.ipsr.na 1335 MOV_TO_IPSR(p0, r16, r18)
1339 mov pr=r31,-1 1336 mov pr=r31,-1
1340 ;; 1337 ;;
1341 rfi 1338 RFI
1342 1339
13431: mov pr=r31,-1 13401: mov pr=r31,-1
1344 ;; 1341 ;;
@@ -1360,26 +1357,26 @@ ENTRY(speculation_vector)
1360 * 1357 *
1361 * cr.imm contains zero_ext(imm21) 1358 * cr.imm contains zero_ext(imm21)
1362 */ 1359 */
1363 mov r18=cr.iim 1360 MOV_FROM_IIM(r18)
1364 ;; 1361 ;;
1365 mov r17=cr.iip 1362 MOV_FROM_IIP(r17)
1366 shl r18=r18,43 // put sign bit in position (43=64-21) 1363 shl r18=r18,43 // put sign bit in position (43=64-21)
1367 ;; 1364 ;;
1368 1365
1369 mov r16=cr.ipsr 1366 MOV_FROM_IPSR(p0, r16)
1370 shr r18=r18,39 // sign extend (39=43-4) 1367 shr r18=r18,39 // sign extend (39=43-4)
1371 ;; 1368 ;;
1372 1369
1373 add r17=r17,r18 // now add the offset 1370 add r17=r17,r18 // now add the offset
1374 ;; 1371 ;;
1375 mov cr.iip=r17 1372 MOV_FROM_IIP(r17)
1376 dep r16=0,r16,41,2 // clear EI 1373 dep r16=0,r16,41,2 // clear EI
1377 ;; 1374 ;;
1378 1375
1379 mov cr.ipsr=r16 1376 MOV_FROM_IPSR(p0, r16)
1380 ;; 1377 ;;
1381 1378
1382 rfi // and go back 1379 RFI
1383END(speculation_vector) 1380END(speculation_vector)
1384 1381
1385 .org ia64_ivt+0x5800 1382 .org ia64_ivt+0x5800
@@ -1517,11 +1514,11 @@ ENTRY(ia32_intercept)
1517 DBG_FAULT(46) 1514 DBG_FAULT(46)
1518#ifdef CONFIG_IA32_SUPPORT 1515#ifdef CONFIG_IA32_SUPPORT
1519 mov r31=pr 1516 mov r31=pr
1520 mov r16=cr.isr 1517 MOV_FROM_ISR(r16)
1521 ;; 1518 ;;
1522 extr.u r17=r16,16,8 // get ISR.code 1519 extr.u r17=r16,16,8 // get ISR.code
1523 mov r18=ar.eflag 1520 mov r18=ar.eflag
1524 mov r19=cr.iim // old eflag value 1521 MOV_FROM_IIM(r19) // old eflag value
1525 ;; 1522 ;;
1526 cmp.ne p6,p0=2,r17 1523 cmp.ne p6,p0=2,r17
1527(p6) br.cond.spnt 1f // not a system flag fault 1524(p6) br.cond.spnt 1f // not a system flag fault
@@ -1533,7 +1530,7 @@ ENTRY(ia32_intercept)
1533(p6) br.cond.spnt 1f // eflags.ac bit didn't change 1530(p6) br.cond.spnt 1f // eflags.ac bit didn't change
1534 ;; 1531 ;;
1535 mov pr=r31,-1 // restore predicate registers 1532 mov pr=r31,-1 // restore predicate registers
1536 rfi 1533 RFI
1537 1534
15381: 15351:
1539#endif // CONFIG_IA32_SUPPORT 1536#endif // CONFIG_IA32_SUPPORT
@@ -1686,11 +1683,10 @@ ENTRY(dispatch_illegal_op_fault)
1686 .prologue 1683 .prologue
1687 .body 1684 .body
1688 SAVE_MIN_WITH_COVER 1685 SAVE_MIN_WITH_COVER
1689 ssm psr.ic | PSR_DEFAULT_BITS 1686 SSM_PSR_IC_AND_DEFAULT_BITS_AND_SRLZ_I(r3, r24)
1687 // guarantee that interruption collection is on
1690 ;; 1688 ;;
1691 srlz.i // guarantee that interruption collection is on 1689 SSM_PSR_I(p15, p15, r3) // restore psr.i
1692 ;;
1693(p15) ssm psr.i // restore psr.i
1694 adds r3=8,r2 // set up second base pointer for SAVE_REST 1690 adds r3=8,r2 // set up second base pointer for SAVE_REST
1695 ;; 1691 ;;
1696 alloc r14=ar.pfs,0,0,1,0 // must be first in insn group 1692 alloc r14=ar.pfs,0,0,1,0 // must be first in insn group
@@ -1729,12 +1725,11 @@ END(dispatch_illegal_op_fault)
1729ENTRY(dispatch_to_ia32_handler) 1725ENTRY(dispatch_to_ia32_handler)
1730 SAVE_MIN 1726 SAVE_MIN
1731 ;; 1727 ;;
1732 mov r14=cr.isr 1728 MOV_FROM_ISR(r14)
1733 ssm psr.ic | PSR_DEFAULT_BITS 1729 SSM_PSR_IC_AND_DEFAULT_BITS_AND_SRLZ_I(r3, r24)
1734 ;; 1730 // guarantee that interruption collection is on
1735 srlz.i // guarantee that interruption collection is on
1736 ;; 1731 ;;
1737(p15) ssm psr.i 1732 SSM_PSR_I(p15, p15, r3)
1738 adds r3=8,r2 // Base pointer for SAVE_REST 1733 adds r3=8,r2 // Base pointer for SAVE_REST
1739 ;; 1734 ;;
1740 SAVE_REST 1735 SAVE_REST