2019-05-29 07:12:40 -07:00
/* SPDX-License-Identifier: GPL-2.0-only */
2008-04-16 23:28:09 -05:00
/ *
*
* Copyright I B M C o r p . 2 0 0 7
2011-06-14 18:34:31 -05:00
* Copyright 2 0 1 1 F r e e s c a l e S e m i c o n d u c t o r , I n c .
2008-04-16 23:28:09 -05:00
*
* Authors : Hollis B l a n c h a r d < h o l l i s b @us.ibm.com>
* /
# include < a s m / p p c _ a s m . h >
# include < a s m / k v m _ a s m . h >
# include < a s m / r e g . h >
# include < a s m / p a g e . h >
# include < a s m / a s m - o f f s e t s . h >
/* The host stack layout: */
# define H O S T _ R 1 0 / * I m p l i e d b y s t w u . * /
# define H O S T _ C A L L E E _ L R 4
# define H O S T _ R U N 8
/ * r2 i s s p e c i a l : i t h o l d s ' c u r r e n t ' , a n d i t m a d e n o n v o l a t i l e i n t h e
* kernel w i t h t h e - f f i x e d - r2 g c c o p t i o n . * /
# define H O S T _ R 2 1 2
2012-03-05 16:00:28 +01:00
# define H O S T _ C R 1 6
# define H O S T _ N V _ G P R S 2 0
2012-06-25 13:33:23 +00:00
# define _ _ H O S T _ N V _ G P R ( n ) ( H O S T _ N V _ G P R S + ( ( n - 1 4 ) * 4 ) )
# define H O S T _ N V _ G P R ( n ) _ _ H O S T _ N V _ G P R ( _ _ R E G _ ## n )
2012-06-25 13:33:10 +00:00
# define H O S T _ M I N _ S T A C K _ S I Z E ( H O S T _ N V _ G P R ( R 3 1 ) + 4 )
2008-04-16 23:28:09 -05:00
# define H O S T _ S T A C K _ S I Z E ( ( ( H O S T _ M I N _ S T A C K _ S I Z E + 1 5 ) / 1 6 ) * 1 6 ) / * A l i g n . * /
# define H O S T _ S T A C K _ L R ( H O S T _ S T A C K _ S I Z E + 4 ) / * I n c a l l e r s t a c k f r a m e . * /
# define N E E D _ I N S T _ M A S K ( ( 1 < < B O O K E _ I N T E R R U P T _ P R O G R A M ) | \
2008-07-25 13:54:49 -05:00
( 1 < < BOOKE_ I N T E R R U P T _ D T L B _ M I S S ) | \
( 1 < < BOOKE_ I N T E R R U P T _ D E B U G ) )
2008-04-16 23:28:09 -05:00
# define N E E D _ D E A R _ M A S K ( ( 1 < < B O O K E _ I N T E R R U P T _ D A T A _ S T O R A G E ) | \
2013-01-31 14:17:38 +01:00
( 1 < < BOOKE_ I N T E R R U P T _ D T L B _ M I S S ) | \
( 1 < < BOOKE_ I N T E R R U P T _ A L I G N M E N T ) )
2008-04-16 23:28:09 -05:00
# define N E E D _ E S R _ M A S K ( ( 1 < < B O O K E _ I N T E R R U P T _ D A T A _ S T O R A G E ) | \
( 1 < < BOOKE_ I N T E R R U P T _ I N S T _ S T O R A G E ) | \
( 1 < < BOOKE_ I N T E R R U P T _ P R O G R A M ) | \
2013-01-31 14:17:38 +01:00
( 1 < < BOOKE_ I N T E R R U P T _ D T L B _ M I S S ) | \
( 1 < < BOOKE_ I N T E R R U P T _ A L I G N M E N T ) )
2008-04-16 23:28:09 -05:00
2013-02-27 18:13:10 +00:00
.macro __KVM_HANDLER ivor_ n r s c r a t c h s r r0
2008-04-16 23:28:09 -05:00
/* Get pointer to vcpu and record exit number. */
2012-06-20 05:56:54 +00:00
mtspr \ s c r a t c h , r4
2013-01-15 22:20:42 +00:00
mfspr r4 , S P R N _ S P R G _ T H R E A D
lwz r4 , T H R E A D _ K V M _ V C P U ( r4 )
2012-07-24 12:01:20 -07:00
stw r3 , V C P U _ G P R ( R 3 ) ( r4 )
2012-06-25 13:33:10 +00:00
stw r5 , V C P U _ G P R ( R 5 ) ( r4 )
stw r6 , V C P U _ G P R ( R 6 ) ( r4 )
2012-06-20 05:56:54 +00:00
mfspr r3 , \ s c r a t c h
2008-04-16 23:28:09 -05:00
mfctr r5
2012-07-24 12:01:20 -07:00
stw r3 , V C P U _ G P R ( R 4 ) ( r4 )
2008-04-16 23:28:09 -05:00
stw r5 , V C P U _ C T R ( r4 )
2012-06-20 05:56:54 +00:00
mfspr r3 , \ s r r0
lis r6 , k v m p p c _ r e s u m e _ h o s t @h
stw r3 , V C P U _ P C ( r4 )
2008-04-16 23:28:09 -05:00
li r5 , \ i v o r _ n r
ori r6 , r6 , k v m p p c _ r e s u m e _ h o s t @l
mtctr r6
bctr
.endm
2013-02-27 18:13:10 +00:00
.macro KVM_HANDLER ivor_ n r s c r a t c h s r r0
_ GLOBAL( k v m p p c _ h a n d l e r _ \ i v o r _ n r )
_ _ KVM_ H A N D L E R \ i v o r _ n r \ s c r a t c h \ s r r0
.endm
.macro KVM_DBG_HANDLER ivor_ n r s c r a t c h s r r0
_ GLOBAL( k v m p p c _ h a n d l e r _ \ i v o r _ n r )
mtspr \ s c r a t c h , r4
mfspr r4 , S P R N _ S P R G _ T H R E A D
lwz r4 , T H R E A D _ K V M _ V C P U ( r4 )
stw r3 , V C P U _ C R I T _ S A V E ( r4 )
mfcr r3
mfspr r4 , S P R N _ C S R R 1
andi. r4 , r4 , M S R _ P R
bne 1 f
/* debug interrupt happened in enter/exit path */
mfspr r4 , S P R N _ C S R R 1
rlwinm r4 , r4 , 0 , ~ M S R _ D E
mtspr S P R N _ C S R R 1 , r4
lis r4 , 0 x f f f f
ori r4 , r4 , 0 x f f f f
mtspr S P R N _ D B S R , r4
mfspr r4 , S P R N _ S P R G _ T H R E A D
lwz r4 , T H R E A D _ K V M _ V C P U ( r4 )
mtcr r3
lwz r3 , V C P U _ C R I T _ S A V E ( r4 )
mfspr r4 , \ s c r a t c h
rfci
1 : /* debug interrupt happened in guest */
mtcr r3
mfspr r4 , S P R N _ S P R G _ T H R E A D
lwz r4 , T H R E A D _ K V M _ V C P U ( r4 )
lwz r3 , V C P U _ C R I T _ S A V E ( r4 )
mfspr r4 , \ s c r a t c h
_ _ KVM_ H A N D L E R \ i v o r _ n r \ s c r a t c h \ s r r0
.endm
2013-01-15 22:24:39 +00:00
.macro KVM_HANDLER_ADDR ivor_ n r
.long kvmppc_ h a n d l e r _ \ i v o r _ n r
.endm
.macro KVM_HANDLER_END
.long kvmppc_handlers_end
.endm
2008-04-16 23:28:09 -05:00
_ GLOBAL( k v m p p c _ h a n d l e r s _ s t a r t )
2012-06-20 05:56:54 +00:00
KVM_ H A N D L E R B O O K E _ I N T E R R U P T _ C R I T I C A L S P R N _ S P R G _ R S C R A T C H _ C R I T S P R N _ C S R R 0
KVM_ H A N D L E R B O O K E _ I N T E R R U P T _ M A C H I N E _ C H E C K S P R N _ S P R G _ R S C R A T C H _ M C S P R N _ M C S R R 0
KVM_ H A N D L E R B O O K E _ I N T E R R U P T _ D A T A _ S T O R A G E S P R N _ S P R G _ R S C R A T C H 0 S P R N _ S R R 0
KVM_ H A N D L E R B O O K E _ I N T E R R U P T _ I N S T _ S T O R A G E S P R N _ S P R G _ R S C R A T C H 0 S P R N _ S R R 0
KVM_ H A N D L E R B O O K E _ I N T E R R U P T _ E X T E R N A L S P R N _ S P R G _ R S C R A T C H 0 S P R N _ S R R 0
KVM_ H A N D L E R B O O K E _ I N T E R R U P T _ A L I G N M E N T S P R N _ S P R G _ R S C R A T C H 0 S P R N _ S R R 0
KVM_ H A N D L E R B O O K E _ I N T E R R U P T _ P R O G R A M S P R N _ S P R G _ R S C R A T C H 0 S P R N _ S R R 0
KVM_ H A N D L E R B O O K E _ I N T E R R U P T _ F P _ U N A V A I L S P R N _ S P R G _ R S C R A T C H 0 S P R N _ S R R 0
KVM_ H A N D L E R B O O K E _ I N T E R R U P T _ S Y S C A L L S P R N _ S P R G _ R S C R A T C H 0 S P R N _ S R R 0
KVM_ H A N D L E R B O O K E _ I N T E R R U P T _ A P _ U N A V A I L S P R N _ S P R G _ R S C R A T C H 0 S P R N _ S R R 0
KVM_ H A N D L E R B O O K E _ I N T E R R U P T _ D E C R E M E N T E R S P R N _ S P R G _ R S C R A T C H 0 S P R N _ S R R 0
KVM_ H A N D L E R B O O K E _ I N T E R R U P T _ F I T S P R N _ S P R G _ R S C R A T C H 0 S P R N _ S R R 0
KVM_ H A N D L E R B O O K E _ I N T E R R U P T _ W A T C H D O G S P R N _ S P R G _ R S C R A T C H _ C R I T S P R N _ C S R R 0
KVM_ H A N D L E R B O O K E _ I N T E R R U P T _ D T L B _ M I S S S P R N _ S P R G _ R S C R A T C H 0 S P R N _ S R R 0
KVM_ H A N D L E R B O O K E _ I N T E R R U P T _ I T L B _ M I S S S P R N _ S P R G _ R S C R A T C H 0 S P R N _ S R R 0
2013-02-27 18:13:10 +00:00
KVM_ D B G _ H A N D L E R B O O K E _ I N T E R R U P T _ D E B U G S P R N _ S P R G _ R S C R A T C H _ C R I T S P R N _ C S R R 0
2012-06-20 05:56:54 +00:00
KVM_ H A N D L E R B O O K E _ I N T E R R U P T _ S P E _ U N A V A I L S P R N _ S P R G _ R S C R A T C H 0 S P R N _ S R R 0
KVM_ H A N D L E R B O O K E _ I N T E R R U P T _ S P E _ F P _ D A T A S P R N _ S P R G _ R S C R A T C H 0 S P R N _ S R R 0
KVM_ H A N D L E R B O O K E _ I N T E R R U P T _ S P E _ F P _ R O U N D S P R N _ S P R G _ R S C R A T C H 0 S P R N _ S R R 0
2013-01-15 22:24:39 +00:00
_ GLOBAL( k v m p p c _ h a n d l e r s _ e n d )
2008-04-16 23:28:09 -05:00
/ * Registers :
2009-07-14 20:52:54 +00:00
* SPRG_SCRATCH0 : guest r4
2008-04-16 23:28:09 -05:00
* r4 : vcpu p o i n t e r
* r5 : KVM e x i t n u m b e r
* /
_ GLOBAL( k v m p p c _ r e s u m e _ h o s t )
mfcr r3
stw r3 , V C P U _ C R ( r4 )
2012-06-25 13:33:10 +00:00
stw r7 , V C P U _ G P R ( R 7 ) ( r4 )
stw r8 , V C P U _ G P R ( R 8 ) ( r4 )
stw r9 , V C P U _ G P R ( R 9 ) ( r4 )
2008-04-16 23:28:09 -05:00
li r6 , 1
slw r6 , r6 , r5
2008-12-02 15:51:57 -06:00
# ifdef C O N F I G _ K V M _ E X I T _ T I M I N G
/* save exit time */
1 :
mfspr r7 , S P R N _ T B R U
mfspr r8 , S P R N _ T B R L
mfspr r9 , S P R N _ T B R U
cmpw r9 , r7
bne 1 b
stw r8 , V C P U _ T I M I N G _ E X I T _ T B L ( r4 )
stw r9 , V C P U _ T I M I N G _ E X I T _ T B U ( r4 )
# endif
2008-04-16 23:28:09 -05:00
/* Save the faulting instruction and all GPRs for emulation. */
andi. r7 , r6 , N E E D _ I N S T _ M A S K
beq . . s k i p _ i n s t _ c o p y
mfspr r9 , S P R N _ S R R 0
mfmsr r8
ori r7 , r8 , M S R _ D S
mtmsr r7
isync
lwz r9 , 0 ( r9 )
mtmsr r8
isync
stw r9 , V C P U _ L A S T _ I N S T ( r4 )
2012-06-25 13:33:10 +00:00
stw r15 , V C P U _ G P R ( R 1 5 ) ( r4 )
stw r16 , V C P U _ G P R ( R 1 6 ) ( r4 )
stw r17 , V C P U _ G P R ( R 1 7 ) ( r4 )
stw r18 , V C P U _ G P R ( R 1 8 ) ( r4 )
stw r19 , V C P U _ G P R ( R 1 9 ) ( r4 )
stw r20 , V C P U _ G P R ( R 2 0 ) ( r4 )
stw r21 , V C P U _ G P R ( R 2 1 ) ( r4 )
stw r22 , V C P U _ G P R ( R 2 2 ) ( r4 )
stw r23 , V C P U _ G P R ( R 2 3 ) ( r4 )
stw r24 , V C P U _ G P R ( R 2 4 ) ( r4 )
stw r25 , V C P U _ G P R ( R 2 5 ) ( r4 )
stw r26 , V C P U _ G P R ( R 2 6 ) ( r4 )
stw r27 , V C P U _ G P R ( R 2 7 ) ( r4 )
stw r28 , V C P U _ G P R ( R 2 8 ) ( r4 )
stw r29 , V C P U _ G P R ( R 2 9 ) ( r4 )
stw r30 , V C P U _ G P R ( R 3 0 ) ( r4 )
stw r31 , V C P U _ G P R ( R 3 1 ) ( r4 )
2008-04-16 23:28:09 -05:00
. .skip_inst_copy :
/* Also grab DEAR and ESR before the host can clobber them. */
andi. r7 , r6 , N E E D _ D E A R _ M A S K
beq . . s k i p _ d e a r
mfspr r9 , S P R N _ D E A R
stw r9 , V C P U _ F A U L T _ D E A R ( r4 )
. .skip_dear :
andi. r7 , r6 , N E E D _ E S R _ M A S K
beq . . s k i p _ e s r
mfspr r9 , S P R N _ E S R
stw r9 , V C P U _ F A U L T _ E S R ( r4 )
. .skip_esr :
/* Save remaining volatile guest register state to vcpu. */
2012-06-25 13:33:10 +00:00
stw r0 , V C P U _ G P R ( R 0 ) ( r4 )
stw r1 , V C P U _ G P R ( R 1 ) ( r4 )
stw r2 , V C P U _ G P R ( R 2 ) ( r4 )
stw r10 , V C P U _ G P R ( R 1 0 ) ( r4 )
stw r11 , V C P U _ G P R ( R 1 1 ) ( r4 )
stw r12 , V C P U _ G P R ( R 1 2 ) ( r4 )
stw r13 , V C P U _ G P R ( R 1 3 ) ( r4 )
stw r14 , V C P U _ G P R ( R 1 4 ) ( r4 ) / * W e n e e d a N V G P R b e l o w . * /
2008-04-16 23:28:09 -05:00
mflr r3
stw r3 , V C P U _ L R ( r4 )
mfxer r3
stw r3 , V C P U _ X E R ( r4 )
/ * Restore h o s t s t a c k p o i n t e r a n d P I D b e f o r e I V P R , s i n c e t h e h o s t
* exception h a n d l e r s u s e t h e m . * /
lwz r1 , V C P U _ H O S T _ S T A C K ( r4 )
lwz r3 , V C P U _ H O S T _ P I D ( r4 )
mtspr S P R N _ P I D , r3
2022-09-19 19:01:31 +02:00
# ifdef C O N F I G _ P P C _ 8 5 x x
2011-06-14 18:35:14 -05:00
/* we cheat and know that Linux doesn't use PID1 which is always 0 */
lis r3 , 0
mtspr S P R N _ P I D 1 , r3
# endif
2008-04-16 23:28:09 -05:00
/ * Restore h o s t I V P R b e f o r e r e - e n a b l i n g i n t e r r u p t s . W e c h e a t a n d k n o w
* that L i n u x I V P R i s a l w a y s 0 x c00 0 0 0 0 0 . * /
lis r3 , 0 x c00 0
mtspr S P R N _ I V P R , r3
/* Switch to kernel stack and jump to handler. */
LOAD_ R E G _ A D D R ( r3 , k v m p p c _ h a n d l e _ e x i t )
mtctr r3
2020-06-23 21:14:16 +08:00
mr r3 , r4
2008-04-16 23:28:09 -05:00
lwz r2 , H O S T _ R 2 ( r1 )
mr r14 , r4 / * S a v e v c p u p o i n t e r . * /
bctrl / * k v m p p c _ h a n d l e _ e x i t ( ) * /
/* Restore vcpu pointer and the nonvolatiles we used. */
mr r4 , r14
2012-06-25 13:33:10 +00:00
lwz r14 , V C P U _ G P R ( R 1 4 ) ( r4 )
2008-04-16 23:28:09 -05:00
/* Sometimes instruction emulation must restore complete GPR state. */
andi. r5 , r3 , R E S U M E _ F L A G _ N V
beq . . s k i p _ n v _ l o a d
2012-06-25 13:33:10 +00:00
lwz r15 , V C P U _ G P R ( R 1 5 ) ( r4 )
lwz r16 , V C P U _ G P R ( R 1 6 ) ( r4 )
lwz r17 , V C P U _ G P R ( R 1 7 ) ( r4 )
lwz r18 , V C P U _ G P R ( R 1 8 ) ( r4 )
lwz r19 , V C P U _ G P R ( R 1 9 ) ( r4 )
lwz r20 , V C P U _ G P R ( R 2 0 ) ( r4 )
lwz r21 , V C P U _ G P R ( R 2 1 ) ( r4 )
lwz r22 , V C P U _ G P R ( R 2 2 ) ( r4 )
lwz r23 , V C P U _ G P R ( R 2 3 ) ( r4 )
lwz r24 , V C P U _ G P R ( R 2 4 ) ( r4 )
lwz r25 , V C P U _ G P R ( R 2 5 ) ( r4 )
lwz r26 , V C P U _ G P R ( R 2 6 ) ( r4 )
lwz r27 , V C P U _ G P R ( R 2 7 ) ( r4 )
lwz r28 , V C P U _ G P R ( R 2 8 ) ( r4 )
lwz r29 , V C P U _ G P R ( R 2 9 ) ( r4 )
lwz r30 , V C P U _ G P R ( R 3 0 ) ( r4 )
lwz r31 , V C P U _ G P R ( R 3 1 ) ( r4 )
2008-04-16 23:28:09 -05:00
. .skip_nv_load :
/* Should we return to the guest? */
andi. r5 , r3 , R E S U M E _ F L A G _ H O S T
beq l i g h t w e i g h t _ e x i t
srawi r3 , r3 , 2 / * S h i f t - E R R b a c k d o w n . * /
heavyweight_exit :
/* Not returning to guest. */
2011-06-14 18:34:31 -05:00
# ifdef C O N F I G _ S P E
/* save guest SPEFSCR and load host SPEFSCR */
mfspr r9 , S P R N _ S P E F S C R
stw r9 , V C P U _ S P E F S C R ( r4 )
lwz r9 , V C P U _ H O S T _ S P E F S C R ( r4 )
mtspr S P R N _ S P E F S C R , r9
# endif
2008-04-16 23:28:09 -05:00
/ * We a l r e a d y s a v e d g u e s t v o l a t i l e r e g i s t e r s t a t e ; now save the
* non- v o l a t i l e s . * /
2012-06-25 13:33:10 +00:00
stw r15 , V C P U _ G P R ( R 1 5 ) ( r4 )
stw r16 , V C P U _ G P R ( R 1 6 ) ( r4 )
stw r17 , V C P U _ G P R ( R 1 7 ) ( r4 )
stw r18 , V C P U _ G P R ( R 1 8 ) ( r4 )
stw r19 , V C P U _ G P R ( R 1 9 ) ( r4 )
stw r20 , V C P U _ G P R ( R 2 0 ) ( r4 )
stw r21 , V C P U _ G P R ( R 2 1 ) ( r4 )
stw r22 , V C P U _ G P R ( R 2 2 ) ( r4 )
stw r23 , V C P U _ G P R ( R 2 3 ) ( r4 )
stw r24 , V C P U _ G P R ( R 2 4 ) ( r4 )
stw r25 , V C P U _ G P R ( R 2 5 ) ( r4 )
stw r26 , V C P U _ G P R ( R 2 6 ) ( r4 )
stw r27 , V C P U _ G P R ( R 2 7 ) ( r4 )
stw r28 , V C P U _ G P R ( R 2 8 ) ( r4 )
stw r29 , V C P U _ G P R ( R 2 9 ) ( r4 )
stw r30 , V C P U _ G P R ( R 3 0 ) ( r4 )
stw r31 , V C P U _ G P R ( R 3 1 ) ( r4 )
2008-04-16 23:28:09 -05:00
/* Load host non-volatile register state from host stack. */
2012-06-25 13:33:10 +00:00
lwz r14 , H O S T _ N V _ G P R ( R 1 4 ) ( r1 )
lwz r15 , H O S T _ N V _ G P R ( R 1 5 ) ( r1 )
lwz r16 , H O S T _ N V _ G P R ( R 1 6 ) ( r1 )
lwz r17 , H O S T _ N V _ G P R ( R 1 7 ) ( r1 )
lwz r18 , H O S T _ N V _ G P R ( R 1 8 ) ( r1 )
lwz r19 , H O S T _ N V _ G P R ( R 1 9 ) ( r1 )
lwz r20 , H O S T _ N V _ G P R ( R 2 0 ) ( r1 )
lwz r21 , H O S T _ N V _ G P R ( R 2 1 ) ( r1 )
lwz r22 , H O S T _ N V _ G P R ( R 2 2 ) ( r1 )
lwz r23 , H O S T _ N V _ G P R ( R 2 3 ) ( r1 )
lwz r24 , H O S T _ N V _ G P R ( R 2 4 ) ( r1 )
lwz r25 , H O S T _ N V _ G P R ( R 2 5 ) ( r1 )
lwz r26 , H O S T _ N V _ G P R ( R 2 6 ) ( r1 )
lwz r27 , H O S T _ N V _ G P R ( R 2 7 ) ( r1 )
lwz r28 , H O S T _ N V _ G P R ( R 2 8 ) ( r1 )
lwz r29 , H O S T _ N V _ G P R ( R 2 9 ) ( r1 )
lwz r30 , H O S T _ N V _ G P R ( R 3 0 ) ( r1 )
lwz r31 , H O S T _ N V _ G P R ( R 3 1 ) ( r1 )
2008-04-16 23:28:09 -05:00
/* Return to kvm_vcpu_run(). */
lwz r4 , H O S T _ S T A C K _ L R ( r1 )
2012-03-05 16:00:28 +01:00
lwz r5 , H O S T _ C R ( r1 )
2008-04-16 23:28:09 -05:00
addi r1 , r1 , H O S T _ S T A C K _ S I Z E
mtlr r4
2012-03-05 16:00:28 +01:00
mtcr r5
2008-04-16 23:28:09 -05:00
/* r3 still contains the return code from kvmppc_handle_exit(). */
blr
/ * Registers :
2020-06-23 21:14:16 +08:00
* r3 : vcpu p o i n t e r
2008-04-16 23:28:09 -05:00
* /
_ GLOBAL( _ _ k v m p p c _ v c p u _ r u n )
stwu r1 , - H O S T _ S T A C K _ S I Z E ( r1 )
2020-06-23 21:14:16 +08:00
stw r1 , V C P U _ H O S T _ S T A C K ( r3 ) / * S a v e s t a c k p o i n t e r t o v c p u . * /
2008-04-16 23:28:09 -05:00
/* Save host state to stack. */
2020-06-23 21:14:16 +08:00
mr r4 , r3
2008-04-16 23:28:09 -05:00
mflr r3
stw r3 , H O S T _ S T A C K _ L R ( r1 )
2012-03-05 16:00:28 +01:00
mfcr r5
stw r5 , H O S T _ C R ( r1 )
2008-04-16 23:28:09 -05:00
/* Save host non-volatile register state to stack. */
2012-06-25 13:33:10 +00:00
stw r14 , H O S T _ N V _ G P R ( R 1 4 ) ( r1 )
stw r15 , H O S T _ N V _ G P R ( R 1 5 ) ( r1 )
stw r16 , H O S T _ N V _ G P R ( R 1 6 ) ( r1 )
stw r17 , H O S T _ N V _ G P R ( R 1 7 ) ( r1 )
stw r18 , H O S T _ N V _ G P R ( R 1 8 ) ( r1 )
stw r19 , H O S T _ N V _ G P R ( R 1 9 ) ( r1 )
stw r20 , H O S T _ N V _ G P R ( R 2 0 ) ( r1 )
stw r21 , H O S T _ N V _ G P R ( R 2 1 ) ( r1 )
stw r22 , H O S T _ N V _ G P R ( R 2 2 ) ( r1 )
stw r23 , H O S T _ N V _ G P R ( R 2 3 ) ( r1 )
stw r24 , H O S T _ N V _ G P R ( R 2 4 ) ( r1 )
stw r25 , H O S T _ N V _ G P R ( R 2 5 ) ( r1 )
stw r26 , H O S T _ N V _ G P R ( R 2 6 ) ( r1 )
stw r27 , H O S T _ N V _ G P R ( R 2 7 ) ( r1 )
stw r28 , H O S T _ N V _ G P R ( R 2 8 ) ( r1 )
stw r29 , H O S T _ N V _ G P R ( R 2 9 ) ( r1 )
stw r30 , H O S T _ N V _ G P R ( R 3 0 ) ( r1 )
stw r31 , H O S T _ N V _ G P R ( R 3 1 ) ( r1 )
2008-04-16 23:28:09 -05:00
/* Load guest non-volatiles. */
2012-06-25 13:33:10 +00:00
lwz r14 , V C P U _ G P R ( R 1 4 ) ( r4 )
lwz r15 , V C P U _ G P R ( R 1 5 ) ( r4 )
lwz r16 , V C P U _ G P R ( R 1 6 ) ( r4 )
lwz r17 , V C P U _ G P R ( R 1 7 ) ( r4 )
lwz r18 , V C P U _ G P R ( R 1 8 ) ( r4 )
lwz r19 , V C P U _ G P R ( R 1 9 ) ( r4 )
lwz r20 , V C P U _ G P R ( R 2 0 ) ( r4 )
lwz r21 , V C P U _ G P R ( R 2 1 ) ( r4 )
lwz r22 , V C P U _ G P R ( R 2 2 ) ( r4 )
lwz r23 , V C P U _ G P R ( R 2 3 ) ( r4 )
lwz r24 , V C P U _ G P R ( R 2 4 ) ( r4 )
lwz r25 , V C P U _ G P R ( R 2 5 ) ( r4 )
lwz r26 , V C P U _ G P R ( R 2 6 ) ( r4 )
lwz r27 , V C P U _ G P R ( R 2 7 ) ( r4 )
lwz r28 , V C P U _ G P R ( R 2 8 ) ( r4 )
lwz r29 , V C P U _ G P R ( R 2 9 ) ( r4 )
lwz r30 , V C P U _ G P R ( R 3 0 ) ( r4 )
lwz r31 , V C P U _ G P R ( R 3 1 ) ( r4 )
2008-04-16 23:28:09 -05:00
2011-06-14 18:34:31 -05:00
# ifdef C O N F I G _ S P E
/* save host SPEFSCR and load guest SPEFSCR */
mfspr r3 , S P R N _ S P E F S C R
stw r3 , V C P U _ H O S T _ S P E F S C R ( r4 )
lwz r3 , V C P U _ S P E F S C R ( r4 )
mtspr S P R N _ S P E F S C R , r3
# endif
2008-04-16 23:28:09 -05:00
lightweight_exit :
stw r2 , H O S T _ R 2 ( r1 )
mfspr r3 , S P R N _ P I D
stw r3 , V C P U _ H O S T _ P I D ( r4 )
2008-07-25 13:54:53 -05:00
lwz r3 , V C P U _ S H A D O W _ P I D ( r4 )
2008-04-16 23:28:09 -05:00
mtspr S P R N _ P I D , r3
2022-09-19 19:01:31 +02:00
# ifdef C O N F I G _ P P C _ 8 5 x x
2011-06-14 18:35:14 -05:00
lwz r3 , V C P U _ S H A D O W _ P I D 1 ( r4 )
mtspr S P R N _ P I D 1 , r3
# endif
2008-04-16 23:28:09 -05:00
/* Load some guest volatiles. */
2012-06-25 13:33:10 +00:00
lwz r0 , V C P U _ G P R ( R 0 ) ( r4 )
lwz r2 , V C P U _ G P R ( R 2 ) ( r4 )
lwz r9 , V C P U _ G P R ( R 9 ) ( r4 )
lwz r10 , V C P U _ G P R ( R 1 0 ) ( r4 )
lwz r11 , V C P U _ G P R ( R 1 1 ) ( r4 )
lwz r12 , V C P U _ G P R ( R 1 2 ) ( r4 )
lwz r13 , V C P U _ G P R ( R 1 3 ) ( r4 )
2008-04-16 23:28:09 -05:00
lwz r3 , V C P U _ L R ( r4 )
mtlr r3
lwz r3 , V C P U _ X E R ( r4 )
mtxer r3
/ * Switch t h e I V P R . X X X I f w e t a k e a T L B m i s s a f t e r t h i s w e ' r e s c r e w e d ,
* so h o w d o w e m a k e s u r e v c p u w o n ' t f a u l t ? * /
lis r8 , k v m p p c _ b o o k e _ h a n d l e r s @ha
lwz r8 , k v m p p c _ b o o k e _ h a n d l e r s @l(r8)
mtspr S P R N _ I V P R , r8
KVM: PPC: Paravirtualize SPRG4-7, ESR, PIR, MASn
This allows additional registers to be accessed by the guest
in PR-mode KVM without trapping.
SPRG4-7 are readable from userspace. On booke, KVM will sync
these registers when it enters the guest, so that accesses from
guest userspace will work. The guest kernel, OTOH, must consistently
use either the real registers or the shared area between exits. This
also applies to the already-paravirted SPRG3.
On non-booke, it's not clear to what extent SPRG4-7 are supported
(they're not architected for book3s, but exist on at least some classic
chips). They are copied in the get/set regs ioctls, but I do not see any
non-booke emulation. I also do not see any syncing with real registers
(in PR-mode) including the user-readable SPRG3. This patch should not
make that situation any worse.
Signed-off-by: Scott Wood <scottwood@freescale.com>
Signed-off-by: Alexander Graf <agraf@suse.de>
Signed-off-by: Avi Kivity <avi@redhat.com>
2011-11-08 18:23:30 -06:00
lwz r5 , V C P U _ S H A R E D ( r4 )
2008-04-16 23:28:09 -05:00
/ * Can' t s w i t c h t h e s t a c k p o i n t e r u n t i l a f t e r I V P R i s s w i t c h e d ,
* because h o s t i n t e r r u p t h a n d l e r s w o u l d g e t c o n f u s e d . * /
2012-06-25 13:33:10 +00:00
lwz r1 , V C P U _ G P R ( R 1 ) ( r4 )
2008-04-16 23:28:09 -05:00
KVM: PPC: Paravirtualize SPRG4-7, ESR, PIR, MASn
This allows additional registers to be accessed by the guest
in PR-mode KVM without trapping.
SPRG4-7 are readable from userspace. On booke, KVM will sync
these registers when it enters the guest, so that accesses from
guest userspace will work. The guest kernel, OTOH, must consistently
use either the real registers or the shared area between exits. This
also applies to the already-paravirted SPRG3.
On non-booke, it's not clear to what extent SPRG4-7 are supported
(they're not architected for book3s, but exist on at least some classic
chips). They are copied in the get/set regs ioctls, but I do not see any
non-booke emulation. I also do not see any syncing with real registers
(in PR-mode) including the user-readable SPRG3. This patch should not
make that situation any worse.
Signed-off-by: Scott Wood <scottwood@freescale.com>
Signed-off-by: Alexander Graf <agraf@suse.de>
Signed-off-by: Avi Kivity <avi@redhat.com>
2011-11-08 18:23:30 -06:00
/ *
* Host i n t e r r u p t h a n d l e r s m a y h a v e c l o b b e r e d t h e s e
* guest- r e a d a b l e S P R G s , o r t h e g u e s t k e r n e l m a y h a v e
* written d i r e c t l y t o t h e s h a r e d a r e a , s o w e
* need t o r e l o a d t h e m h e r e w i t h t h e g u e s t ' s v a l u e s .
* /
2012-04-25 01:27:34 +00:00
PPC_ L D ( r3 , V C P U _ S H A R E D _ S P R G 4 , r5 )
2009-07-14 20:52:54 +00:00
mtspr S P R N _ S P R G 4 W , r3
2012-04-25 01:27:34 +00:00
PPC_ L D ( r3 , V C P U _ S H A R E D _ S P R G 5 , r5 )
2009-07-14 20:52:54 +00:00
mtspr S P R N _ S P R G 5 W , r3
2012-04-25 01:27:34 +00:00
PPC_ L D ( r3 , V C P U _ S H A R E D _ S P R G 6 , r5 )
2009-07-14 20:52:54 +00:00
mtspr S P R N _ S P R G 6 W , r3
2012-04-25 01:27:34 +00:00
PPC_ L D ( r3 , V C P U _ S H A R E D _ S P R G 7 , r5 )
2009-07-14 20:52:54 +00:00
mtspr S P R N _ S P R G 7 W , r3
2008-04-16 23:28:09 -05:00
2008-12-02 15:51:57 -06:00
# ifdef C O N F I G _ K V M _ E X I T _ T I M I N G
/* save enter time */
1 :
mfspr r6 , S P R N _ T B R U
mfspr r7 , S P R N _ T B R L
mfspr r8 , S P R N _ T B R U
cmpw r8 , r6
bne 1 b
stw r7 , V C P U _ T I M I N G _ L A S T _ E N T E R _ T B L ( r4 )
stw r8 , V C P U _ T I M I N G _ L A S T _ E N T E R _ T B U ( r4 )
# endif
2008-04-16 23:28:09 -05:00
/* Finish loading guest volatiles and jump to guest. */
lwz r3 , V C P U _ C T R ( r4 )
2011-06-14 18:34:29 -05:00
lwz r5 , V C P U _ C R ( r4 )
lwz r6 , V C P U _ P C ( r4 )
lwz r7 , V C P U _ S H A D O W _ M S R ( r4 )
2008-04-16 23:28:09 -05:00
mtctr r3
2011-06-14 18:34:29 -05:00
mtcr r5
mtsrr0 r6
mtsrr1 r7
2012-06-25 13:33:10 +00:00
lwz r5 , V C P U _ G P R ( R 5 ) ( r4 )
lwz r6 , V C P U _ G P R ( R 6 ) ( r4 )
lwz r7 , V C P U _ G P R ( R 7 ) ( r4 )
lwz r8 , V C P U _ G P R ( R 8 ) ( r4 )
2008-07-25 13:54:49 -05:00
/ * Clear a n y d e b u g e v e n t s w h i c h o c c u r r e d s i n c e w e d i s a b l e d M S R [ D E ] .
* XXX T h i s g i v e s u s a 3 - i n s t r u c t i o n w i n d o w i n w h i c h a b r e a k p o i n t
* intended f o r g u e s t c o n t e x t c o u l d f i r e i n t h e h o s t i n s t e a d . * /
lis r3 , 0 x f f f f
ori r3 , r3 , 0 x f f f f
mtspr S P R N _ D B S R , r3
2012-06-25 13:33:10 +00:00
lwz r3 , V C P U _ G P R ( R 3 ) ( r4 )
lwz r4 , V C P U _ G P R ( R 4 ) ( r4 )
2008-04-16 23:28:09 -05:00
rfi
2011-06-14 18:34:31 -05:00
2013-01-15 22:24:39 +00:00
.data
.align 4
.globl kvmppc_booke_handler_addr
kvmppc_booke_handler_addr :
KVM_ H A N D L E R _ A D D R B O O K E _ I N T E R R U P T _ C R I T I C A L
KVM_ H A N D L E R _ A D D R B O O K E _ I N T E R R U P T _ M A C H I N E _ C H E C K
KVM_ H A N D L E R _ A D D R B O O K E _ I N T E R R U P T _ D A T A _ S T O R A G E
KVM_ H A N D L E R _ A D D R B O O K E _ I N T E R R U P T _ I N S T _ S T O R A G E
KVM_ H A N D L E R _ A D D R B O O K E _ I N T E R R U P T _ E X T E R N A L
KVM_ H A N D L E R _ A D D R B O O K E _ I N T E R R U P T _ A L I G N M E N T
KVM_ H A N D L E R _ A D D R B O O K E _ I N T E R R U P T _ P R O G R A M
KVM_ H A N D L E R _ A D D R B O O K E _ I N T E R R U P T _ F P _ U N A V A I L
KVM_ H A N D L E R _ A D D R B O O K E _ I N T E R R U P T _ S Y S C A L L
KVM_ H A N D L E R _ A D D R B O O K E _ I N T E R R U P T _ A P _ U N A V A I L
KVM_ H A N D L E R _ A D D R B O O K E _ I N T E R R U P T _ D E C R E M E N T E R
KVM_ H A N D L E R _ A D D R B O O K E _ I N T E R R U P T _ F I T
KVM_ H A N D L E R _ A D D R B O O K E _ I N T E R R U P T _ W A T C H D O G
KVM_ H A N D L E R _ A D D R B O O K E _ I N T E R R U P T _ D T L B _ M I S S
KVM_ H A N D L E R _ A D D R B O O K E _ I N T E R R U P T _ I T L B _ M I S S
KVM_ H A N D L E R _ A D D R B O O K E _ I N T E R R U P T _ D E B U G
KVM_ H A N D L E R _ A D D R B O O K E _ I N T E R R U P T _ S P E _ U N A V A I L
KVM_ H A N D L E R _ A D D R B O O K E _ I N T E R R U P T _ S P E _ F P _ D A T A
KVM_ H A N D L E R _ A D D R B O O K E _ I N T E R R U P T _ S P E _ F P _ R O U N D
KVM_ H A N D L E R _ E N D / * A l w a y s k e e p t h i s i n e n d * /
2011-06-14 18:34:31 -05:00
# ifdef C O N F I G _ S P E
_ GLOBAL( k v m p p c _ s a v e _ g u e s t _ s p e )
cmpi 0 ,r3 ,0
beqlr-
SAVE_ 3 2 E V R S ( 0 , r4 , r3 , V C P U _ E V R )
evxor e v r6 , e v r6 , e v r6
evmwumiaa e v r6 , e v r6 , e v r6
li r4 ,V C P U _ A C C
evstddx e v r6 , r4 , r3 / * s a v e a c c * /
blr
_ GLOBAL( k v m p p c _ l o a d _ g u e s t _ s p e )
cmpi 0 ,r3 ,0
beqlr-
li r4 ,V C P U _ A C C
evlddx e v r6 ,r4 ,r3
evmra e v r6 ,e v r6 / * l o a d a c c * /
REST_ 3 2 E V R S ( 0 , r4 , r3 , V C P U _ E V R )
blr
# endif