diff options
author | Zi Shen Lim <zlim.lnx@gmail.com> | 2014-08-27 00:15:25 -0400 |
---|---|---|
committer | Will Deacon <will.deacon@arm.com> | 2014-09-08 09:39:20 -0400 |
commit | 5fdc639a7a5b187f75b7408ee7ae9f9c06771218 (patch) | |
tree | 498e9d59d73fd9e977e8abab44e1728f75681744 /arch/arm64 | |
parent | 6098f2d5c7a349d388499503bb129d0088870dd6 (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.h | 11 | ||||
-rw-r--r-- | arch/arm64/kernel/insn.c | 49 |
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); |
273 | u32 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 | ||
269 | bool aarch64_insn_hotpatch_safe(u32 old_insn, u32 new_insn); | 280 | bool 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 | |||
703 | u32 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 | } | ||