mirror of
				https://github.com/torvalds/linux.git
				synced 2025-11-04 02:30:34 +02:00 
			
		
		
		
	Most of ARCHs use empty ftrace_dyn_arch_init(), introduce a weak common ftrace_dyn_arch_init() to cleanup them. Link: https://lkml.kernel.org/r/20210909090216.1955240-1-o451686892@gmail.com Acked-by: Heiko Carstens <hca@linux.ibm.com> (s390) Acked-by: Helge Deller <deller@gmx.de> (parisc) Signed-off-by: Weizhao Ouyang <o451686892@gmail.com> Signed-off-by: Steven Rostedt (VMware) <rostedt@goodmis.org>
		
			
				
	
	
		
			234 lines
		
	
	
	
		
			5.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			234 lines
		
	
	
	
		
			5.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
// SPDX-License-Identifier: GPL-2.0
 | 
						|
// Copyright (C) 2018 Hangzhou C-SKY Microsystems co.,ltd.
 | 
						|
 | 
						|
#include <linux/ftrace.h>
 | 
						|
#include <linux/uaccess.h>
 | 
						|
#include <linux/stop_machine.h>
 | 
						|
#include <asm/cacheflush.h>
 | 
						|
 | 
						|
#ifdef CONFIG_DYNAMIC_FTRACE
 | 
						|
 | 
						|
#define NOP		0x4000
 | 
						|
#define NOP32_HI	0xc400
 | 
						|
#define NOP32_LO	0x4820
 | 
						|
#define PUSH_LR		0x14d0
 | 
						|
#define MOVIH_LINK	0xea3a
 | 
						|
#define ORI_LINK	0xef5a
 | 
						|
#define JSR_LINK	0xe8fa
 | 
						|
#define BSR_LINK	0xe000
 | 
						|
 | 
						|
/*
 | 
						|
 * Gcc-csky with -pg will insert stub in function prologue:
 | 
						|
 *	push	lr
 | 
						|
 *	jbsr	_mcount
 | 
						|
 *	nop32
 | 
						|
 *	nop32
 | 
						|
 *
 | 
						|
 * If the (callee - current_pc) is less then 64MB, we'll use bsr:
 | 
						|
 *	push	lr
 | 
						|
 *	bsr	_mcount
 | 
						|
 *	nop32
 | 
						|
 *	nop32
 | 
						|
 * else we'll use (movih + ori + jsr):
 | 
						|
 *	push	lr
 | 
						|
 *	movih	r26, ...
 | 
						|
 *	ori	r26, ...
 | 
						|
 *	jsr	r26
 | 
						|
 *
 | 
						|
 * (r26 is our reserved link-reg)
 | 
						|
 *
 | 
						|
 */
 | 
						|
