Commit 03bac0df authored by Masami Hiramatsu's avatar Masami Hiramatsu Committed by Steven Rostedt (VMware)

kprobes: Add kretprobe_find_ret_addr() for searching return address

Introduce kretprobe_find_ret_addr() and is_kretprobe_trampoline().
These APIs will be used by the ORC stack unwinder and ftrace, so that
they can check whether the given address points kretprobe trampoline
code and query the correct return address in that case.

Link: https://lkml.kernel.org/r/163163046461.489837.1044778356430293962.stgit@devnote2Signed-off-by: default avatarMasami Hiramatsu <mhiramat@kernel.org>
Tested-by: default avatarAndrii Nakryiko <andrii@kernel.org>
Signed-off-by: default avatarSteven Rostedt (VMware) <rostedt@goodmis.org>
parent adf8a61a
...@@ -505,6 +505,28 @@ static inline bool is_kprobe_optinsn_slot(unsigned long addr) ...@@ -505,6 +505,28 @@ static inline bool is_kprobe_optinsn_slot(unsigned long addr)
} }
#endif /* !CONFIG_OPTPROBES */ #endif /* !CONFIG_OPTPROBES */
#ifdef CONFIG_KRETPROBES
static nokprobe_inline bool is_kretprobe_trampoline(unsigned long addr)
{
return (void *)addr == kretprobe_trampoline_addr();
}
unsigned long kretprobe_find_ret_addr(struct task_struct *tsk, void *fp,
struct llist_node **cur);
#else
static nokprobe_inline bool is_kretprobe_trampoline(unsigned long addr)
{
return false;
}
static nokprobe_inline
unsigned long kretprobe_find_ret_addr(struct task_struct *tsk, void *fp,
struct llist_node **cur)
{
return 0;
}
#endif
/* Returns true if kprobes handled the fault */ /* Returns true if kprobes handled the fault */
static nokprobe_inline bool kprobe_page_fault(struct pt_regs *regs, static nokprobe_inline bool kprobe_page_fault(struct pt_regs *regs,
unsigned int trap) unsigned int trap)
......
...@@ -1863,45 +1863,87 @@ static struct notifier_block kprobe_exceptions_nb = { ...@@ -1863,45 +1863,87 @@ static struct notifier_block kprobe_exceptions_nb = {
#ifdef CONFIG_KRETPROBES #ifdef CONFIG_KRETPROBES
unsigned long __kretprobe_trampoline_handler(struct pt_regs *regs, /* This assumes the 'tsk' is the current task or the is not running. */
void *frame_pointer) static kprobe_opcode_t *__kretprobe_find_ret_addr(struct task_struct *tsk,
struct llist_node **cur)
{ {
kprobe_opcode_t *correct_ret_addr = NULL;
struct kretprobe_instance *ri = NULL; struct kretprobe_instance *ri = NULL;
struct llist_node *first, *node; struct llist_node *node = *cur;
struct kretprobe *rp;
if (!node)
node = tsk->kretprobe_instances.first;
else
node = node->next;
/* Find all nodes for this frame. */
first = node = current->kretprobe_instances.first;
while (node) { while (node) {
ri = container_of(node, struct kretprobe_instance, llist); ri = container_of(node, struct kretprobe_instance, llist);
BUG_ON(ri->fp != frame_pointer);
if (ri->ret_addr != kretprobe_trampoline_addr()) { if (ri->ret_addr != kretprobe_trampoline_addr()) {
correct_ret_addr = ri->ret_addr; *cur = node;
/* return ri->ret_addr;
* This is the real return address. Any other
* instances associated with this task are for
* other calls deeper on the call stack
*/
goto found;
} }
node = node->next; node = node->next;
} }
pr_err("kretprobe: Return address not found, not execute handler. Maybe there is a bug in the kernel.\n"); return NULL;
BUG_ON(1); }
NOKPROBE_SYMBOL(__kretprobe_find_ret_addr);
found: /**
/* Unlink all nodes for this frame. */ * kretprobe_find_ret_addr -- Find correct return address modified by kretprobe
current->kretprobe_instances.first = node->next; * @tsk: Target task
node->next = NULL; * @fp: A frame pointer
* @cur: a storage of the loop cursor llist_node pointer for next call
*
* Find the correct return address modified by a kretprobe on @tsk in unsigned
* long type. If it finds the return address, this returns that address value,
* or this returns 0.
* The @tsk must be 'current' or a task which is not running. @fp is a hint
* to get the currect return address - which is compared with the
* kretprobe_instance::fp field. The @cur is a loop cursor for searching the
* kretprobe return addresses on the @tsk. The '*@cur' should be NULL at the
* first call, but '@cur' itself must NOT NULL.
*/
unsigned long kretprobe_find_ret_addr(struct task_struct *tsk, void *fp,
struct llist_node **cur)
{
struct kretprobe_instance *ri = NULL;
kprobe_opcode_t *ret;
if (WARN_ON_ONCE(!cur))
return 0;
do {
ret = __kretprobe_find_ret_addr(tsk, cur);
if (!ret)
break;
ri = container_of(*cur, struct kretprobe_instance, llist);
} while (ri->fp != fp);
/* Run them.. */ return (unsigned long)ret;
}
NOKPROBE_SYMBOL(kretprobe_find_ret_addr);
unsigned long __kretprobe_trampoline_handler(struct pt_regs *regs,
void *frame_pointer)
{
kprobe_opcode_t *correct_ret_addr = NULL;
struct kretprobe_instance *ri = NULL;
struct llist_node *first, *node = NULL;
struct kretprobe *rp;
/* Find correct address and all nodes for this frame. */
correct_ret_addr = __kretprobe_find_ret_addr(current, &node);
if (!correct_ret_addr) {
pr_err("kretprobe: Return address not found, not execute handler. Maybe there is a bug in the kernel.\n");
BUG_ON(1);
}
/* Run the user handler of the nodes. */
first = current->kretprobe_instances.first;
while (first) { while (first) {
ri = container_of(first, struct kretprobe_instance, llist); ri = container_of(first, struct kretprobe_instance, llist);
first = first->next;
if (WARN_ON_ONCE(ri->fp != frame_pointer))
break;
rp = get_kretprobe(ri); rp = get_kretprobe(ri);
if (rp && rp->handler) { if (rp && rp->handler) {
...@@ -1912,6 +1954,21 @@ unsigned long __kretprobe_trampoline_handler(struct pt_regs *regs, ...@@ -1912,6 +1954,21 @@ unsigned long __kretprobe_trampoline_handler(struct pt_regs *regs,
rp->handler(ri, regs); rp->handler(ri, regs);
__this_cpu_write(current_kprobe, prev); __this_cpu_write(current_kprobe, prev);
} }
if (first == node)
break;
first = first->next;
}
/* Unlink all nodes for this frame. */
first = current->kretprobe_instances.first;
current->kretprobe_instances.first = node->next;
node->next = NULL;
/* Recycle free instances. */
while (first) {
ri = container_of(first, struct kretprobe_instance, llist);
first = first->next;
recycle_rp_inst(ri); recycle_rp_inst(ri);
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment