2020-10-11 06:03:51 +03:00
// SPDX-License-Identifier: GPL-2.0
2017-12-18 12:52:48 +03:00
/*
* Copyright ( C ) 2013 Linaro Limited
* Author : AKASHI Takahiro < takahiro . akashi @ linaro . org >
* Copyright ( C ) 2017 Andes Technology Corporation
*/
# include <linux/ftrace.h>
2018-02-13 08:13:17 +03:00
# include <linux/uaccess.h>
2020-04-21 10:30:01 +03:00
# include <linux/memory.h>
2018-02-13 08:13:17 +03:00
# include <asm/cacheflush.h>
2020-03-09 19:55:44 +03:00
# include <asm/patch.h>
2017-12-18 12:52:48 +03:00
2018-02-13 08:13:17 +03:00
# ifdef CONFIG_DYNAMIC_FTRACE
2020-04-21 10:30:01 +03:00
int ftrace_arch_code_modify_prepare ( void ) __acquires ( & text_mutex )
{
mutex_lock ( & text_mutex ) ;
return 0 ;
}
int ftrace_arch_code_modify_post_process ( void ) __releases ( & text_mutex )
{
mutex_unlock ( & text_mutex ) ;
return 0 ;
}
2018-02-13 08:13:17 +03:00
static int ftrace_check_current_call ( unsigned long hook_pos ,
unsigned int * expected )
{
unsigned int replaced [ 2 ] ;
unsigned int nops [ 2 ] = { NOP4 , NOP4 } ;
/* we expect nops at the hook position */
if ( ! expected )
expected = nops ;
/*
* Read the text we want to modify ;
* return must be - EFAULT on read error
*/
2020-06-17 10:37:53 +03:00
if ( copy_from_kernel_nofault ( replaced , ( void * ) hook_pos ,
MCOUNT_INSN_SIZE ) )
2018-02-13 08:13:17 +03:00
return - EFAULT ;
/*
* Make sure it is what we expect it to be ;
* return must be - EINVAL on failed comparison
*/
if ( memcmp ( expected , replaced , sizeof ( replaced ) ) ) {
2019-01-18 17:03:04 +03:00
pr_err ( " %p: expected (%08x %08x) but got (%08x %08x) \n " ,
2018-02-13 08:13:17 +03:00
( void * ) hook_pos , expected [ 0 ] , expected [ 1 ] , replaced [ 0 ] ,
replaced [ 1 ] ) ;
return - EINVAL ;
}
return 0 ;
}
static int __ftrace_modify_call ( unsigned long hook_pos , unsigned long target ,
bool enable )
{
unsigned int call [ 2 ] ;
unsigned int nops [ 2 ] = { NOP4 , NOP4 } ;
make_call ( hook_pos , target , call ) ;
2020-03-09 19:55:44 +03:00
/* Replace the auipc-jalr pair at once. Return -EPERM on write error. */
2020-04-21 10:29:59 +03:00
if ( patch_text_nosync
2020-03-09 19:55:44 +03:00
( ( void * ) hook_pos , enable ? call : nops , MCOUNT_INSN_SIZE ) )
2018-02-13 08:13:17 +03:00
return - EPERM ;
return 0 ;
}
int ftrace_make_call ( struct dyn_ftrace * rec , unsigned long addr )
{
int ret = ftrace_check_current_call ( rec - > ip , NULL ) ;
if ( ret )
return ret ;
return __ftrace_modify_call ( rec - > ip , addr , true ) ;
}
int ftrace_make_nop ( struct module * mod , struct dyn_ftrace * rec ,
unsigned long addr )
{
unsigned int call [ 2 ] ;
int ret ;
make_call ( rec - > ip , addr , call ) ;
ret = ftrace_check_current_call ( rec - > ip , call ) ;
if ( ret )
return ret ;
return __ftrace_modify_call ( rec - > ip , addr , false ) ;
}
2020-08-25 03:21:22 +03:00
/*
* This is called early on , and isn ' t wrapped by
* ftrace_arch_code_modify_ { prepare , post_process } ( ) and therefor doesn ' t hold
* text_mutex , which triggers a lockdep failure . SMP isn ' t running so we could
* just directly poke the text , but it ' s simpler to just take the lock
* ourselves .
*/
int ftrace_init_nop ( struct module * mod , struct dyn_ftrace * rec )
{
int out ;
ftrace_arch_code_modify_prepare ( ) ;
out = ftrace_make_nop ( mod , rec , MCOUNT_ADDR ) ;
ftrace_arch_code_modify_post_process ( ) ;
return out ;
}
2018-02-13 08:13:17 +03:00
int ftrace_update_ftrace_func ( ftrace_func_t func )
{
int ret = __ftrace_modify_call ( ( unsigned long ) & ftrace_call ,
( unsigned long ) func , true ) ;
if ( ! ret ) {
ret = __ftrace_modify_call ( ( unsigned long ) & ftrace_regs_call ,
( unsigned long ) func , true ) ;
}
return ret ;
}
int __init ftrace_dyn_arch_init ( void )
{
return 0 ;
}
# endif
2018-02-13 08:13:20 +03:00
# ifdef CONFIG_DYNAMIC_FTRACE_WITH_REGS
int ftrace_modify_call ( struct dyn_ftrace * rec , unsigned long old_addr ,
unsigned long addr )
{
unsigned int call [ 2 ] ;
int ret ;
make_call ( rec - > ip , old_addr , call ) ;
ret = ftrace_check_current_call ( rec - > ip , call ) ;
if ( ret )
return ret ;
return __ftrace_modify_call ( rec - > ip , addr , true ) ;
}
# endif
2018-02-13 08:13:17 +03:00
# ifdef CONFIG_FUNCTION_GRAPH_TRACER
2017-12-18 12:52:48 +03:00
/*
2018-02-13 08:13:17 +03:00
* Most of this function is copied from arm64 .
2017-12-18 12:52:48 +03:00
*/
void prepare_ftrace_return ( unsigned long * parent , unsigned long self_addr ,
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 ;
/*
* We don ' t suffer access faults , so no extra fault - recovery assembly
* is needed here .
*/
old = * parent ;
2019-12-23 11:46:13 +03:00
if ( ! function_graph_enter ( old , self_addr , frame_pointer , parent ) )
2018-11-19 01:31:44 +03:00
* parent = return_hooker ;
2017-12-18 12:52:48 +03:00
}
2018-02-13 08:13:18 +03:00
# ifdef CONFIG_DYNAMIC_FTRACE
extern void ftrace_graph_call ( void ) ;
int ftrace_enable_ftrace_graph_caller ( void )
{
unsigned int call [ 2 ] ;
static int init_graph = 1 ;
int ret ;
make_call ( & ftrace_graph_call , & ftrace_stub , call ) ;
/*
* When enabling graph tracer for the first time , ftrace_graph_call
* should contains a call to ftrace_stub . Once it has been disabled ,
* the 8 - bytes at the position becomes NOPs .
*/
if ( init_graph ) {
ret = ftrace_check_current_call ( ( unsigned long ) & ftrace_graph_call ,
call ) ;
init_graph = 0 ;
} else {
ret = ftrace_check_current_call ( ( unsigned long ) & ftrace_graph_call ,
NULL ) ;
}
if ( ret )
return ret ;
return __ftrace_modify_call ( ( unsigned long ) & ftrace_graph_call ,
( unsigned long ) & prepare_ftrace_return , true ) ;
}
int ftrace_disable_ftrace_graph_caller ( void )
{
unsigned int call [ 2 ] ;
int ret ;
make_call ( & ftrace_graph_call , & prepare_ftrace_return , call ) ;
/*
* This is to make sure that ftrace_enable_ftrace_graph_caller
* did the right thing .
*/
ret = ftrace_check_current_call ( ( unsigned long ) & ftrace_graph_call ,
call ) ;
if ( ret )
return ret ;
return __ftrace_modify_call ( ( unsigned long ) & ftrace_graph_call ,
( unsigned long ) & prepare_ftrace_return , false ) ;
}
# endif /* CONFIG_DYNAMIC_FTRACE */
# endif /* CONFIG_FUNCTION_GRAPH_TRACER */