1 // SPDX-License-Identifier: GPL-2.0
3 * Copyright (C) 2022 Loongson Technology Corporation Limited
6 #include <linux/init.h>
7 #include <linux/ftrace.h>
8 #include <linux/syscalls.h>
9 #include <linux/uaccess.h>
12 #include <asm/asm-offsets.h>
13 #include <asm/cacheflush.h>
15 #include <asm/loongarch.h>
16 #include <asm/syscall.h>
18 #include <asm-generic/sections.h>
20 #ifdef CONFIG_FUNCTION_GRAPH_TRACER
23 * As `call _mcount` follows LoongArch psABI, ra-saved operation and
24 * stack operation can be found before this insn.
27 static int ftrace_get_parent_ra_addr(unsigned long insn_addr, int *ra_off)
30 union loongarch_instruction *insn;
32 insn = (union loongarch_instruction *)insn_addr;
38 if (is_ra_save_ins(insn))
39 *ra_off = -((1 << 12) - insn->reg2i12_format.immediate);
41 } while (!is_stack_alloc_ins(insn) && limit);
49 void prepare_ftrace_return(unsigned long self_addr,
50 unsigned long callsite_sp, unsigned long old)
53 unsigned long return_hooker = (unsigned long)&return_to_handler;
55 if (unlikely(ftrace_graph_is_dead()))
58 if (unlikely(atomic_read(¤t->tracing_graph_pause)))
61 if (ftrace_get_parent_ra_addr(self_addr, &ra_off))
64 if (!function_graph_enter(old, self_addr, 0, NULL))
65 *(unsigned long *)(callsite_sp + ra_off) = return_hooker;
73 #endif /* CONFIG_FUNCTION_GRAPH_TRACER */