2005-04-17 02:20:36 +04:00
/ *
* Access t o t h e s h a r e d d a t a p a g e b y t h e v D S O & s y s c a l l m a p
*
* Copyright ( C ) 2 0 0 4 B e n j a m i n H e r r e n s c h m u i d t ( b e n h @kernel.crashing.org), IBM Corp.
*
* 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 i t 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
* as p u b l i s h e d 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 ; either version
* 2 of t h e L i c e n s e , o r ( a t y o u r o p t i o n ) a n y l a t e r v e r s i o n .
* /
# include < a s m / p r o c e s s o r . h >
# include < a s m / p p c _ a s m . h >
2005-09-09 22:57:26 +04:00
# include < a s m / a s m - o f f s e t s . h >
2005-04-17 02:20:36 +04:00
# include < a s m / u n i s t d . h >
# include < a s m / v d s o . h >
.text
powerpc/vdso: Avoid link stack corruption in __get_datapage()
powerpc has a link register (lr) used for calling functions. We "bl
<func>" to call a function, and "blr" to return back to the call site.
The lr is only a single register, so if we call another function from
inside this function (ie. nested calls), software must save away the
lr on the software stack before calling the new function. Before
returning (ie. before the "blr"), the lr is restored by software from
the software stack.
This makes branch prediction quite difficult for the processor as it
will only know the branch target just before the "blr".
To help with this, modern powerpc processors keep a (non-architected)
hardware stack of lr called a "link stack". When a "bl <func>" is
run, the lr is pushed onto this stack. When a "blr" is called, the
branch predictor pops the lr value from the top of the link stack, and
uses it to predict the branch target. Hence the processor pipeline
knows a lot earlier the branch target.
This works great but there are some cases where you call "bl" but
without a matching "blr". Once such case is when trying to determine
the program counter (which can't be read directly). Here you "bl+4;
mflr" to get the program counter. If you do this, the link stack will
get out of sync with reality, causing the branch predictor to
mis-predict subsequent function returns.
To avoid this, modern micro-architectures have a special case of bl.
Using the form "bcl 20,31,+4", ensures the processor doesn't push to
the link stack.
The 32 and 64 bit variants of __get_datapage() use a "bl; mflr" to
determine the loaded address of the VDSO. The current versions of
these attempt to use this special bl variant.
Unfortunately they use +8 rather than the required +4. Hence the
current code results in the link stack getting out of sync with
reality and hence the resulting performance degradation.
This patch moves it to bcl+4 by moving __kernel_datapage_offset out of
__get_datapage().
With this patch, running a gettimeofday() (which uses
__get_datapage()) microbenchmark we get a decent bump in performance
on POWER7/8.
For the benchmark in tools/testing/selftests/powerpc/benchmarks/gettimeofday.c
POWER8:
64bit gets ~4% improvement
32bit gets ~9% improvement
POWER7:
64bit gets ~7% improvement
Signed-off-by: Michael Neuling <mikey@neuling.org>
Reported-by: Aaron Sawdey <sawdey@us.ibm.com>
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
2015-09-25 07:01:40 +03:00
.global _ _ kernel_ d a t a p a g e _ o f f s e t ;
__kernel_datapage_offset :
.long 0
2005-04-17 02:20:36 +04:00
V_ F U N C T I O N _ B E G I N ( _ _ g e t _ d a t a p a g e )
.cfi_startproc
/ * We d o n ' t w a n t t h a t e x p o s e d o r o v e r r i d a b l e a s w e w a n t o t h e r o b j e c t s
* to b e a b l e t o b l d i r e c t l y t o h e r e
* /
.protected __get_datapage
.hidden __get_datapage
mflr r0
.cfi_register lr,r0
powerpc/vdso: Avoid link stack corruption in __get_datapage()
powerpc has a link register (lr) used for calling functions. We "bl
<func>" to call a function, and "blr" to return back to the call site.
The lr is only a single register, so if we call another function from
inside this function (ie. nested calls), software must save away the
lr on the software stack before calling the new function. Before
returning (ie. before the "blr"), the lr is restored by software from
the software stack.
This makes branch prediction quite difficult for the processor as it
will only know the branch target just before the "blr".
To help with this, modern powerpc processors keep a (non-architected)
hardware stack of lr called a "link stack". When a "bl <func>" is
run, the lr is pushed onto this stack. When a "blr" is called, the
branch predictor pops the lr value from the top of the link stack, and
uses it to predict the branch target. Hence the processor pipeline
knows a lot earlier the branch target.
This works great but there are some cases where you call "bl" but
without a matching "blr". Once such case is when trying to determine
the program counter (which can't be read directly). Here you "bl+4;
mflr" to get the program counter. If you do this, the link stack will
get out of sync with reality, causing the branch predictor to
mis-predict subsequent function returns.
To avoid this, modern micro-architectures have a special case of bl.
Using the form "bcl 20,31,+4", ensures the processor doesn't push to
the link stack.
The 32 and 64 bit variants of __get_datapage() use a "bl; mflr" to
determine the loaded address of the VDSO. The current versions of
these attempt to use this special bl variant.
Unfortunately they use +8 rather than the required +4. Hence the
current code results in the link stack getting out of sync with
reality and hence the resulting performance degradation.
This patch moves it to bcl+4 by moving __kernel_datapage_offset out of
__get_datapage().
With this patch, running a gettimeofday() (which uses
__get_datapage()) microbenchmark we get a decent bump in performance
on POWER7/8.
For the benchmark in tools/testing/selftests/powerpc/benchmarks/gettimeofday.c
POWER8:
64bit gets ~4% improvement
32bit gets ~9% improvement
POWER7:
64bit gets ~7% improvement
Signed-off-by: Michael Neuling <mikey@neuling.org>
Reported-by: Aaron Sawdey <sawdey@us.ibm.com>
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
2015-09-25 07:01:40 +03:00
bcl 2 0 ,3 1 ,d a t a _ p a g e _ b r a n c h
data_page_branch :
2005-04-17 02:20:36 +04:00
mflr r3
mtlr r0
powerpc/vdso: Avoid link stack corruption in __get_datapage()
powerpc has a link register (lr) used for calling functions. We "bl
<func>" to call a function, and "blr" to return back to the call site.
The lr is only a single register, so if we call another function from
inside this function (ie. nested calls), software must save away the
lr on the software stack before calling the new function. Before
returning (ie. before the "blr"), the lr is restored by software from
the software stack.
This makes branch prediction quite difficult for the processor as it
will only know the branch target just before the "blr".
To help with this, modern powerpc processors keep a (non-architected)
hardware stack of lr called a "link stack". When a "bl <func>" is
run, the lr is pushed onto this stack. When a "blr" is called, the
branch predictor pops the lr value from the top of the link stack, and
uses it to predict the branch target. Hence the processor pipeline
knows a lot earlier the branch target.
This works great but there are some cases where you call "bl" but
without a matching "blr". Once such case is when trying to determine
the program counter (which can't be read directly). Here you "bl+4;
mflr" to get the program counter. If you do this, the link stack will
get out of sync with reality, causing the branch predictor to
mis-predict subsequent function returns.
To avoid this, modern micro-architectures have a special case of bl.
Using the form "bcl 20,31,+4", ensures the processor doesn't push to
the link stack.
The 32 and 64 bit variants of __get_datapage() use a "bl; mflr" to
determine the loaded address of the VDSO. The current versions of
these attempt to use this special bl variant.
Unfortunately they use +8 rather than the required +4. Hence the
current code results in the link stack getting out of sync with
reality and hence the resulting performance degradation.
This patch moves it to bcl+4 by moving __kernel_datapage_offset out of
__get_datapage().
With this patch, running a gettimeofday() (which uses
__get_datapage()) microbenchmark we get a decent bump in performance
on POWER7/8.
For the benchmark in tools/testing/selftests/powerpc/benchmarks/gettimeofday.c
POWER8:
64bit gets ~4% improvement
32bit gets ~9% improvement
POWER7:
64bit gets ~7% improvement
Signed-off-by: Michael Neuling <mikey@neuling.org>
Reported-by: Aaron Sawdey <sawdey@us.ibm.com>
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
2015-09-25 07:01:40 +03:00
addi r3 , r3 , _ _ k e r n e l _ d a t a p a g e _ o f f s e t - d a t a _ p a g e _ b r a n c h
2005-04-17 02:20:36 +04:00
lwz r0 ,0 ( r3 )
add r3 ,r0 ,r3
blr
.cfi_endproc
V_ F U N C T I O N _ E N D ( _ _ g e t _ d a t a p a g e )
/ *
* void * _ _ k e r n e l _ g e t _ s y s c a l l _ m a p ( u n s i g n e d i n t * s y s c a l l _ c o u n t ) ;
*
* returns a p o i n t e r t o t h e s y s c a l l m a p . t h e m a p i s a g n o s t i c t o t h e
* size o f " l o n g " , u n l i k e k e r n e l b i t o p s , i t s t o r e s b i t s f r o m t o p t o
* bottom s o t h a t m e m o r y a c t u a l l y c o n t a i n s a l i n e a r b i t m a p
* check f o r s y s c a l l N b y t e s t i n g b i t ( 0 x80 0 0 0 0 0 0 > > ( N & 0 x1 f ) ) o f
* 3 2 bits i n t a t N > > 5 .
* /
V_ F U N C T I O N _ B E G I N ( _ _ k e r n e l _ g e t _ s y s c a l l _ m a p )
.cfi_startproc
mflr r12
.cfi_register lr,r12
mr r4 ,r3
bl _ _ g e t _ d a t a p a g e @local
mtlr r12
addi r3 ,r3 ,C F G _ S Y S C A L L _ M A P 3 2
cmpli c r0 ,r4 ,0
beqlr
2015-11-19 09:04:53 +03:00
li r0 ,N R _ s y s c a l l s
2005-04-17 02:20:36 +04:00
stw r0 ,0 ( r4 )
2005-11-16 05:54:32 +03:00
crclr c r0 * 4 + s o
2005-04-17 02:20:36 +04:00
blr
.cfi_endproc
V_ F U N C T I O N _ E N D ( _ _ k e r n e l _ g e t _ s y s c a l l _ m a p )
2005-11-11 13:15:21 +03:00
/ *
* void u n s i g n e d l o n g l o n g _ _ k e r n e l _ g e t _ t b f r e q ( v o i d ) ;
*
* returns t h e t i m e b a s e f r e q u e n c y i n H Z
* /
V_ F U N C T I O N _ B E G I N ( _ _ k e r n e l _ g e t _ t b f r e q )
.cfi_startproc
mflr r12
.cfi_register lr,r12
bl _ _ g e t _ d a t a p a g e @local
lwz r4 ,( C F G _ T B _ T I C K S _ P E R _ S E C + 4 ) ( r3 )
2005-11-14 06:55:58 +03:00
lwz r3 ,C F G _ T B _ T I C K S _ P E R _ S E C ( r3 )
2005-11-11 13:15:21 +03:00
mtlr r12
2005-11-16 05:54:32 +03:00
crclr c r0 * 4 + s o
2005-11-14 06:55:58 +03:00
blr
2005-11-11 13:15:21 +03:00
.cfi_endproc
V_ F U N C T I O N _ E N D ( _ _ k e r n e l _ g e t _ t b f r e q )