Skip to content

Commit 3732710

Browse files
jpoimboeIngo Molnar
authored andcommitted
objtool: Improve rare switch jump table pattern detection
GCC 6 added a new switch statement jump table optimization which makes objtool's life harder. It looks like: mov [rodata addr],%reg1 ... some instructions ... jmpq *(%reg1,%reg2,8) The optimization is quite rare, but objtool still needs to be able to identify the pattern so that it can follow all possible control flow paths related to the switch statement. In order to detect the pattern, objtool starts from the indirect jump and scans backwards through the function until it finds the first instruction in the pattern. If it encounters an unconditional jump along the way, it stops and considers the pattern to be not found. As it turns out, unconditional jumps can happen, as long as they are small forward jumps within the range being scanned. This fixes the following warnings: drivers/infiniband/sw/rxe/rxe_comp.o: warning: objtool: rxe_completer()+0x2f4: sibling call from callable instruction with changed frame pointer drivers/infiniband/sw/rxe/rxe_resp.o: warning: objtool: rxe_responder()+0x10f: sibling call from callable instruction with changed frame pointer Reported-by: Arnd Bergmann <arnd@arndb.de> Signed-off-by: Josh Poimboeuf <jpoimboe@redhat.com> Cc: Andy Lutomirski <luto@kernel.org> Cc: Borislav Petkov <bp@alien8.de> Cc: Brian Gerst <brgerst@gmail.com> Cc: Denys Vlasenko <dvlasenk@redhat.com> Cc: H. Peter Anvin <hpa@zytor.com> Cc: Linus Torvalds <torvalds@linux-foundation.org> Cc: Peter Zijlstra <peterz@infradead.org> Cc: Thomas Gleixner <tglx@linutronix.de> Link: http://lkml.kernel.org/r/8a9ed68ae1780e8d3963e4ee13f2f257fe3a3c33.1476393584.git.jpoimboe@redhat.com Signed-off-by: Ingo Molnar <mingo@kernel.org>
1 parent 2cc17fd commit 3732710

File tree

1 file changed

+9
-2
lines changed

1 file changed

+9
-2
lines changed

tools/objtool/builtin-check.c

Lines changed: 9 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -713,6 +713,7 @@ static struct rela *find_switch_table(struct objtool_file *file,
713713
struct instruction *insn)
714714
{
715715
struct rela *text_rela, *rodata_rela;
716+
struct instruction *orig_insn = insn;
716717

717718
text_rela = find_rela_by_dest_range(insn->sec, insn->offset, insn->len);
718719
if (text_rela && text_rela->sym == file->rodata->sym) {
@@ -733,10 +734,16 @@ static struct rela *find_switch_table(struct objtool_file *file,
733734

734735
/* case 3 */
735736
func_for_each_insn_continue_reverse(file, func, insn) {
736-
if (insn->type == INSN_JUMP_UNCONDITIONAL ||
737-
insn->type == INSN_JUMP_DYNAMIC)
737+
if (insn->type == INSN_JUMP_DYNAMIC)
738738
break;
739739

740+
/* allow small jumps within the range */
741+
if (insn->type == INSN_JUMP_UNCONDITIONAL &&
742+
insn->jump_dest &&
743+
(insn->jump_dest->offset <= insn->offset ||
744+
insn->jump_dest->offset >= orig_insn->offset))
745+
break;
746+
740747
text_rela = find_rela_by_dest_range(insn->sec, insn->offset,
741748
insn->len);
742749
if (text_rela && text_rela->sym == file->rodata->sym)

0 commit comments

Comments
 (0)