Commit d2bf98e6 authored by Steven J. Magnani's avatar Steven J. Magnani Committed by Michal Simek

microblaze: Fix ftrace_update_ftrace_func panic

The Microblaze dynamic ftrace code assumes a call ordering that is not met
in all scenarios. Specifically, executing a command similar to:

  echo 105 > /sys/kernel/debug/tracing/set_ftrace_pid

before any other tracing-related commands results in a kernel panic:

  BUG: failure at arch/microblaze/kernel/ftrace.c:198/ftrace_update_ftrace_func()!

Recoding ftrace_update_ftrace_func() to use &ftrace_caller directly eliminates
the need to capture its address elsewhere (and thus rely on a particular call
sequence).
Signed-off-by: default avatarSteven J. Magnani <steve@digidescorp.com>
Signed-off-by: default avatarMichal Simek <monstr@monstr.eu>
parent 0fdf8675
...@@ -151,13 +151,10 @@ int ftrace_make_nop(struct module *mod, ...@@ -151,13 +151,10 @@ int ftrace_make_nop(struct module *mod,
return ret; return ret;
} }
static int ret_addr; /* initialized as 0 by default */
/* I believe that first is called ftrace_make_nop before this function */ /* I believe that first is called ftrace_make_nop before this function */
int ftrace_make_call(struct dyn_ftrace *rec, unsigned long addr) int ftrace_make_call(struct dyn_ftrace *rec, unsigned long addr)
{ {
int ret; int ret;
ret_addr = addr; /* saving where the barrier jump is */
pr_debug("%s: addr:0x%x, rec->ip: 0x%x, imm:0x%x\n", pr_debug("%s: addr:0x%x, rec->ip: 0x%x, imm:0x%x\n",
__func__, (unsigned int)addr, (unsigned int)rec->ip, imm); __func__, (unsigned int)addr, (unsigned int)rec->ip, imm);
ret = ftrace_modify_code(rec->ip, imm); ret = ftrace_modify_code(rec->ip, imm);
...@@ -194,12 +191,9 @@ int ftrace_update_ftrace_func(ftrace_func_t func) ...@@ -194,12 +191,9 @@ int ftrace_update_ftrace_func(ftrace_func_t func)
ret = ftrace_modify_code(ip, upper); ret = ftrace_modify_code(ip, upper);
ret += ftrace_modify_code(ip + 4, lower); ret += ftrace_modify_code(ip + 4, lower);
/* We just need to remove the rtsd r15, 8 by NOP */ /* We just need to replace the rtsd r15, 8 with NOP */
BUG_ON(!ret_addr); ret += ftrace_modify_code((unsigned long)&ftrace_caller,
if (ret_addr) MICROBLAZE_NOP);
ret += ftrace_modify_code(ret_addr, MICROBLAZE_NOP);
else
ret = 1; /* fault */
/* All changes are done - lets do caches consistent */ /* All changes are done - lets do caches consistent */
flush_icache(); flush_icache();
......
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