diff options
author | Jon Medhurst <tixy@yxit.co.uk> | 2011-04-07 08:25:16 -0400 |
---|---|---|
committer | Nicolas Pitre <nicolas.pitre@linaro.org> | 2011-04-28 23:40:55 -0400 |
commit | c412aba2a1243192a4d53736805a96bdda915608 (patch) | |
tree | f04010b8c82fcc4ef9934144dd8eb8cbf6f8ad1c /arch/arm/kernel | |
parent | 51468ea91efad9c7e6dbae43cd8bdc423ec61709 (diff) |
ARM: kprobes: Fix emulation of MRS instruction
The MRS instruction should set mode and interrupt bits in the read value
so it is simpler to use a new simulation routine (simulate_mrs) rather
than some modified emulation.
prep_emulate_rd12 is now unused and removed.
Signed-off-by: Jon Medhurst <tixy@yxit.co.uk>
Signed-off-by: Nicolas Pitre <nicolas.pitre@linaro.org>
Diffstat (limited to 'arch/arm/kernel')
-rw-r--r-- | arch/arm/kernel/kprobes-decode.c | 23 |
1 files changed, 12 insertions, 11 deletions
diff --git a/arch/arm/kernel/kprobes-decode.c b/arch/arm/kernel/kprobes-decode.c index e5bc576ba3fb..4ab83f4c47cf 100644 --- a/arch/arm/kernel/kprobes-decode.c +++ b/arch/arm/kernel/kprobes-decode.c | |||
@@ -451,6 +451,14 @@ static void __kprobes simulate_blx2bx(struct kprobe *p, struct pt_regs *regs) | |||
451 | regs->ARM_cpsr |= PSR_T_BIT; | 451 | regs->ARM_cpsr |= PSR_T_BIT; |
452 | } | 452 | } |
453 | 453 | ||
454 | static void __kprobes simulate_mrs(struct kprobe *p, struct pt_regs *regs) | ||
455 | { | ||
456 | kprobe_opcode_t insn = p->opcode; | ||
457 | int rd = (insn >> 12) & 0xf; | ||
458 | unsigned long mask = 0xf8ff03df; /* Mask out execution state */ | ||
459 | regs->uregs[rd] = regs->ARM_cpsr & mask; | ||
460 | } | ||
461 | |||
454 | static void __kprobes simulate_ldm1stm1(struct kprobe *p, struct pt_regs *regs) | 462 | static void __kprobes simulate_ldm1stm1(struct kprobe *p, struct pt_regs *regs) |
455 | { | 463 | { |
456 | kprobe_opcode_t insn = p->opcode; | 464 | kprobe_opcode_t insn = p->opcode; |
@@ -896,15 +904,6 @@ prep_emulate_rd12rm0(kprobe_opcode_t insn, struct arch_specific_insn *asi) | |||
896 | } | 904 | } |
897 | 905 | ||
898 | static enum kprobe_insn __kprobes | 906 | static enum kprobe_insn __kprobes |
899 | prep_emulate_rd12(kprobe_opcode_t insn, struct arch_specific_insn *asi) | ||
900 | { | ||
901 | insn &= 0xffff0fff; /* Rd = r0 */ | ||
902 | asi->insn[0] = insn; | ||
903 | asi->insn_handler = emulate_rd12; | ||
904 | return INSN_GOOD; | ||
905 | } | ||
906 | |||
907 | static enum kprobe_insn __kprobes | ||
908 | prep_emulate_rd12rn16rm0_wflags(kprobe_opcode_t insn, | 907 | prep_emulate_rd12rn16rm0_wflags(kprobe_opcode_t insn, |
909 | struct arch_specific_insn *asi) | 908 | struct arch_specific_insn *asi) |
910 | { | 909 | { |
@@ -1035,8 +1034,10 @@ space_cccc_000x(kprobe_opcode_t insn, struct arch_specific_insn *asi) | |||
1035 | return INSN_REJECTED; | 1034 | return INSN_REJECTED; |
1036 | 1035 | ||
1037 | /* MRS cpsr : cccc 0001 0000 xxxx xxxx xxxx 0000 xxxx */ | 1036 | /* MRS cpsr : cccc 0001 0000 xxxx xxxx xxxx 0000 xxxx */ |
1038 | if ((insn & 0x0ff000f0) == 0x01000000) | 1037 | if ((insn & 0x0ff000f0) == 0x01000000) { |
1039 | return prep_emulate_rd12(insn, asi); | 1038 | asi->insn_handler = simulate_mrs; |
1039 | return INSN_GOOD_NO_SLOT; | ||
1040 | } | ||
1040 | 1041 | ||
1041 | /* SMLALxy : cccc 0001 0100 xxxx xxxx xxxx 1xx0 xxxx */ | 1042 | /* SMLALxy : cccc 0001 0100 xxxx xxxx xxxx 1xx0 xxxx */ |
1042 | if ((insn & 0x0ff00090) == 0x01400080) | 1043 | if ((insn & 0x0ff00090) == 0x01400080) |