2014-10-28 13:26:48 +00: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 .
* /
# include < a s m / a s s e m b l e r . h >
# include < a s m / f t r a c e . h >
# include < a s m / u n w i n d . h >
# include " e n t r y - h e a d e r . S "
/ *
* When c o m p i l i n g w i t h - p g , g c c i n s e r t s a c a l l t o t h e m c o u n t r o u t i n e a t t h e
* start o f e v e r y f u n c t i o n . I n m c o u n t , a p a r t f r o m t h e f u n c t i o n ' s a d d r e s s ( i n
* lr) , w e n e e d t o g e t h o l d o f t h e f u n c t i o n ' s c a l l e r ' s a d d r e s s .
*
* Older G C C s ( p r e - 4 . 4 ) i n s e r t e d a c a l l t o a r o u t i n e c a l l e d m c o u n t l i k e t h i s :
*
* bl m c o u n t
*
* These v e r s i o n s h a v e t h e l i m i t a t i o n t h a t i n o r d e r f o r t h e m c o u n t r o u t i n e t o
* be a b l e t o d e t e r m i n e t h e f u n c t i o n ' s c a l l e r ' s a d d r e s s , a n A P C S - s t y l e f r a m e
* pointer ( w h i c h i s s e t u p w i t h s o m e t h i n g l i k e t h e c o d e b e l o w ) i s r e q u i r e d .
*
* mov i p , s p
* push { f p , i p , l r , p c }
* sub f p , i p , #4
*
* With E A B I , t h e s e f r a m e p o i n t e r s a r e n o t a v a i l a b l e u n l e s s - m a p c s - f r a m e i s
* specified, a n d i f b u i l d i n g a s T h u m b - 2 , n o t e v e n t h e n .
*
* Newer G C C s ( 4 . 4 + ) s o l v e t h i s p r o b l e m b y i n t r o d u c i n g a n e w v e r s i o n o f m c o u n t ,
* with c a l l s i t e s l i k e :
*
* push { l r }
* bl _ _ g n u _ m c o u n t _ n c
*
* With t h e s e c o m p i l e r s , f r a m e p o i n t e r s a r e n o t n e c e s s a r y .
*
* mcount c a n b e t h o u g h t o f a s a f u n c t i o n c a l l e d i n t h e m i d d l e o f a s u b r o u t i n e
* call. A s s u c h , i t n e e d s t o b e t r a n s p a r e n t f o r b o t h t h e c a l l e r a n d t h e
* callee : the o r i g i n a l l r n e e d s t o b e r e s t o r e d w h e n l e a v i n g m c o u n t , a n d n o
* registers s h o u l d b e c l o b b e r e d . ( I n t h e _ _ g n u _ m c o u n t _ n c i m p l e m e n t a t i o n , w e
* clobber t h e i p r e g i s t e r . T h i s i s O K b e c a u s e t h e A R M c a l l i n g c o n v e n t i o n
* allows i t t o b e c l o b b e r e d i n s u b r o u t i n e s a n d d o e s n ' t u s e i t t o h o l d
* parameters. )
*
* When u s i n g d y n a m i c f t r a c e , w e p a t c h o u t t h e m c o u n t c a l l b y a " m o v r0 , r0 "
* for t h e m c o u n t c a s e , a n d a " p o p { l r } " f o r t h e _ _ g n u _ m c o u n t _ n c c a s e ( s e e
* arch/ a r m / k e r n e l / f t r a c e . c ) .
* /
# ifndef C O N F I G _ O L D _ M C O U N T
# if ( _ _ G N U C _ _ < 4 | | ( _ _ G N U C _ _ = = 4 & & _ _ G N U C _ M I N O R _ _ < 4 ) )
# error F t r a c e r e q u i r e s C O N F I G _ F R A M E _ P O I N T E R =y w i t h G C C o l d e r t h a n 4 . 4 . 0 .
# endif
# endif
.macro mcount_adjust_addr rd, r n
bic \ r d , \ r n , #1 @ clear the Thumb bit if present
sub \ r d , \ r d , #M C O U N T _ I N S N _ S I Z E
.endm
.macro __mcount suffix
mcount_ e n t e r
ldr r0 , =ftrace_trace_function
ldr r2 , [ r0 ]
adr r0 , . L f t r a c e _ s t u b
cmp r0 , r2
bne 1 f
# ifdef C O N F I G _ F U N C T I O N _ G R A P H _ T R A C E R
ldr r1 , =ftrace_graph_return
ldr r2 , [ r1 ]
cmp r0 , r2
bne f t r a c e _ g r a p h _ c a l l e r \ s u f f i x
ldr r1 , =ftrace_graph_entry
ldr r2 , [ r1 ]
ldr r0 , =ftrace_graph_entry_stub
cmp r0 , r2
bne f t r a c e _ g r a p h _ c a l l e r \ s u f f i x
# endif
mcount_ e x i t
1 : mcount_ g e t _ l r r1 @ lr of instrumented func
mcount_ a d j u s t _ a d d r r0 , l r @ instrumented function
2015-04-21 14:17:25 +01:00
badr l r , 2 f
2014-10-28 13:26:48 +00:00
mov p c , r2
2 : mcount_ e x i t
.endm
.macro __ftrace_caller suffix
mcount_ e n t e r
mcount_ g e t _ l r r1 @ lr of instrumented func
mcount_ a d j u s t _ a d d r r0 , l r @ instrumented function
.globl ftrace_ c a l l \ s u f f i x
ftrace_ c a l l \ s u f f i x :
bl f t r a c e _ s t u b
# ifdef C O N F I G _ F U N C T I O N _ G R A P H _ T R A C E R
.globl ftrace_ g r a p h _ c a l l \ s u f f i x
ftrace_ g r a p h _ c a l l \ s u f f i x :
mov r0 , r0
# endif
mcount_ e x i t
.endm
.macro __ftrace_graph_caller
sub r0 , f p , #4 @ &lr of instrumented routine (&parent)
# ifdef C O N F I G _ D Y N A M I C _ F T R A C E
@ called from __ftrace_caller, saved in mcount_enter
ldr r1 , [ s p , #16 ] @ instrumented routine (func)
mcount_ a d j u s t _ a d d r r1 , r1
# else
@ called from __mcount, untouched in lr
mcount_ a d j u s t _ a d d r r1 , l r @ instrumented routine (func)
# endif
mov r2 , f p @ frame pointer
bl p r e p a r e _ f t r a c e _ r e t u r n
mcount_ e x i t
.endm
# ifdef C O N F I G _ O L D _ M C O U N T
/ *
* mcount
* /
.macro mcount_enter
stmdb s p ! , { r0 - r3 , l r }
.endm
.macro mcount_get_lr reg
ldr \ r e g , [ f p , #- 4 ]
.endm
.macro mcount_exit
ldr l r , [ f p , #- 4 ]
ldmia s p ! , { r0 - r3 , p c }
.endm
ENTRY( m c o u n t )
# ifdef C O N F I G _ D Y N A M I C _ F T R A C E
stmdb s p ! , { l r }
ldr l r , [ f p , #- 4 ]
ldmia s p ! , { p c }
# else
_ _ mcount _ o l d
# endif
ENDPROC( m c o u n t )
# ifdef C O N F I G _ D Y N A M I C _ F T R A C E
ENTRY( f t r a c e _ c a l l e r _ o l d )
_ _ ftrace_ c a l l e r _ o l d
ENDPROC( f t r a c e _ c a l l e r _ o l d )
# endif
# ifdef C O N F I G _ F U N C T I O N _ G R A P H _ T R A C E R
ENTRY( f t r a c e _ g r a p h _ c a l l e r _ o l d )
_ _ ftrace_ g r a p h _ c a l l e r
ENDPROC( f t r a c e _ g r a p h _ c a l l e r _ o l d )
# endif
.purgem mcount_enter
.purgem mcount_get_lr
.purgem mcount_exit
# endif
/ *
* _ _ gnu_ m c o u n t _ n c
* /
.macro mcount_enter
/ *
* This p a d c o m p e n s a t e s f o r t h e p u s h { l r } a t t h e c a l l s i t e . N o t e t h a t w e a r e
* unable t o u n w i n d t h r o u g h a f u n c t i o n w h i c h d o e s n o t o t h e r w i s e s a v e i t s l r .
* /
UNWIND( . p a d #4 )
stmdb s p ! , { r0 - r3 , l r }
UNWIND( . s a v e { r0 - r3 , l r } )
.endm
.macro mcount_get_lr reg
ldr \ r e g , [ s p , #20 ]
.endm
.macro mcount_exit
ldmia s p ! , { r0 - r3 , i p , l r }
ret i p
.endm
ENTRY( _ _ g n u _ m c o u n t _ n c )
UNWIND( . f n s t a r t )
# ifdef C O N F I G _ D Y N A M I C _ F T R A C E
mov i p , l r
ldmia s p ! , { l r }
ret i p
# else
_ _ mcount
# endif
UNWIND( . f n e n d )
ENDPROC( _ _ g n u _ m c o u n t _ n c )
# ifdef C O N F I G _ D Y N A M I C _ F T R A C E
ENTRY( f t r a c e _ c a l l e r )
UNWIND( . f n s t a r t )
_ _ ftrace_ c a l l e r
UNWIND( . f n e n d )
ENDPROC( f t r a c e _ c a l l e r )
# endif
# ifdef C O N F I G _ F U N C T I O N _ G R A P H _ T R A C E R
ENTRY( f t r a c e _ g r a p h _ c a l l e r )
UNWIND( . f n s t a r t )
_ _ ftrace_ g r a p h _ c a l l e r
UNWIND( . f n e n d )
ENDPROC( f t r a c e _ g r a p h _ c a l l e r )
# endif
.purgem mcount_enter
.purgem mcount_get_lr
.purgem mcount_exit
# ifdef C O N F I G _ F U N C T I O N _ G R A P H _ T R A C E R
.globl return_to_handler
return_to_handler :
stmdb s p ! , { r0 - r3 }
mov r0 , f p @ frame pointer
bl f t r a c e _ r e t u r n _ t o _ h a n d l e r
mov l r , r0 @ r0 has real ret addr
ldmia s p ! , { r0 - r3 }
ret l r
# endif
ENTRY( f t r a c e _ s t u b )
.Lftrace_stub :
ret l r
ENDPROC( f t r a c e _ s t u b )