|
@@ -14,6 +14,7 @@
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
*/
|
|
|
+#include <linux/bitops.h>
|
|
|
#include <linux/compiler.h>
|
|
|
#include <linux/kernel.h>
|
|
|
#include <linux/smp.h>
|
|
@@ -262,3 +263,42 @@ u32 __kprobes aarch64_insn_encode_immediate(enum aarch64_insn_imm_type type,
|
|
|
|
|
|
return insn;
|
|
|
}
|
|
|
+
|
|
|
+u32 __kprobes aarch64_insn_gen_branch_imm(unsigned long pc, unsigned long addr,
|
|
|
+ enum aarch64_insn_branch_type type)
|
|
|
+{
|
|
|
+ u32 insn;
|
|
|
+ long offset;
|
|
|
+
|
|
|
+ /*
|
|
|
+ * PC: A 64-bit Program Counter holding the address of the current
|
|
|
+ * instruction. A64 instructions must be word-aligned.
|
|
|
+ */
|
|
|
+ BUG_ON((pc & 0x3) || (addr & 0x3));
|
|
|
+
|
|
|
+ /*
|
|
|
+ * B/BL support [-128M, 128M) offset
|
|
|
+ * ARM64 virtual address arrangement guarantees all kernel and module
|
|
|
+ * texts are within +/-128M.
|
|
|
+ */
|
|
|
+ offset = ((long)addr - (long)pc);
|
|
|
+ BUG_ON(offset < -SZ_128M || offset >= SZ_128M);
|
|
|
+
|
|
|
+ if (type == AARCH64_INSN_BRANCH_LINK)
|
|
|
+ insn = aarch64_insn_get_bl_value();
|
|
|
+ else
|
|
|
+ insn = aarch64_insn_get_b_value();
|
|
|
+
|
|
|
+ return aarch64_insn_encode_immediate(AARCH64_INSN_IMM_26, insn,
|
|
|
+ offset >> 2);
|
|
|
+}
|
|
|
+
|
|
|
+u32 __kprobes aarch64_insn_gen_hint(enum aarch64_insn_hint_op op)
|
|
|
+{
|
|
|
+ return aarch64_insn_get_hint_value() | op;
|
|
|
+}
|
|
|
+
|
|
|
+u32 __kprobes aarch64_insn_gen_nop(void)
|
|
|
+{
|
|
|
+ return aarch64_insn_gen_hint(AARCH64_INSN_HINT_NOP);
|
|
|
+}
|