2010-07-29 14:47:57 +02:00
/ *
* This p r o g r a m i s f r e e s o f t w a r e ; you can redistribute it and/or modify
* it u n d e r t h e t e r m s o f t h e G N U G e n e r a l P u b l i c L i c e n s e , v e r s i o n 2 , a s
* published b y t h e F r e e S o f t w a r e F o u n d a t i o n .
*
* This p r o g r a m i s d i s t r i b u t e d i n t h e h o p e t h a t i t w i l l b e u s e f u l ,
* but W I T H O U T A N Y W A R R A N T Y ; without even the implied warranty of
* MERCHANTABILITY o r F I T N E S S F O R A P A R T I C U L A R P U R P O S E . S e e t h e
* GNU G e n e r a l P u b l i c L i c e n s e f o r m o r e d e t a i l s .
*
* You s h o u l d h a v e r e c e i v e d a c o p y o f t h e G N U G e n e r a l P u b l i c L i c e n s e
* along w i t h t h i s p r o g r a m ; if not, write to the Free Software
* Foundation, 5 1 F r a n k l i n S t r e e t , F i f t h F l o o r , B o s t o n , M A 0 2 1 1 0 - 1 3 0 1 , U S A .
*
* Copyright S U S E L i n u x P r o d u c t s G m b H 2 0 1 0
*
* Authors : Alexander G r a f < a g r a f @suse.de>
* /
# 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 >
/* Hypercall entry point. Will be patched with device tree instructions. */
.global kvm_hypercall_start
kvm_hypercall_start :
li r3 , - 1
nop
nop
nop
blr
# define K V M _ M A G I C _ P A G E ( - 4 0 9 6 )
2010-07-29 14:48:03 +02:00
# ifdef C O N F I G _ 6 4 B I T
# define L L 6 4 ( r e g , o f f s , r e g 2 ) l d r e g , ( o f f s ) ( r e g 2 )
# define S T L 6 4 ( r e g , o f f s , r e g 2 ) s t d r e g , ( o f f s ) ( r e g 2 )
# else
# define L L 6 4 ( r e g , o f f s , r e g 2 ) l w z r e g , ( o f f s + 4 ) ( r e g 2 )
# define S T L 6 4 ( r e g , o f f s , r e g 2 ) s t w r e g , ( o f f s + 4 ) ( r e g 2 )
# endif
# define S C R A T C H _ S A V E \
/ * Enable c r i t i c a l s e c t i o n . W e a r e c r i t i c a l i f \
shared- > c r i t i c a l = = r1 * / \
STL6 4 ( r1 , K V M _ M A G I C _ P A G E + K V M _ M A G I C _ C R I T I C A L , 0 ) ; \
\
/* Save state */ \
PPC_ S T L r31 , ( K V M _ M A G I C _ P A G E + K V M _ M A G I C _ S C R A T C H 1 ) ( 0 ) ; \
PPC_ S T L r30 , ( K V M _ M A G I C _ P A G E + K V M _ M A G I C _ S C R A T C H 2 ) ( 0 ) ; \
mfcr r31 ; \
stw r31 , ( K V M _ M A G I C _ P A G E + K V M _ M A G I C _ S C R A T C H 3 ) ( 0 ) ;
# define S C R A T C H _ R E S T O R E \
/* Restore state */ \
PPC_ L L r31 , ( K V M _ M A G I C _ P A G E + K V M _ M A G I C _ S C R A T C H 1 ) ( 0 ) ; \
lwz r30 , ( K V M _ M A G I C _ P A G E + K V M _ M A G I C _ S C R A T C H 3 ) ( 0 ) ; \
mtcr r30 ; \
PPC_ L L r30 , ( K V M _ M A G I C _ P A G E + K V M _ M A G I C _ S C R A T C H 2 ) ( 0 ) ; \
\
/ * Disable c r i t i c a l s e c t i o n . W e a r e c r i t i c a l i f \
shared- > c r i t i c a l = = r1 a n d r2 i s a l w a y s ! = r1 * / \
STL6 4 ( r2 , K V M _ M A G I C _ P A G E + K V M _ M A G I C _ C R I T I C A L , 0 ) ;
2010-07-29 14:48:04 +02:00
.global kvm_emulate_mtmsrd
kvm_emulate_mtmsrd :
SCRATCH_ S A V E
/* Put MSR & ~(MSR_EE|MSR_RI) in r31 */
LL6 4 ( r31 , K V M _ M A G I C _ P A G E + K V M _ M A G I C _ M S R , 0 )
lis r30 , ( ~ ( M S R _ E E | M S R _ R I ) ) @h
ori r30 , r30 , ( ~ ( M S R _ E E | M S R _ R I ) ) @l
and r31 , r31 , r30
/* OR the register's (MSR_EE|MSR_RI) on MSR */
kvm_emulate_mtmsrd_reg :
andi. r30 , r0 , ( M S R _ E E | M S R _ R I )
or r31 , r31 , r30
/* Put MSR back into magic page */
STL6 4 ( r31 , K V M _ M A G I C _ P A G E + K V M _ M A G I C _ M S R , 0 )
/* Check if we have to fetch an interrupt */
lwz r31 , ( K V M _ M A G I C _ P A G E + K V M _ M A G I C _ I N T ) ( 0 )
cmpwi r31 , 0
beq+ n o _ c h e c k
/* Check if we may trigger an interrupt */
andi. r30 , r30 , M S R _ E E
beq n o _ c h e c k
SCRATCH_ R E S T O R E
/* Nag hypervisor */
tlbsync
b k v m _ e m u l a t e _ m t m s r d _ b r a n c h
no_check :
SCRATCH_ R E S T O R E
/* Go back to caller */
kvm_emulate_mtmsrd_branch :
b .
kvm_emulate_mtmsrd_end :
.global kvm_emulate_mtmsrd_branch_offs
kvm_emulate_mtmsrd_branch_offs :
.long ( kvm_ e m u l a t e _ m t m s r d _ b r a n c h - k v m _ e m u l a t e _ m t m s r d ) / 4
.global kvm_emulate_mtmsrd_reg_offs
kvm_emulate_mtmsrd_reg_offs :
.long ( kvm_ e m u l a t e _ m t m s r d _ r e g - k v m _ e m u l a t e _ m t m s r d ) / 4
.global kvm_emulate_mtmsrd_len
kvm_emulate_mtmsrd_len :
.long ( kvm_ e m u l a t e _ m t m s r d _ e n d - k v m _ e m u l a t e _ m t m s r d ) / 4