static inline void make_jbsr(unsigned long callee, unsigned long pc,
 | 
						|
			     uint16_t *call, bool nolr)
 | 
						|
{
 | 
						|
	long offset;
 | 
						|
 | 
						|
	call[0]	= nolr ? NOP : PUSH_LR;
 | 
						|
 | 
						|
	offset = (long) callee - (long) pc;
 | 
						|
 | 
						|
	if (unlikely(offset < -67108864 || offset > 67108864)) {
 | 
						|
		call[1] = MOVIH_LINK;
 | 
						|
		call[2] = callee >> 16;
 | 
						|
		call[3] = ORI_LINK;
 | 
						|
		call[4] = callee & 0xffff;
 | 
						|
		call[5] = JSR_LINK;
 | 
						|
		call[6] = 0;
 | 
						|
	} else {
 | 
						|
		offset = offset >> 1;
 | 
						|
 | 
						|
		call[1] = BSR_LINK |
 | 
						|
			 ((uint16_t)((unsigned long) offset >> 16) & 0x3ff);
 | 
						|
		call[2] = (uint16_t)((unsigned long) offset & 0xffff);
 | 
						|
		call[3] = call[5] = NOP32_HI;
 | 
						|
		call[4] = call[6] = NOP32_LO;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static uint16_t nops[7] = {NOP, NOP32_HI, NOP32_LO, NOP32_HI, NOP32_LO,
 | 
						|
				NOP32_HI, NOP32_LO};
 | 
						|
static int ftrace_check_current_nop(unsigned long hook)
 | 
						|
{
 | 
						|
	uint16_t olds[7];
 | 
						|
	unsigned long hook_pos = hook - 2;
 | 
						|
 | 
						|
	if (copy_from_kernel_nofault((void *)olds, (void *)hook_pos,
 | 
						|
			sizeof(nops)))
 | 
						|
		return -EFAULT;
 | 
						|
 | 
						|
	if (memcmp((void *)nops, (void *)olds, sizeof(nops))) {
 | 
						|
		pr_err("%p: nop but get (%04x %04x %04x %04x %04x %04x %04x)\n",
 | 
						|
			(void *)hook_pos,
 | 
						|
			olds[0], olds[1], olds[2], olds[3], olds[4], olds[5],
 | 
						|
			olds[6]);
 | 
						|
 | 
						|
		return -EINVAL;
 | 
						|
	}
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static int ftrace_modify_code(unsigned long hook, unsigned long target,
 | 
						|
			      bool enable, bool nolr)
 | 
						|
{
 | 
						|
	uint16_t call[7];
 | 
						|
 | 
						|
	unsigned long hook_pos = hook - 2;
 | 
						|
	int ret = 0;
 | 
						|
 | 
						|
	make_jbsr(target, hook, call, nolr);
 | 
						|
 | 
						|
	ret = copy_to_kernel_nofault((void *)hook_pos, enable ? call : nops,
 | 
						|
				 sizeof(nops));
 | 
						|
	if (ret)
 | 
						|
		return -EPERM;
 | 
						|
 | 
						|
	flush_icache_range(hook_pos, hook_pos + MCOUNT_INSN_SIZE);
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
int ftrace_make_call(struct dyn_ftrace *rec, unsigned long addr)
 | 
						|
{
 | 
						|
	int ret = ftrace_check_current_nop(rec->ip);
 | 
						|
 | 
						|
	if (ret)
 | 
						|
		return ret;
 | 
						|
 | 
						|
	return ftrace_modify_code(rec->ip, addr, true, false);
 | 
						|
}
 | 
						|
 | 
						|
int ftrace_make_nop(struct module *mod, struct dyn_ftrace *rec,
 | 
						|
		    unsigned long addr)
 | 
						|
{
 | 
						|
	return ftrace_modify_code(rec->ip, addr, false, false);
 | 
						|
}
 | 
						|
 | 
						|
int ftrace_update_ftrace_func(ftrace_func_t func)
 | 
						|
{
 | 
						|
	int ret = ftrace_modify_code((unsigned long)&ftrace_call,
 | 
						|
				(unsigned long)func, true, true);
 | 
						|
	if (!ret)
 | 
						|
		ret = ftrace_modify_code((unsigned long)&ftrace_regs_call,
 | 
						|
				(unsigned long)func, true, true);
 | 
						|
	return ret;
 | 
						|
}
 | 
						|
#endif /* CONFIG_DYNAMIC_FTRACE */
 | 
						|
 | 
						|
#ifdef CONFIG_DYNAMIC_FTRACE_WITH_REGS
 | 
						|
int ftrace_modify_call(struct dyn_ftrace *rec, unsigned long old_addr,
 | 
						|
		       unsigned long addr)
 | 
						|
{
 | 
						|
	return ftrace_modify_code(rec->ip, addr, true, true);
 | 
						|
}
 | 
						|
#endif
 | 
						|
 | 
						|
#ifdef CONFIG_FUNCTION_GRAPH_TRACER
 | 
						|
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(¤t->tracing_graph_pause)))
 | 
						|
		return;
 | 
						|
 | 
						|
	old = *parent;
 | 
						|
 | 
						|
	if (!function_graph_enter(old, self_addr,
 | 
						|
			*(unsigned long *)frame_pointer, parent)) {
 | 
						|
		/*
 | 
						|
		 * For csky-gcc function has sub-call:
 | 
						|
		 * subi	sp,	sp, 8
 | 
						|
		 * stw	r8,	(sp, 0)
 | 
						|
		 * mov	r8,	sp
 | 
						|
		 * st.w r15,	(sp, 0x4)
 | 
						|
		 * push	r15
 | 
						|
		 * jl	_mcount
 | 
						|
		 * We only need set *parent for resume
 | 
						|
		 *
 | 
						|
		 * For csky-gcc function has no sub-call:
 | 
						|
		 * subi	sp,	sp, 4
 | 
						|
		 * stw	r8,	(sp, 0)
 | 
						|
		 * mov	r8,	sp
 | 
						|
		 * push	r15
 | 
						|
		 * jl	_mcount
 | 
						|
		 * We need set *parent and *(frame_pointer + 4) for resume,
 | 
						|
		 * because lr is resumed twice.
 | 
						|
		 */
 | 
						|
		*parent = return_hooker;
 | 
						|
		frame_pointer += 4;
 | 
						|
		if (*(unsigned long *)frame_pointer == old)
 | 
						|
			*(unsigned long *)frame_pointer = return_hooker;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
#ifdef CONFIG_DYNAMIC_FTRACE
 | 
						|
int ftrace_enable_ftrace_graph_caller(void)
 | 
						|
{
 | 
						|
	return ftrace_modify_code((unsigned long)&ftrace_graph_call,
 | 
						|
			(unsigned long)&ftrace_graph_caller, true, true);
 | 
						|
}
 | 
						|
 | 
						|
int ftrace_disable_ftrace_graph_caller(void)
 | 
						|
{
 | 
						|
	return ftrace_modify_code((unsigned long)&ftrace_graph_call,
 | 
						|
			(unsigned long)&ftrace_graph_caller, false, true);
 | 
						|
}
 | 
						|
#endif /* CONFIG_DYNAMIC_FTRACE */
 | 
						|
#endif /* CONFIG_FUNCTION_GRAPH_TRACER */
 | 
						|
 | 
						|
#ifdef CONFIG_DYNAMIC_FTRACE
 | 
						|
#ifndef CONFIG_CPU_HAS_ICACHE_INS
 | 
						|
struct ftrace_modify_param {
 | 
						|
	int command;
 | 
						|
	atomic_t cpu_count;
 | 
						|
};
 | 
						|
 | 
						|
static int __ftrace_modify_code(void *data)
 | 
						|
{
 | 
						|
	struct ftrace_modify_param *param = data;
 | 
						|
 | 
						|
	if (atomic_inc_return(¶m->cpu_count) == 1) {
 | 
						|
		ftrace_modify_all_code(param->command);
 | 
						|
		atomic_inc(¶m->cpu_count);
 | 
						|
	} else {
 | 
						|
		while (atomic_read(¶m->cpu_count) <= num_online_cpus())
 | 
						|
			cpu_relax();
 | 
						|
		local_icache_inv_all(NULL);
 | 
						|
	}
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
void arch_ftrace_update_code(int command)
 | 
						|
{
 | 
						|
	struct ftrace_modify_param param = { command, ATOMIC_INIT(0) };
 | 
						|
 | 
						|
	stop_machine(__ftrace_modify_code, ¶m, cpu_online_mask);
 | 
						|
}
 | 
						|
#endif
 | 
						|
#endif /* CONFIG_DYNAMIC_FTRACE */
 | 
						|
 | 
						|
/* _mcount is defined in abi's mcount.S */
 | 
						|
EXPORT_SYMBOL(_mcount);
 |