2019-06-04 11:11:33 +03:00
// SPDX-License-Identifier: GPL-2.0-only
2014-04-30 13:54:33 +04:00
/*
* arch / arm64 / kernel / ftrace . c
*
* Copyright ( C ) 2013 Linaro Limited
* Author : AKASHI Takahiro < takahiro . akashi @ linaro . org >
*/
# include <linux/ftrace.h>
2017-06-06 20:00:22 +03:00
# include <linux/module.h>
2014-04-30 13:54:33 +04:00
# include <linux/swab.h>
# include <linux/uaccess.h>
# include <asm/cacheflush.h>
2017-06-06 20:00:22 +03:00
# include <asm/debug-monitors.h>
2014-04-30 13:54:33 +04:00
# include <asm/ftrace.h>
# include <asm/insn.h>
2014-04-30 13:54:34 +04:00
# ifdef CONFIG_DYNAMIC_FTRACE
/*
* Replace a single instruction , which may be a branch or NOP .
* If @ validate = = true , a replaced instruction is checked against ' old ' .
*/
static int ftrace_modify_code ( unsigned long pc , u32 old , u32 new ,
bool validate )
{
u32 replaced ;
/*
* Note :
2015-12-04 06:38:40 +03:00
* We are paranoid about modifying text , as if a bug were to happen , it
* could cause us to read or write to someplace that could cause harm .
* Carefully read and modify the code with aarch64_insn_ * ( ) which uses
* probe_kernel_ * ( ) , and make sure what we read is what we expected it
* to be before modifying it .
2014-04-30 13:54:34 +04:00
*/
if ( validate ) {
if ( aarch64_insn_read ( ( void * ) pc , & replaced ) )
return - EFAULT ;
if ( replaced ! = old )
return - EINVAL ;
}
if ( aarch64_insn_patch_text_nosync ( ( void * ) pc , new ) )
return - EPERM ;
return 0 ;
}
/*
* Replace tracer function in ftrace_caller ( )
*/
int ftrace_update_ftrace_func ( ftrace_func_t func )
{
unsigned long pc ;
u32 new ;
pc = ( unsigned long ) & ftrace_call ;
2014-09-19 15:05:45 +04:00
new = aarch64_insn_gen_branch_imm ( pc , ( unsigned long ) func ,
AARCH64_INSN_BRANCH_LINK ) ;
2014-04-30 13:54:34 +04:00
return ftrace_modify_code ( pc , 0 , new , false ) ;
}
/*
* Turn on the call to ftrace_caller ( ) in instrumented function
*/
int ftrace_make_call ( struct dyn_ftrace * rec , unsigned long addr )
{
unsigned long pc = rec - > ip ;
u32 old , new ;
2017-06-12 16:43:25 +03:00
long offset = ( long ) pc - ( long ) addr ;
if ( offset < - SZ_128M | | offset > = SZ_128M ) {
2017-06-23 20:02:06 +03:00
# ifdef CONFIG_ARM64_MODULE_PLTS
2017-06-06 20:00:22 +03:00
struct module * mod ;
2019-10-17 17:26:38 +03:00
/*
* There is only one ftrace trampoline per module . For now ,
* this is not a problem since on arm64 , all dynamic ftrace
* invocations are routed via ftrace_caller ( ) . This will need
* to be revisited if support for multiple ftrace entry points
* is added in the future , but for now , the pr_err ( ) below
* deals with a theoretical issue only .
*/
if ( addr ! = FTRACE_ADDR ) {
pr_err ( " ftrace: far branches to multiple entry points unsupported inside a single module \n " ) ;
return - EINVAL ;
}
2017-06-06 20:00:22 +03:00
/*
* On kernels that support module PLTs , the offset between the
* branch instruction and its target may legally exceed the
* range of an ordinary relative ' bl ' opcode . In this case , we
* need to branch via a trampoline in the module .
*
* NOTE : __module_text_address ( ) must be called with preemption
* disabled , but we can rely on ftrace_lock to ensure that ' mod '
* retains its validity throughout the remainder of this code .
*/
preempt_disable ( ) ;
mod = __module_text_address ( pc ) ;
preempt_enable ( ) ;
if ( WARN_ON ( ! mod ) )
return - EINVAL ;
2019-10-17 17:26:38 +03:00
addr = ( unsigned long ) mod - > arch . ftrace_trampoline ;
2017-06-23 20:02:06 +03:00
# else /* CONFIG_ARM64_MODULE_PLTS */
return - EINVAL ;
2017-06-12 16:43:25 +03:00
# endif /* CONFIG_ARM64_MODULE_PLTS */
2017-06-23 20:02:06 +03:00
}
2017-06-06 20:00:22 +03:00
2014-04-30 13:54:34 +04:00
old = aarch64_insn_gen_nop ( ) ;
2014-09-19 15:05:45 +04:00
new = aarch64_insn_gen_branch_imm ( pc , addr , AARCH64_INSN_BRANCH_LINK ) ;
2014-04-30 13:54:34 +04:00
return ftrace_modify_code ( pc , old , new , true ) ;
}
/*
* Turn off the call to ftrace_caller ( ) in instrumented function
*/
int ftrace_make_nop ( struct module * mod , struct dyn_ftrace * rec ,
unsigned long addr )
{
unsigned long pc = rec - > ip ;
2017-06-06 20:00:21 +03:00
bool validate = true ;
u32 old = 0 , new ;
2017-06-12 16:43:25 +03:00
long offset = ( long ) pc - ( long ) addr ;
if ( offset < - SZ_128M | | offset > = SZ_128M ) {
2017-06-23 20:02:06 +03:00
# ifdef CONFIG_ARM64_MODULE_PLTS
2017-06-06 20:00:21 +03:00
u32 replaced ;
/*
* ' mod ' is only set at module load time , but if we end up
* dealing with an out - of - range condition , we can assume it
* is due to a module being loaded far away from the kernel .
*/
if ( ! mod ) {
preempt_disable ( ) ;
mod = __module_text_address ( pc ) ;
preempt_enable ( ) ;
if ( WARN_ON ( ! mod ) )
return - EINVAL ;
}
/*
* The instruction we are about to patch may be a branch and
* link instruction that was redirected via a PLT entry . In
* this case , the normal validation will fail , but we can at
* least check that we are dealing with a branch and link
* instruction that points into the right module .
*/
if ( aarch64_insn_read ( ( void * ) pc , & replaced ) )
return - EFAULT ;
if ( ! aarch64_insn_is_bl ( replaced ) | |
! within_module ( pc + aarch64_get_branch_offset ( replaced ) ,
mod ) )
return - EINVAL ;
validate = false ;
2017-06-23 20:02:06 +03:00
# else /* CONFIG_ARM64_MODULE_PLTS */
return - EINVAL ;
# endif /* CONFIG_ARM64_MODULE_PLTS */
2017-06-06 20:00:21 +03:00
} else {
old = aarch64_insn_gen_branch_imm ( pc , addr ,
AARCH64_INSN_BRANCH_LINK ) ;
}
2014-04-30 13:54:34 +04:00
new = aarch64_insn_gen_nop ( ) ;
2017-06-06 20:00:21 +03:00
return ftrace_modify_code ( pc , old , new , validate ) ;
2014-04-30 13:54:34 +04:00
}
2015-12-04 06:38:39 +03:00
void arch_ftrace_update_code ( int command )
{
2018-12-05 20:48:54 +03:00
command | = FTRACE_MAY_SLEEP ;
2015-12-04 06:38:39 +03:00
ftrace_modify_all_code ( command ) ;
}
2014-04-30 13:54:34 +04:00
int __init ftrace_dyn_arch_init ( void )
{
return 0 ;
}
# endif /* CONFIG_DYNAMIC_FTRACE */
2014-04-30 13:54:33 +04:00
# ifdef CONFIG_FUNCTION_GRAPH_TRACER
/*
* function_graph tracer expects ftrace_return_to_handler ( ) to be called
* on the way back to parent . For this purpose , this function is called
* in _mcount ( ) or ftrace_caller ( ) to replace return address ( * parent ) on
* the call stack to return_to_handler .
*
* Note that @ frame_pointer is used only for sanity check later .
*/
2018-11-16 01:42:03 +03:00
void prepare_ftrace_return ( unsigned long self_addr , unsigned long * parent ,
2014-04-30 13:54:33 +04:00
unsigned long frame_pointer )
{
unsigned long return_hooker = ( unsigned long ) & return_to_handler ;
unsigned long old ;
if ( unlikely ( atomic_read ( & current - > tracing_graph_pause ) ) )
return ;
/*
* Note :
* No protection against faulting at * parent , which may be seen
* on other archs . It ' s unlikely on AArch64 .
*/
old = * parent ;
2018-11-19 01:21:51 +03:00
if ( ! function_graph_enter ( old , self_addr , frame_pointer , NULL ) )
2015-12-15 11:33:39 +03:00
* parent = return_hooker ;
2014-04-30 13:54:33 +04:00
}
2014-04-30 13:54:34 +04:00
# ifdef CONFIG_DYNAMIC_FTRACE
/*
* Turn on / off the call to ftrace_graph_caller ( ) in ftrace_caller ( )
* depending on @ enable .
*/
static int ftrace_modify_graph_caller ( bool enable )
{
unsigned long pc = ( unsigned long ) & ftrace_graph_call ;
u32 branch , nop ;
branch = aarch64_insn_gen_branch_imm ( pc ,
2014-09-19 15:05:45 +04:00
( unsigned long ) ftrace_graph_caller ,
2015-02-13 07:06:21 +03:00
AARCH64_INSN_BRANCH_NOLINK ) ;
2014-04-30 13:54:34 +04:00
nop = aarch64_insn_gen_nop ( ) ;
if ( enable )
return ftrace_modify_code ( pc , nop , branch , true ) ;
else
return ftrace_modify_code ( pc , branch , nop , true ) ;
}
int ftrace_enable_ftrace_graph_caller ( void )
{
return ftrace_modify_graph_caller ( true ) ;
}
int ftrace_disable_ftrace_graph_caller ( void )
{
return ftrace_modify_graph_caller ( false ) ;
}
# endif /* CONFIG_DYNAMIC_FTRACE */
2014-04-30 13:54:33 +04:00
# endif /* CONFIG_FUNCTION_GRAPH_TRACER */