aboutsummaryrefslogtreecommitdiffstats
path: root/arch/arm/kernel/kprobes-decode.c
diff options
context:
space:
mode:
authorJon Medhurst <tixy@yxit.co.uk>2011-04-19 05:52:18 -0400
committerNicolas Pitre <nicolas.pitre@linaro.org>2011-04-28 23:41:00 -0400
commit20e8155e24ba5c04558780d0a8da13c39f98c002 (patch)
tree27d7bcb62bb4964442a92eb05c2e679c2ea477af /arch/arm/kernel/kprobes-decode.c
parentc9836777d5f22e64eefc759d682511560b6d6d78 (diff)
ARM: kprobes: Add emulation of SBFX, UBFX, BFI and BFC instructions
These bit field manipulation instructions occur several thousand times in an ARMv7 kernel. 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.c42
1 files changed, 42 insertions, 0 deletions
diff --git a/arch/arm/kernel/kprobes-decode.c b/arch/arm/kernel/kprobes-decode.c
index a063f152d9b3..6bed8b089af0 100644
--- a/arch/arm/kernel/kprobes-decode.c
+++ b/arch/arm/kernel/kprobes-decode.c
@@ -672,6 +672,19 @@ emulate_rd12_modify(struct kprobe *p, struct pt_regs *regs)
672 regs->uregs[rd] = insnslot_1arg_rflags(rdv, regs->ARM_cpsr, i_fn); 672 regs->uregs[rd] = insnslot_1arg_rflags(rdv, regs->ARM_cpsr, i_fn);
673} 673}
674 674
675static void __kprobes
676emulate_rd12rn0_modify(struct kprobe *p, struct pt_regs *regs)
677{
678 insn_2arg_fn_t *i_fn = (insn_2arg_fn_t *)&p->ainsn.insn[0];
679 kprobe_opcode_t insn = p->opcode;
680 int rd = (insn >> 12) & 0xf;
681 int rn = insn & 0xf;
682 long rdv = regs->uregs[rd];
683 long rnv = regs->uregs[rn];
684
685 regs->uregs[rd] = insnslot_2arg_rflags(rdv, rnv, regs->ARM_cpsr, i_fn);
686}
687
675static void __kprobes emulate_rd12rm0(struct kprobe *p, struct pt_regs *regs) 688static void __kprobes emulate_rd12rm0(struct kprobe *p, struct pt_regs *regs)
676{ 689{
677 insn_1arg_fn_t *i_fn = (insn_1arg_fn_t *)&p->ainsn.insn[0]; 690 insn_1arg_fn_t *i_fn = (insn_1arg_fn_t *)&p->ainsn.insn[0];
@@ -870,6 +883,20 @@ prep_emulate_rd12_modify(kprobe_opcode_t insn, struct arch_specific_insn *asi)
870} 883}
871 884
872static enum kprobe_insn __kprobes 885static enum kprobe_insn __kprobes
886prep_emulate_rd12rn0_modify(kprobe_opcode_t insn,
887 struct arch_specific_insn *asi)
888{
889 if (is_r15(insn, 12))
890 return INSN_REJECTED; /* Rd is PC */
891
892 insn &= 0xffff0ff0; /* Rd = r0 */
893 insn |= 0x00000001; /* Rn = r1 */
894 asi->insn[0] = insn;
895 asi->insn_handler = emulate_rd12rn0_modify;
896 return INSN_GOOD;
897}
898
899static enum kprobe_insn __kprobes
873prep_emulate_rd12rm0(kprobe_opcode_t insn, struct arch_specific_insn *asi) 900prep_emulate_rd12rm0(kprobe_opcode_t insn, struct arch_specific_insn *asi)
874{ 901{
875 if (is_r15(insn, 12)) 902 if (is_r15(insn, 12))
@@ -1396,6 +1423,21 @@ space_cccc_0111__1(kprobe_opcode_t insn, struct arch_specific_insn *asi)
1396 if ((insn & 0x0ff000d0) == 0x075000d0) 1423 if ((insn & 0x0ff000d0) == 0x075000d0)
1397 return prep_emulate_rd16rn12rs8rm0_wflags(insn, asi); 1424 return prep_emulate_rd16rn12rs8rm0_wflags(insn, asi);
1398 1425
1426 /* SBFX : cccc 0111 101x xxxx xxxx xxxx x101 xxxx : */
1427 /* UBFX : cccc 0111 111x xxxx xxxx xxxx x101 xxxx : */
1428 if ((insn & 0x0fa00070) == 0x07a00050)
1429 return prep_emulate_rd12rm0(insn, asi);
1430
1431 /* BFI : cccc 0111 110x xxxx xxxx xxxx x001 xxxx : */
1432 /* BFC : cccc 0111 110x xxxx xxxx xxxx x001 1111 : */
1433 if ((insn & 0x0fe00070) == 0x07c00010) {
1434
1435 if ((insn & 0x0000000f) == 0x0000000f)
1436 return prep_emulate_rd12_modify(insn, asi);
1437 else
1438 return prep_emulate_rd12rn0_modify(insn, asi);
1439 }
1440
1399 return INSN_REJECTED; 1441 return INSN_REJECTED;
1400} 1442}
1401 1443