diff options
author | Jon Medhurst <tixy@yxit.co.uk> | 2011-04-08 10:32:54 -0400 |
---|---|---|
committer | Nicolas Pitre <nicolas.pitre@linaro.org> | 2011-04-28 23:40:56 -0400 |
commit | 6823fc85fcfba11675f2027aadf2d5291c6f351b (patch) | |
tree | 44da95dd42e7b98e475a644f1aaac46310b69492 /arch/arm/kernel/kprobes-decode.c | |
parent | ec58d7f2373cec47f30b773d40a89f18bf6fc489 (diff) |
ARM: kprobes: Fix emulation of LDRH, STRH, LDRSB and LDRSH instructions
The decoding of these instructions got the register indexed and
immediate indexed forms the wrong way around, causing incorrect
emulation.
Signed-off-by: Jon Medhurst <tixy@yxit.co.uk>
Signed-off-by: Nicolas Pitre <nicolas.pitre@linaro.org>
Diffstat (limited to 'arch/arm/kernel/kprobes-decode.c')
-rw-r--r-- | arch/arm/kernel/kprobes-decode.c | 5 |
1 files changed, 3 insertions, 2 deletions
diff --git a/arch/arm/kernel/kprobes-decode.c b/arch/arm/kernel/kprobes-decode.c index c24e21ec427f..348ff47acd06 100644 --- a/arch/arm/kernel/kprobes-decode.c +++ b/arch/arm/kernel/kprobes-decode.c | |||
@@ -883,11 +883,12 @@ emulate_alu_tests(struct kprobe *p, struct pt_regs *regs) | |||
883 | static enum kprobe_insn __kprobes | 883 | static enum kprobe_insn __kprobes |
884 | prep_emulate_ldr_str(kprobe_opcode_t insn, struct arch_specific_insn *asi) | 884 | prep_emulate_ldr_str(kprobe_opcode_t insn, struct arch_specific_insn *asi) |
885 | { | 885 | { |
886 | int ibit = (insn & (1 << 26)) ? 25 : 22; | 886 | int not_imm = (insn & (1 << 26)) ? (insn & (1 << 25)) |
887 | : (~insn & (1 << 22)); | ||
887 | 888 | ||
888 | insn &= 0xfff00fff; | 889 | insn &= 0xfff00fff; |
889 | insn |= 0x00001000; /* Rn = r0, Rd = r1 */ | 890 | insn |= 0x00001000; /* Rn = r0, Rd = r1 */ |
890 | if (insn & (1 << ibit)) { | 891 | if (not_imm) { |
891 | insn &= ~0xf; | 892 | insn &= ~0xf; |
892 | insn |= 2; /* Rm = r2 */ | 893 | insn |= 2; /* Rm = r2 */ |
893 | } | 894 | } |