Commit cbf82a3d authored by Peter Zijlstra's avatar Peter Zijlstra Committed by Ingo Molnar

objtool: Decode jump_entry::key addend

Teach objtool about the the low bits in the struct static_key pointer.

That is, the low two bits of @key in:

  struct jump_entry {
	s32 code;
	s32 target;
	long key;
  }

as found in the __jump_table section. Since @key has a relocation to
the variable (to be resolved by the linker), the low two bits will be
reflected in the relocation's addend.

As such, find the reloc and store the addend, such that we can access
these bits.
Signed-off-by: default avatarPeter Zijlstra (Intel) <peterz@infradead.org>
Signed-off-by: default avatarIngo Molnar <mingo@kernel.org>
Link: https://lore.kernel.org/r/20210506194158.028024143@infradead.org
parent e7bf1ba9
...@@ -9,6 +9,7 @@ ...@@ -9,6 +9,7 @@
#define JUMP_ENTRY_SIZE 16 #define JUMP_ENTRY_SIZE 16
#define JUMP_ORIG_OFFSET 0 #define JUMP_ORIG_OFFSET 0
#define JUMP_NEW_OFFSET 4 #define JUMP_NEW_OFFSET 4
#define JUMP_KEY_OFFSET 8
#define ALT_ENTRY_SIZE 12 #define ALT_ENTRY_SIZE 12
#define ALT_ORIG_OFFSET 0 #define ALT_ORIG_OFFSET 0
......
...@@ -27,6 +27,7 @@ struct special_alt { ...@@ -27,6 +27,7 @@ struct special_alt {
unsigned long new_off; unsigned long new_off;
unsigned int orig_len, new_len; /* group only */ unsigned int orig_len, new_len; /* group only */
u8 key_addend;
}; };
int special_get_alts(struct elf *elf, struct list_head *alts); int special_get_alts(struct elf *elf, struct list_head *alts);
......
...@@ -23,6 +23,7 @@ struct special_entry { ...@@ -23,6 +23,7 @@ struct special_entry {
unsigned char size, orig, new; unsigned char size, orig, new;
unsigned char orig_len, new_len; /* group only */ unsigned char orig_len, new_len; /* group only */
unsigned char feature; /* ALTERNATIVE macro CPU feature */ unsigned char feature; /* ALTERNATIVE macro CPU feature */
unsigned char key; /* jump_label key */
}; };
struct special_entry entries[] = { struct special_entry entries[] = {
...@@ -42,6 +43,7 @@ struct special_entry entries[] = { ...@@ -42,6 +43,7 @@ struct special_entry entries[] = {
.size = JUMP_ENTRY_SIZE, .size = JUMP_ENTRY_SIZE,
.orig = JUMP_ORIG_OFFSET, .orig = JUMP_ORIG_OFFSET,
.new = JUMP_NEW_OFFSET, .new = JUMP_NEW_OFFSET,
.key = JUMP_KEY_OFFSET,
}, },
{ {
.sec = "__ex_table", .sec = "__ex_table",
...@@ -122,6 +124,18 @@ static int get_alt_entry(struct elf *elf, struct special_entry *entry, ...@@ -122,6 +124,18 @@ static int get_alt_entry(struct elf *elf, struct special_entry *entry,
alt->new_off -= 0x7ffffff0; alt->new_off -= 0x7ffffff0;
} }
if (entry->key) {
struct reloc *key_reloc;
key_reloc = find_reloc_by_dest(elf, sec, offset + entry->key);
if (!key_reloc) {
WARN_FUNC("can't find key reloc",
sec, offset + entry->key);
return -1;
}
alt->key_addend = key_reloc->addend;
}
return 0; return 0;
} }
......
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