123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380 |
- /*
- * Floating-point, VMX/Altivec and VSX loads and stores
- * for use in instruction emulation.
- *
- * Copyright 2010 Paul Mackerras, IBM Corp. <paulus@au1.ibm.com>
- *
- * This program is free software; you can redistribute it and/or
- * modify it under the terms of the GNU General Public License
- * as published by the Free Software Foundation; either version
- * 2 of the License, or (at your option) any later version.
- */
- #include <asm/processor.h>
- #include <asm/ppc_asm.h>
- #include <asm/ppc-opcode.h>
- #include <asm/reg.h>
- #include <asm/asm-offsets.h>
- #include <linux/errno.h>
- #ifdef CONFIG_PPC_FPU
- #define STKFRM (PPC_MIN_STKFRM + 16)
- .macro extab instr,handler
- .section __ex_table,"a"
- PPC_LONG \instr,\handler
- .previous
- .endm
- .macro inst32 op
- reg = 0
- .rept 32
- 20: \op reg,0,r4
- b 3f
- extab 20b,99f
- reg = reg + 1
- .endr
- .endm
- /* Get the contents of frN into fr0; N is in r3. */
- _GLOBAL(get_fpr)
- mflr r0
- rlwinm r3,r3,3,0xf8
- bcl 20,31,1f
- blr /* fr0 is already in fr0 */
- nop
- reg = 1
- .rept 31
- fmr fr0,reg
- blr
- reg = reg + 1
- .endr
- 1: mflr r5
- add r5,r3,r5
- mtctr r5
- mtlr r0
- bctr
- /* Put the contents of fr0 into frN; N is in r3. */
- _GLOBAL(put_fpr)
- mflr r0
- rlwinm r3,r3,3,0xf8
- bcl 20,31,1f
- blr /* fr0 is already in fr0 */
- nop
- reg = 1
- .rept 31
- fmr reg,fr0
- blr
- reg = reg + 1
- .endr
- 1: mflr r5
- add r5,r3,r5
- mtctr r5
- mtlr r0
- bctr
- /* Load FP reg N from float at *p. N is in r3, p in r4. */
- _GLOBAL(do_lfs)
- PPC_STLU r1,-STKFRM(r1)
- mflr r0
- PPC_STL r0,STKFRM+PPC_LR_STKOFF(r1)
- mfmsr r6
- ori r7,r6,MSR_FP
- cmpwi cr7,r3,0
- MTMSRD(r7)
- isync
- beq cr7,1f
- stfd fr0,STKFRM-16(r1)
- 1: li r9,-EFAULT
- 2: lfs fr0,0(r4)
- li r9,0
- 3: bl put_fpr
- beq cr7,4f
- lfd fr0,STKFRM-16(r1)
- 4: PPC_LL r0,STKFRM+PPC_LR_STKOFF(r1)
- mtlr r0
- MTMSRD(r6)
- isync
- mr r3,r9
- addi r1,r1,STKFRM
- blr
- extab 2b,3b
- /* Load FP reg N from double at *p. N is in r3, p in r4. */
- _GLOBAL(do_lfd)
- PPC_STLU r1,-STKFRM(r1)
- mflr r0
- PPC_STL r0,STKFRM+PPC_LR_STKOFF(r1)
- mfmsr r6
- ori r7,r6,MSR_FP
- cmpwi cr7,r3,0
- MTMSRD(r7)
- isync
- beq cr7,1f
- stfd fr0,STKFRM-16(r1)
- 1: li r9,-EFAULT
- 2: lfd fr0,0(r4)
- li r9,0
- 3: beq cr7,4f
- bl put_fpr
- lfd fr0,STKFRM-16(r1)
- 4: PPC_LL r0,STKFRM+PPC_LR_STKOFF(r1)
- mtlr r0
- MTMSRD(r6)
- isync
- mr r3,r9
- addi r1,r1,STKFRM
- blr
- extab 2b,3b
- /* Store FP reg N to float at *p. N is in r3, p in r4. */
- _GLOBAL(do_stfs)
- PPC_STLU r1,-STKFRM(r1)
- mflr r0
- PPC_STL r0,STKFRM+PPC_LR_STKOFF(r1)
- mfmsr r6
- ori r7,r6,MSR_FP
- cmpwi cr7,r3,0
- MTMSRD(r7)
- isync
- beq cr7,1f
- stfd fr0,STKFRM-16(r1)
- bl get_fpr
- 1: li r9,-EFAULT
- 2: stfs fr0,0(r4)
- li r9,0
- 3: beq cr7,4f
- lfd fr0,STKFRM-16(r1)
- 4: PPC_LL r0,STKFRM+PPC_LR_STKOFF(r1)
- mtlr r0
- MTMSRD(r6)
- isync
- mr r3,r9
- addi r1,r1,STKFRM
- blr
- extab 2b,3b
- /* Store FP reg N to double at *p. N is in r3, p in r4. */
- _GLOBAL(do_stfd)
- PPC_STLU r1,-STKFRM(r1)
- mflr r0
- PPC_STL r0,STKFRM+PPC_LR_STKOFF(r1)
- mfmsr r6
- ori r7,r6,MSR_FP
- cmpwi cr7,r3,0
- MTMSRD(r7)
- isync
- beq cr7,1f
- stfd fr0,STKFRM-16(r1)
- bl get_fpr
- 1: li r9,-EFAULT
- 2: stfd fr0,0(r4)
- li r9,0
- 3: beq cr7,4f
- lfd fr0,STKFRM-16(r1)
- 4: PPC_LL r0,STKFRM+PPC_LR_STKOFF(r1)
- mtlr r0
- MTMSRD(r6)
- isync
- mr r3,r9
- addi r1,r1,STKFRM
- blr
- extab 2b,3b
- #ifdef CONFIG_ALTIVEC
- /* Get the contents of vrN into v0; N is in r3. */
- _GLOBAL(get_vr)
- mflr r0
- rlwinm r3,r3,3,0xf8
- bcl 20,31,1f
- blr /* v0 is already in v0 */
- nop
- reg = 1
- .rept 31
- vor v0,reg,reg /* assembler doesn't know vmr? */
- blr
- reg = reg + 1
- .endr
- 1: mflr r5
- add r5,r3,r5
- mtctr r5
- mtlr r0
- bctr
- /* Put the contents of v0 into vrN; N is in r3. */
- _GLOBAL(put_vr)
- mflr r0
- rlwinm r3,r3,3,0xf8
- bcl 20,31,1f
- blr /* v0 is already in v0 */
- nop
- reg = 1
- .rept 31
- vor reg,v0,v0
- blr
- reg = reg + 1
- .endr
- 1: mflr r5
- add r5,r3,r5
- mtctr r5
- mtlr r0
- bctr
- /* Load vector reg N from *p. N is in r3, p in r4. */
- _GLOBAL(do_lvx)
- PPC_STLU r1,-STKFRM(r1)
- mflr r0
- PPC_STL r0,STKFRM+PPC_LR_STKOFF(r1)
- mfmsr r6
- oris r7,r6,MSR_VEC@h
- cmpwi cr7,r3,0
- li r8,STKFRM-16
- MTMSRD(r7)
- isync
- beq cr7,1f
- stvx v0,r1,r8
- 1: li r9,-EFAULT
- 2: lvx v0,0,r4
- li r9,0
- 3: beq cr7,4f
- bl put_vr
- lvx v0,r1,r8
- 4: PPC_LL r0,STKFRM+PPC_LR_STKOFF(r1)
- mtlr r0
- MTMSRD(r6)
- isync
- mr r3,r9
- addi r1,r1,STKFRM
- blr
- extab 2b,3b
- /* Store vector reg N to *p. N is in r3, p in r4. */
- _GLOBAL(do_stvx)
- PPC_STLU r1,-STKFRM(r1)
- mflr r0
- PPC_STL r0,STKFRM+PPC_LR_STKOFF(r1)
- mfmsr r6
- oris r7,r6,MSR_VEC@h
- cmpwi cr7,r3,0
- li r8,STKFRM-16
- MTMSRD(r7)
- isync
- beq cr7,1f
- stvx v0,r1,r8
- bl get_vr
- 1: li r9,-EFAULT
- 2: stvx v0,0,r4
- li r9,0
- 3: beq cr7,4f
- lvx v0,r1,r8
- 4: PPC_LL r0,STKFRM+PPC_LR_STKOFF(r1)
- mtlr r0
- MTMSRD(r6)
- isync
- mr r3,r9
- addi r1,r1,STKFRM
- blr
- extab 2b,3b
- #endif /* CONFIG_ALTIVEC */
- #ifdef CONFIG_VSX
- /* Get the contents of vsN into vs0; N is in r3. */
- _GLOBAL(get_vsr)
- mflr r0
- rlwinm r3,r3,3,0x1f8
- bcl 20,31,1f
- blr /* vs0 is already in vs0 */
- nop
- reg = 1
- .rept 63
- XXLOR(0,reg,reg)
- blr
- reg = reg + 1
- .endr
- 1: mflr r5
- add r5,r3,r5
- mtctr r5
- mtlr r0
- bctr
- /* Put the contents of vs0 into vsN; N is in r3. */
- _GLOBAL(put_vsr)
- mflr r0
- rlwinm r3,r3,3,0x1f8
- bcl 20,31,1f
- blr /* v0 is already in v0 */
- nop
- reg = 1
- .rept 63
- XXLOR(reg,0,0)
- blr
- reg = reg + 1
- .endr
- 1: mflr r5
- add r5,r3,r5
- mtctr r5
- mtlr r0
- bctr
- /* Load VSX reg N from vector doubleword *p. N is in r3, p in r4. */
- _GLOBAL(do_lxvd2x)
- PPC_STLU r1,-STKFRM(r1)
- mflr r0
- PPC_STL r0,STKFRM+PPC_LR_STKOFF(r1)
- mfmsr r6
- oris r7,r6,MSR_VSX@h
- cmpwi cr7,r3,0
- li r8,STKFRM-16
- MTMSRD(r7)
- isync
- beq cr7,1f
- STXVD2X(0,R1,R8)
- 1: li r9,-EFAULT
- 2: LXVD2X(0,R0,R4)
- li r9,0
- 3: beq cr7,4f
- bl put_vsr
- LXVD2X(0,R1,R8)
- 4: PPC_LL r0,STKFRM+PPC_LR_STKOFF(r1)
- mtlr r0
- MTMSRD(r6)
- isync
- mr r3,r9
- addi r1,r1,STKFRM
- blr
- extab 2b,3b
- /* Store VSX reg N to vector doubleword *p. N is in r3, p in r4. */
- _GLOBAL(do_stxvd2x)
- PPC_STLU r1,-STKFRM(r1)
- mflr r0
- PPC_STL r0,STKFRM+PPC_LR_STKOFF(r1)
- mfmsr r6
- oris r7,r6,MSR_VSX@h
- cmpwi cr7,r3,0
- li r8,STKFRM-16
- MTMSRD(r7)
- isync
- beq cr7,1f
- STXVD2X(0,R1,R8)
- bl get_vsr
- 1: li r9,-EFAULT
- 2: STXVD2X(0,R0,R4)
- li r9,0
- 3: beq cr7,4f
- LXVD2X(0,R1,R8)
- 4: PPC_LL r0,STKFRM+PPC_LR_STKOFF(r1)
- mtlr r0
- MTMSRD(r6)
- isync
- mr r3,r9
- addi r1,r1,STKFRM
- blr
- extab 2b,3b
- #endif /* CONFIG_VSX */
- #endif /* CONFIG_PPC_FPU */
|