aboutsummaryrefslogtreecommitdiffstats
path: root/arch/arm64
diff options
context:
space:
mode:
authorZi Shen Lim <zlim.lnx@gmail.com>2014-08-27 00:15:25 -0400
committerWill Deacon <will.deacon@arm.com>2014-09-08 09:39:20 -0400
commit5fdc639a7a5b187f75b7408ee7ae9f9c06771218 (patch)
tree498e9d59d73fd9e977e8abab44e1728f75681744 /arch/arm64
parent6098f2d5c7a349d388499503bb129d0088870dd6 (diff)
arm64: introduce aarch64_insn_gen_add_sub_shifted_reg()
Introduce function to generate add/subtract (shifted register) instructions. Signed-off-by: Zi Shen Lim <zlim.lnx@gmail.com> Acked-by: Will Deacon <will.deacon@arm.com> Signed-off-by: Will Deacon <will.deacon@arm.com>
Diffstat (limited to 'arch/arm64')
-rw-r--r--arch/arm64/include/asm/insn.h11
-rw-r--r--arch/arm64/kernel/insn.c49
2 files changed, 60 insertions, 0 deletions
diff --git a/arch/arm64/include/asm/insn.h b/arch/arm64/include/asm/insn.h
index 49dec288f5ac..c0a765dae1f0 100644
--- a/arch/arm64/include/asm/insn.h
+++ b/arch/arm64/include/asm/insn.h
@@ -67,6 +67,7 @@ enum aarch64_insn_imm_type {
67 AARCH64_INSN_IMM_12, 67 AARCH64_INSN_IMM_12,
68 AARCH64_INSN_IMM_9, 68 AARCH64_INSN_IMM_9,
69 AARCH64_INSN_IMM_7, 69 AARCH64_INSN_IMM_7,
70 AARCH64_INSN_IMM_6,
70 AARCH64_INSN_IMM_S, 71 AARCH64_INSN_IMM_S,
71 AARCH64_INSN_IMM_R, 72 AARCH64_INSN_IMM_R,
72 AARCH64_INSN_IMM_MAX 73 AARCH64_INSN_IMM_MAX
@@ -206,6 +207,10 @@ __AARCH64_INSN_FUNCS(bfm, 0x7F800000, 0x33000000)
206__AARCH64_INSN_FUNCS(movz, 0x7F800000, 0x52800000) 207__AARCH64_INSN_FUNCS(movz, 0x7F800000, 0x52800000)
207__AARCH64_INSN_FUNCS(ubfm, 0x7F800000, 0x53000000) 208__AARCH64_INSN_FUNCS(ubfm, 0x7F800000, 0x53000000)
208__AARCH64_INSN_FUNCS(movk, 0x7F800000, 0x72800000) 209__AARCH64_INSN_FUNCS(movk, 0x7F800000, 0x72800000)
210__AARCH64_INSN_FUNCS(add, 0x7F200000, 0x0B000000)
211__AARCH64_INSN_FUNCS(adds, 0x7F200000, 0x2B000000)
212__AARCH64_INSN_FUNCS(sub, 0x7F200000, 0x4B000000)
213__AARCH64_INSN_FUNCS(subs, 0x7F200000, 0x6B000000)
209__AARCH64_INSN_FUNCS(b, 0xFC000000, 0x14000000) 214__AARCH64_INSN_FUNCS(b, 0xFC000000, 0x14000000)
210__AARCH64_INSN_FUNCS(bl, 0xFC000000, 0x94000000) 215__AARCH64_INSN_FUNCS(bl, 0xFC000000, 0x94000000)
211__AARCH64_INSN_FUNCS(cbz, 0xFE000000, 0x34000000) 216__AARCH64_INSN_FUNCS(cbz, 0xFE000000, 0x34000000)
@@ -265,6 +270,12 @@ u32 aarch64_insn_gen_movewide(enum aarch64_insn_register dst,
265 int imm, int shift, 270 int imm, int shift,
266 enum aarch64_insn_variant variant, 271 enum aarch64_insn_variant variant,
267 enum aarch64_insn_movewide_type type); 272 enum aarch64_insn_movewide_type type);
273u32 aarch64_insn_gen_add_sub_shifted_reg(enum aarch64_insn_register dst,
274 enum aarch64_insn_register src,
275 enum aarch64_insn_register reg,
276 int shift,
277 enum aarch64_insn_variant variant,
278 enum aarch64_insn_adsb_type type);
268 279
269bool aarch64_insn_hotpatch_safe(u32 old_insn, u32 new_insn); 280bool aarch64_insn_hotpatch_safe(u32 old_insn, u32 new_insn);
270 281
diff --git a/arch/arm64/kernel/insn.c b/arch/arm64/kernel/insn.c
index 7aa278404b80..d7a4dd48e959 100644
--- a/arch/arm64/kernel/insn.c
+++ b/arch/arm64/kernel/insn.c
@@ -260,6 +260,7 @@ u32 __kprobes aarch64_insn_encode_immediate(enum aarch64_insn_imm_type type,
260 mask = BIT(7) - 1; 260 mask = BIT(7) - 1;
261 shift = 15; 261 shift = 15;
262 break; 262 break;
263 case AARCH64_INSN_IMM_6:
263 case AARCH64_INSN_IMM_S: 264 case AARCH64_INSN_IMM_S:
264 mask = BIT(6) - 1; 265 mask = BIT(6) - 1;
265 shift = 10; 266 shift = 10;
@@ -698,3 +699,51 @@ u32 aarch64_insn_gen_movewide(enum aarch64_insn_register dst,
698 699
699 return aarch64_insn_encode_immediate(AARCH64_INSN_IMM_16, insn, imm); 700 return aarch64_insn_encode_immediate(AARCH64_INSN_IMM_16, insn, imm);
700} 701}
702
703u32 aarch64_insn_gen_add_sub_shifted_reg(enum aarch64_insn_register dst,
704 enum aarch64_insn_register src,
705 enum aarch64_insn_register reg,
706 int shift,
707 enum aarch64_insn_variant variant,
708 enum aarch64_insn_adsb_type type)
709{
710 u32 insn;
711
712 switch (type) {
713 case AARCH64_INSN_ADSB_ADD:
714 insn = aarch64_insn_get_add_value();
715 break;
716 case AARCH64_INSN_ADSB_SUB:
717 insn = aarch64_insn_get_sub_value();
718 break;
719 case AARCH64_INSN_ADSB_ADD_SETFLAGS:
720 insn = aarch64_insn_get_adds_value();
721 break;
722 case AARCH64_INSN_ADSB_SUB_SETFLAGS:
723 insn = aarch64_insn_get_subs_value();
724 break;
725 default:
726 BUG_ON(1);
727 }
728
729 switch (variant) {
730 case AARCH64_INSN_VARIANT_32BIT:
731 BUG_ON(shift & ~(SZ_32 - 1));
732 break;
733 case AARCH64_INSN_VARIANT_64BIT:
734 insn |= AARCH64_INSN_SF_BIT;
735 BUG_ON(shift & ~(SZ_64 - 1));
736 break;
737 default:
738 BUG_ON(1);
739 }
740
741
742 insn = aarch64_insn_encode_register(AARCH64_INSN_REGTYPE_RD, insn, dst);
743
744 insn = aarch64_insn_encode_register(AARCH64_INSN_REGTYPE_RN, insn, src);
745
746 insn = aarch64_insn_encode_register(AARCH64_INSN_REGTYPE_RM, insn, reg);
747
748 return aarch64_insn_encode_immediate(AARCH64_INSN_IMM_6, insn, shift);
749}