diff options
Diffstat (limited to 'arch/s390/kernel/jump_label.c')
-rw-r--r-- | arch/s390/kernel/jump_label.c | 56 |
1 files changed, 42 insertions, 14 deletions
diff --git a/arch/s390/kernel/jump_label.c b/arch/s390/kernel/jump_label.c index b987ab2c1541..25aef40584f7 100644 --- a/arch/s390/kernel/jump_label.c +++ b/arch/s390/kernel/jump_label.c | |||
@@ -22,31 +22,59 @@ struct insn_args { | |||
22 | enum jump_label_type type; | 22 | enum jump_label_type type; |
23 | }; | 23 | }; |
24 | 24 | ||
25 | static void jump_label_make_nop(struct jump_entry *entry, struct insn *insn) | ||
26 | { | ||
27 | /* brcl 0,0 */ | ||
28 | insn->opcode = 0xc004; | ||
29 | insn->offset = 0; | ||
30 | } | ||
31 | |||
32 | static void jump_label_make_branch(struct jump_entry *entry, struct insn *insn) | ||
33 | { | ||
34 | /* brcl 15,offset */ | ||
35 | insn->opcode = 0xc0f4; | ||
36 | insn->offset = (entry->target - entry->code) >> 1; | ||
37 | } | ||
38 | |||
39 | static void jump_label_bug(struct jump_entry *entry, struct insn *insn) | ||
40 | { | ||
41 | unsigned char *ipc = (unsigned char *)entry->code; | ||
42 | unsigned char *ipe = (unsigned char *)insn; | ||
43 | |||
44 | pr_emerg("Jump label code mismatch at %pS [%p]\n", ipc, ipc); | ||
45 | pr_emerg("Found: %02x %02x %02x %02x %02x %02x\n", | ||
46 | ipc[0], ipc[1], ipc[2], ipc[3], ipc[4], ipc[5]); | ||
47 | pr_emerg("Expected: %02x %02x %02x %02x %02x %02x\n", | ||
48 | ipe[0], ipe[1], ipe[2], ipe[3], ipe[4], ipe[5]); | ||
49 | panic("Corrupted kernel text"); | ||
50 | } | ||
51 | |||
25 | static void __jump_label_transform(struct jump_entry *entry, | 52 | static void __jump_label_transform(struct jump_entry *entry, |
26 | enum jump_label_type type) | 53 | enum jump_label_type type, |
54 | int init) | ||
27 | { | 55 | { |
28 | struct insn insn; | 56 | struct insn old, new; |
29 | int rc; | ||
30 | 57 | ||
31 | if (type == JUMP_LABEL_ENABLE) { | 58 | if (type == JUMP_LABEL_ENABLE) { |
32 | /* brcl 15,offset */ | 59 | jump_label_make_nop(entry, &old); |
33 | insn.opcode = 0xc0f4; | 60 | jump_label_make_branch(entry, &new); |
34 | insn.offset = (entry->target - entry->code) >> 1; | ||
35 | } else { | 61 | } else { |
36 | /* brcl 0,0 */ | 62 | if (init) |
37 | insn.opcode = 0xc004; | 63 | jump_label_make_nop(entry, &old); |
38 | insn.offset = 0; | 64 | else |
65 | jump_label_make_branch(entry, &old); | ||
66 | jump_label_make_nop(entry, &new); | ||
39 | } | 67 | } |
40 | 68 | if (memcmp((void *)entry->code, &old, sizeof(old))) | |
41 | rc = probe_kernel_write((void *)entry->code, &insn, JUMP_LABEL_NOP_SIZE); | 69 | jump_label_bug(entry, &old); |
42 | WARN_ON_ONCE(rc < 0); | 70 | probe_kernel_write((void *)entry->code, &new, sizeof(new)); |
43 | } | 71 | } |
44 | 72 | ||
45 | static int __sm_arch_jump_label_transform(void *data) | 73 | static int __sm_arch_jump_label_transform(void *data) |
46 | { | 74 | { |
47 | struct insn_args *args = data; | 75 | struct insn_args *args = data; |
48 | 76 | ||
49 | __jump_label_transform(args->entry, args->type); | 77 | __jump_label_transform(args->entry, args->type, 0); |
50 | return 0; | 78 | return 0; |
51 | } | 79 | } |
52 | 80 | ||
@@ -64,7 +92,7 @@ void arch_jump_label_transform(struct jump_entry *entry, | |||
64 | void arch_jump_label_transform_static(struct jump_entry *entry, | 92 | void arch_jump_label_transform_static(struct jump_entry *entry, |
65 | enum jump_label_type type) | 93 | enum jump_label_type type) |
66 | { | 94 | { |
67 | __jump_label_transform(entry, type); | 95 | __jump_label_transform(entry, type, 1); |
68 | } | 96 | } |
69 | 97 | ||
70 | #endif | 98 | #endif |