mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/netdev/net.git/
synced 2026-04-03 23:37:40 -04:00
Mahe reported issue with bpf_override_return helper not working when executed from kprobe.multi bpf program on arm. The problem is that on arm we use alternate storage for pt_regs object that is passed to bpf_prog_run and if any register is changed (which is the case of bpf_override_return) it's not propagated back to actual pt_regs object. Fixing this by introducing and calling ftrace_partial_regs_update function to propagate the values of changed registers (ip and stack). Reported-by: Mahe Tardy <mahe.tardy@gmail.com> Signed-off-by: Jiri Olsa <jolsa@kernel.org> Signed-off-by: Andrii Nakryiko <andrii@kernel.org> Reviewed-by: Steven Rostedt (Google) <rostedt@goodmis.org> Acked-by: Will Deacon <will@kernel.org> Link: https://lore.kernel.org/bpf/20260112121157.854473-1-jolsa@kernel.org
69 lines
2.4 KiB
C
69 lines
2.4 KiB
C
/* SPDX-License-Identifier: GPL-2.0 */
|
|
#ifndef _LINUX_FTRACE_REGS_H
|
|
#define _LINUX_FTRACE_REGS_H
|
|
|
|
/*
|
|
* For archs that just copy pt_regs in ftrace regs, it can use this default.
|
|
* If an architecture does not use pt_regs, it must define all the below
|
|
* accessor functions.
|
|
*/
|
|
#ifndef HAVE_ARCH_FTRACE_REGS
|
|
struct __arch_ftrace_regs {
|
|
struct pt_regs regs;
|
|
};
|
|
|
|
#define arch_ftrace_regs(fregs) ((struct __arch_ftrace_regs *)(fregs))
|
|
|
|
struct ftrace_regs;
|
|
|
|
#define ftrace_regs_get_instruction_pointer(fregs) \
|
|
instruction_pointer(&arch_ftrace_regs(fregs)->regs)
|
|
#define ftrace_regs_get_argument(fregs, n) \
|
|
regs_get_kernel_argument(&arch_ftrace_regs(fregs)->regs, n)
|
|
#define ftrace_regs_get_stack_pointer(fregs) \
|
|
kernel_stack_pointer(&arch_ftrace_regs(fregs)->regs)
|
|
#define ftrace_regs_get_return_value(fregs) \
|
|
regs_return_value(&arch_ftrace_regs(fregs)->regs)
|
|
#define ftrace_regs_set_return_value(fregs, ret) \
|
|
regs_set_return_value(&arch_ftrace_regs(fregs)->regs, ret)
|
|
#define ftrace_override_function_with_return(fregs) \
|
|
override_function_with_return(&arch_ftrace_regs(fregs)->regs)
|
|
#define ftrace_regs_query_register_offset(name) \
|
|
regs_query_register_offset(name)
|
|
#define ftrace_regs_get_frame_pointer(fregs) \
|
|
frame_pointer(&arch_ftrace_regs(fregs)->regs)
|
|
|
|
static __always_inline void
|
|
ftrace_partial_regs_update(struct ftrace_regs *fregs, struct pt_regs *regs) { }
|
|
|
|
#else
|
|
|
|
/*
|
|
* ftrace_partial_regs_update - update the original ftrace_regs from regs
|
|
* @fregs: The ftrace_regs to update from @regs
|
|
* @regs: The partial regs from ftrace_partial_regs() that was updated
|
|
*
|
|
* Some architectures have the partial regs living in the ftrace_regs
|
|
* structure, whereas other architectures need to make a different copy
|
|
* of the @regs. If a partial @regs is retrieved by ftrace_partial_regs() and
|
|
* if the code using @regs updates a field (like the instruction pointer or
|
|
* stack pointer) it may need to propagate that change to the original @fregs
|
|
* it retrieved the partial @regs from. Use this function to guarantee that
|
|
* update happens.
|
|
*/
|
|
static __always_inline void
|
|
ftrace_partial_regs_update(struct ftrace_regs *fregs, struct pt_regs *regs)
|
|
{
|
|
ftrace_regs_set_instruction_pointer(fregs, instruction_pointer(regs));
|
|
ftrace_regs_set_return_value(fregs, regs_return_value(regs));
|
|
}
|
|
|
|
#endif /* HAVE_ARCH_FTRACE_REGS */
|
|
|
|
/* This can be overridden by the architectures */
|
|
#ifndef FTRACE_REGS_MAX_ARGS
|
|
# define FTRACE_REGS_MAX_ARGS 6
|
|
#endif
|
|
|
|
#endif /* _LINUX_FTRACE_REGS_H */
|