commit 520c6647c32f02d83083d969d416154aa95e922c
Merge: 6bb29b12 b999f507
Author: mio <mio@lazym.io>
Date: Sun Apr 13 00:14:23 2025 +0800
merge dev
commit 6bb29b12f1d9f452365cc9cb5bc2d65ef376af30
Author: mio <mio@lazym.io>
Date: Sun Apr 13 00:13:12 2025 +0800
enable test
commit bcb8b363ef12ac295cf4fe4f1645416e5f0ea6ae
Author: mio <mio@lazym.io>
Date: Sun Apr 13 00:13:06 2025 +0800
also logging
commit 5972fc156b7379d09582c745d6d597e07555f2f4
Author: mio <mio@lazym.io>
Date: Sun Apr 13 00:12:58 2025 +0800
no unlimited translation
commit 7d600feebf9055505918e50d0af8b529a3eba542
Author: mio <mio@lazym.io>
Date: Sun Apr 13 00:12:47 2025 +0800
Ignore bindings.rs
commit dde4d50f2c7713156ac3bc284287480e4d92005f
Author: Amaan Qureshi <amaanq12@gmail.com>
Date: Sun Apr 6 03:26:22 2025 -0400
alias `uc_mips_reg` to `UC_MIPS_REG`
commit 04234ae01ba7c82d9717eaae64cdda289ce3b832
Author: Amaan Qureshi <amaanq12@gmail.com>
Date: Sun Apr 6 01:13:00 2025 -0400
remove bindings.rs
commit edec1300cd7c2d8ef4babbd51f6bcba2e126bdd7
Author: Amaan Qureshi <amaanq12@gmail.com>
Date: Sat Apr 5 14:29:40 2025 -0400
address review
commit feb157b28b6c262c5dc3d810ec54de55a25bcd6e
Author: Amaan Qureshi <amaanq12@gmail.com>
Date: Sat Mar 29 22:40:53 2025 -0400
ci(rust): rework workflow
The notable changes are migrating to
`actions-rust-lang/setup-rust-toolchain` for setting up Rust as it's
maintained, and using `katyo/publish-crates` for publishing crates in a
workspace
commit c1c7a8f8ed841b6ec5b4abe57013a1c2c9748c60
Author: Amaan Qureshi <amaanq12@gmail.com>
Date: Sat Mar 29 22:40:06 2025 -0400
build(rust): set `rust-version` to 1.85
commit 8df938c9f8b478160213707674157103b0893caf
Author: Amaan Qureshi <amaanq12@gmail.com>
Date: Sat Mar 29 21:53:21 2025 -0400
fix(rust): correct unsound pointer cast
The size of `T` is not guaranteed to be the size of `i32` - all we know
is that `T` is `Into<i32>`, so we should first copy them over into an
`i32` array
commit 3059b2583a60aa0cac9278afc945ed87f7ddb65e
Author: Amaan Qureshi <amaanq12@gmail.com>
Date: Sat Mar 29 20:13:26 2025 -0400
docs(rust): update readme
commit 7db69a888e58a4bda20083e4e0771d26a327ad13
Author: Amaan Qureshi <amaanq12@gmail.com>
Date: Sat Mar 29 13:58:30 2025 -0400
feat(rust): add comprehensive tests
These tests are copied over from the C tests
commit 78f2207f0e0481aef4de6d5908f8dc699a39a8d5
Author: Amaan Qureshi <amaanq12@gmail.com>
Date: Sat Mar 29 13:57:27 2025 -0400
feat(rust): add tcg hook
commit 46e53328531ec3279dadbf18c16b493432227b31
Author: Amaan Qureshi <amaanq12@gmail.com>
Date: Sat Mar 29 13:56:55 2025 -0400
feat(rust): add a hook for arm64 sys instructions
commit d1b58ee8282bf1eeeefbf68c87c2cf7c50c90320
Author: Amaan Qureshi <amaanq12@gmail.com>
Date: Sat Mar 29 13:56:35 2025 -0400
feat(rust): add the ability to read the arm coprocessor register
commit d304da18b9e6741042b2a70657437be8f39f5c7c
Author: Amaan Qureshi <amaanq12@gmail.com>
Date: Sat Mar 29 13:55:29 2025 -0400
feat(rust): add missing `Context` methods
commit 0dd87833081ac9db1feaf5bae8c839a7a2ae4947
Author: Amaan Qureshi <amaanq12@gmail.com>
Date: Sat Mar 29 13:44:51 2025 -0400
refactor(rust): remove unnecessary code
`unicorn-engine-sys` will provide the necessary constants & types
commit da3d2fa7c3ecd3ae8fdb6672b6c5ea23da4570ff
Author: Amaan Qureshi <amaanq12@gmail.com>
Date: Sat Mar 29 13:43:57 2025 -0400
feat(rust): add a workspace `Cargo.toml`, and use `unicorn-engine-sys`
commit b27a2a93e4ac43aa2079e936df4dd30a1f8f329a
Author: Amaan Qureshi <amaanq12@gmail.com>
Date: Sat Mar 29 13:38:06 2025 -0400
feat(rust): introduce `unicorn-engine-sys` crate
This crate contains generated Rust bindings to the C library via
bindgen. It is independent from the main `unicorn-engine` bindings,
which will leverage this
commit bcec87a3f6e316e328683c303ccfa89e530a6c56
Author: Amaan Qureshi <amaanq12@gmail.com>
Date: Sat Mar 29 13:31:24 2025 -0400
test(m68k): actually assert an expectation
This test did not actually test for anything before
commit bc7e65ca96164496eb2e250b1f296a33a8aa58ee
Author: Amaan Qureshi <amaanq12@gmail.com>
Date: Sat Mar 29 13:31:09 2025 -0400
style(test): use bitflag shorthands
commit 0ab4b7fefb3ca17b0b5977d7b204291c5de184ad
Author: Amaan Qureshi <amaanq12@gmail.com>
Date: Sat Mar 29 13:22:13 2025 -0400
fix(mips): lowercase enum name `uc_mips_reg`
This aligns with other architectures
Co-authored-by: Amaan Qureshi <amaanq12@gmail.com>
687 lines
21 KiB
C
687 lines
21 KiB
C
#include "acutest.h"
|
|
#include "unicorn/unicorn.h"
|
|
#include "unicorn_test.h"
|
|
#include <stdbool.h>
|
|
#include <stdint.h>
|
|
#include <stdio.h>
|
|
|
|
const uint64_t code_start = 0x1000;
|
|
const uint64_t code_len = 0x4000;
|
|
|
|
static void uc_common_setup(uc_engine **uc, uc_arch arch, uc_mode mode,
|
|
const char *code, uint64_t size, uc_cpu_arm64 cpu)
|
|
{
|
|
OK(uc_open(arch, mode, uc));
|
|
OK(uc_ctl_set_cpu_model(*uc, cpu));
|
|
OK(uc_mem_map(*uc, code_start, code_len, UC_PROT_ALL));
|
|
OK(uc_mem_write(*uc, code_start, code, size));
|
|
}
|
|
|
|
static void test_arm64_until(void)
|
|
{
|
|
uc_engine *uc;
|
|
char code[] = "\x30\x00\x80\xd2\x11\x04\x80\xd2\x9c\x23\x00\x91";
|
|
|
|
/*
|
|
mov x16, #1
|
|
mov x17, #0x20
|
|
add x28, x28, 8
|
|
*/
|
|
|
|
uint64_t r_x16 = 0x12341234;
|
|
uint64_t r_x17 = 0x78907890;
|
|
uint64_t r_pc = 0x00000000;
|
|
uint64_t r_x28 = 0x12341234;
|
|
|
|
uc_common_setup(&uc, UC_ARCH_ARM64, UC_MODE_ARM, code, sizeof(code) - 1,
|
|
UC_CPU_ARM64_A72);
|
|
|
|
// initialize machine registers
|
|
OK(uc_reg_write(uc, UC_ARM64_REG_X16, &r_x16));
|
|
OK(uc_reg_write(uc, UC_ARM64_REG_X17, &r_x17));
|
|
OK(uc_reg_write(uc, UC_ARM64_REG_X28, &r_x28));
|
|
|
|
// emulate the three instructions
|
|
OK(uc_emu_start(uc, code_start, code_start + sizeof(code) - 1, 0, 3));
|
|
|
|
OK(uc_reg_read(uc, UC_ARM64_REG_X16, &r_x16));
|
|
OK(uc_reg_read(uc, UC_ARM64_REG_X17, &r_x17));
|
|
OK(uc_reg_read(uc, UC_ARM64_REG_X28, &r_x28));
|
|
OK(uc_reg_read(uc, UC_ARM64_REG_PC, &r_pc));
|
|
|
|
TEST_CHECK(r_x16 == 0x1);
|
|
TEST_CHECK(r_x17 == 0x20);
|
|
TEST_CHECK(r_x28 == 0x1234123c);
|
|
TEST_CHECK(r_pc == (code_start + sizeof(code) - 1));
|
|
|
|
OK(uc_close(uc));
|
|
}
|
|
|
|
static void test_arm64_code_patching(void)
|
|
{
|
|
uc_engine *uc;
|
|
char code[] = "\x00\x04\x00\x11"; // add w0, w0, 0x1
|
|
uc_common_setup(&uc, UC_ARCH_ARM64, UC_MODE_ARM, code, sizeof(code) - 1,
|
|
UC_CPU_ARM64_A72);
|
|
// zero out x0
|
|
uint64_t r_x0 = 0x0;
|
|
OK(uc_reg_write(uc, UC_ARM64_REG_X0, &r_x0));
|
|
// emulate the instruction
|
|
OK(uc_emu_start(uc, code_start, code_start + sizeof(code) - 1, 0, 0));
|
|
// check value
|
|
OK(uc_reg_read(uc, UC_ARM64_REG_X0, &r_x0));
|
|
TEST_CHECK(r_x0 == 0x1);
|
|
// patch instruction
|
|
char patch_code[] = "\x00\xfc\x1f\x11"; // add w0, w0, 0x7FF
|
|
OK(uc_mem_write(uc, code_start, patch_code, sizeof(patch_code) - 1));
|
|
// zero out x0
|
|
r_x0 = 0x0;
|
|
OK(uc_reg_write(uc, UC_ARM64_REG_X0, &r_x0));
|
|
OK(uc_emu_start(uc, code_start, code_start + sizeof(patch_code) - 1, 0, 0));
|
|
// check value
|
|
OK(uc_reg_read(uc, UC_ARM64_REG_X0, &r_x0));
|
|
TEST_CHECK(r_x0 != 0x1);
|
|
TEST_CHECK(r_x0 == 0x7ff);
|
|
|
|
OK(uc_close(uc));
|
|
}
|
|
|
|
// Need to flush the cache before running the emulation after patching
|
|
static void test_arm64_code_patching_count(void)
|
|
{
|
|
uc_engine *uc;
|
|
char code[] = "\x00\x04\x00\x11"; // add w0, w0, 0x1
|
|
uc_common_setup(&uc, UC_ARCH_ARM64, UC_MODE_ARM, code, sizeof(code) - 1,
|
|
UC_CPU_ARM64_A72);
|
|
// zero out x0
|
|
uint64_t r_x0 = 0x0;
|
|
OK(uc_reg_write(uc, UC_ARM64_REG_X0, &r_x0));
|
|
// emulate the instruction
|
|
OK(uc_emu_start(uc, code_start, -1, 0, 1));
|
|
// check value
|
|
OK(uc_reg_read(uc, UC_ARM64_REG_X0, &r_x0));
|
|
TEST_CHECK(r_x0 == 0x1);
|
|
// patch instruction
|
|
char patch_code[] = "\x00\xfc\x1f\x11"; // add w0, w0, 0x7FF
|
|
OK(uc_mem_write(uc, code_start, patch_code, sizeof(patch_code) - 1));
|
|
OK(uc_ctl_remove_cache(uc, code_start,
|
|
code_start + sizeof(patch_code) - 1));
|
|
// zero out x0
|
|
r_x0 = 0x0;
|
|
OK(uc_reg_write(uc, UC_ARM64_REG_X0, &r_x0));
|
|
OK(uc_emu_start(uc, code_start, -1, 0, 1));
|
|
// check value
|
|
OK(uc_reg_read(uc, UC_ARM64_REG_X0, &r_x0));
|
|
TEST_CHECK(r_x0 != 0x1);
|
|
TEST_CHECK(r_x0 == 0x7ff);
|
|
|
|
OK(uc_close(uc));
|
|
}
|
|
|
|
static void test_arm64_v8_pac(void)
|
|
{
|
|
uc_engine *uc;
|
|
char code[] = "\x28\xfd\xea\xc8"; // casal x10, x8, [x9]
|
|
uint64_t r_x9, r_x8, mem;
|
|
|
|
uc_common_setup(&uc, UC_ARCH_ARM64, UC_MODE_ARM, code, sizeof(code) - 1,
|
|
UC_CPU_ARM64_MAX);
|
|
|
|
OK(uc_mem_map(uc, 0x40000, 0x1000, UC_PROT_ALL));
|
|
OK(uc_mem_write(uc, 0x40000, "\x00\x00\x00\x00\x00\x00\x00\x00", 8));
|
|
r_x9 = 0x40000;
|
|
OK(uc_reg_write(uc, UC_ARM64_REG_X9, &r_x9));
|
|
r_x8 = 0xdeadbeafdeadbeaf;
|
|
OK(uc_reg_write(uc, UC_ARM64_REG_X8, &r_x8));
|
|
|
|
OK(uc_emu_start(uc, code_start, code_start + sizeof(code) - 1, 0, 0));
|
|
|
|
OK(uc_mem_read(uc, 0x40000, (void *)&mem, 8));
|
|
|
|
TEST_CHECK(LEINT64(mem) == r_x8);
|
|
|
|
OK(uc_close(uc));
|
|
}
|
|
|
|
static void test_arm64_read_sctlr(void)
|
|
{
|
|
uc_engine *uc;
|
|
uc_arm64_cp_reg reg;
|
|
|
|
OK(uc_open(UC_ARCH_ARM64, UC_MODE_LITTLE_ENDIAN | UC_MODE_ARM, &uc));
|
|
|
|
// SCTLR_EL1. See arm reference.
|
|
reg.crn = 1;
|
|
reg.crm = 0;
|
|
reg.op0 = 0b11;
|
|
reg.op1 = 0;
|
|
reg.op2 = 0;
|
|
|
|
OK(uc_reg_read(uc, UC_ARM64_REG_CP_REG, ®));
|
|
|
|
TEST_CHECK((reg.val >> 58) == 0);
|
|
|
|
OK(uc_close(uc));
|
|
}
|
|
|
|
static uint32_t test_arm64_mrs_hook_cb(uc_engine *uc, uc_arm64_reg reg,
|
|
const uc_arm64_cp_reg *cp_reg)
|
|
{
|
|
uint64_t r_x2 = 0x114514;
|
|
|
|
OK(uc_reg_write(uc, reg, &r_x2));
|
|
|
|
// Skip
|
|
return 1;
|
|
}
|
|
|
|
static void test_arm64_mrs_hook(void)
|
|
{
|
|
uc_engine *uc;
|
|
uc_hook hk;
|
|
uint64_t r_x2;
|
|
// mrs x2, tpidrro_el0
|
|
char code[] = "\x62\xd0\x3b\xd5";
|
|
|
|
uc_common_setup(&uc, UC_ARCH_ARM64, UC_MODE_LITTLE_ENDIAN | UC_MODE_ARM,
|
|
code, sizeof(code) - 1, UC_CPU_ARM64_A72);
|
|
|
|
OK(uc_hook_add(uc, &hk, UC_HOOK_INSN, (void *)test_arm64_mrs_hook_cb, NULL,
|
|
1, 0, UC_ARM64_INS_MRS));
|
|
|
|
OK(uc_emu_start(uc, code_start, code_start + sizeof(code) - 1, 0, 0));
|
|
|
|
OK(uc_reg_read(uc, UC_ARM64_REG_X2, &r_x2));
|
|
|
|
TEST_CHECK(r_x2 == 0x114514);
|
|
|
|
OK(uc_hook_del(uc, hk));
|
|
|
|
OK(uc_close(uc));
|
|
}
|
|
|
|
static bool test_arm64_correct_address_in_small_jump_hook_callback(
|
|
uc_engine *uc, int type, uint64_t address, int size, int64_t value,
|
|
void *user_data)
|
|
{
|
|
// Check registers
|
|
uint64_t r_x0 = 0x0;
|
|
uint64_t r_pc = 0x0;
|
|
OK(uc_reg_read(uc, UC_ARM64_REG_X0, &r_x0));
|
|
OK(uc_reg_read(uc, UC_ARM64_REG_PC, &r_pc));
|
|
TEST_CHECK(r_x0 == 0x7F00);
|
|
TEST_CHECK(r_pc == 0x7F00);
|
|
|
|
// Check address
|
|
// printf("%lx\n", address);
|
|
TEST_CHECK(address == 0x7F00);
|
|
|
|
return false;
|
|
}
|
|
|
|
static void test_arm64_correct_address_in_small_jump_hook(void)
|
|
{
|
|
uc_engine *uc;
|
|
// mov x0, 0x7F00;
|
|
// br x0
|
|
char code[] = "\x00\xe0\x8f\xd2\x00\x00\x1f\xd6";
|
|
|
|
uint64_t r_x0 = 0x0;
|
|
uint64_t r_pc = 0x0;
|
|
uc_hook hook;
|
|
|
|
uc_common_setup(&uc, UC_ARCH_ARM64, UC_MODE_ARM, code, sizeof(code) - 1,
|
|
UC_CPU_ARM64_A72);
|
|
OK(uc_hook_add(uc, &hook, UC_HOOK_MEM_UNMAPPED,
|
|
test_arm64_correct_address_in_small_jump_hook_callback, NULL,
|
|
1, 0));
|
|
|
|
uc_assert_err(
|
|
UC_ERR_FETCH_UNMAPPED,
|
|
uc_emu_start(uc, code_start, code_start + sizeof(code) - 1, 0, 0));
|
|
|
|
OK(uc_reg_read(uc, UC_ARM64_REG_X0, &r_x0));
|
|
OK(uc_reg_read(uc, UC_ARM64_REG_PC, &r_pc));
|
|
TEST_CHECK(r_x0 == 0x7F00);
|
|
TEST_CHECK(r_pc == 0x7F00);
|
|
|
|
OK(uc_close(uc));
|
|
}
|
|
|
|
static bool test_arm64_correct_address_in_long_jump_hook_callback(
|
|
uc_engine *uc, int type, uint64_t address, int size, int64_t value,
|
|
void *user_data)
|
|
{
|
|
// Check registers
|
|
uint64_t r_x0 = 0x0;
|
|
uint64_t r_pc = 0x0;
|
|
OK(uc_reg_read(uc, UC_ARM64_REG_X0, &r_x0));
|
|
OK(uc_reg_read(uc, UC_ARM64_REG_PC, &r_pc));
|
|
TEST_CHECK(r_x0 == 0x7FFFFFFFFFFFFF00);
|
|
TEST_CHECK(r_pc == 0x7FFFFFFFFFFFFF00);
|
|
|
|
// Check address
|
|
// printf("%lx\n", address);
|
|
TEST_CHECK(address == 0x7FFFFFFFFFFFFF00);
|
|
|
|
return false;
|
|
}
|
|
|
|
static void test_arm64_correct_address_in_long_jump_hook(void)
|
|
{
|
|
uc_engine *uc;
|
|
// mov x0, 0x7FFFFFFFFFFFFF00;
|
|
// br x0
|
|
char code[] = "\xe0\xdb\x78\xb2\x00\x00\x1f\xd6";
|
|
|
|
uint64_t r_x0 = 0x0;
|
|
uint64_t r_pc = 0x0;
|
|
uc_hook hook;
|
|
|
|
uc_common_setup(&uc, UC_ARCH_ARM64, UC_MODE_ARM, code, sizeof(code) - 1,
|
|
UC_CPU_ARM64_A72);
|
|
OK(uc_hook_add(uc, &hook, UC_HOOK_MEM_UNMAPPED,
|
|
test_arm64_correct_address_in_long_jump_hook_callback, NULL,
|
|
1, 0));
|
|
|
|
uc_assert_err(
|
|
UC_ERR_FETCH_UNMAPPED,
|
|
uc_emu_start(uc, code_start, code_start + sizeof(code) - 1, 0, 0));
|
|
|
|
OK(uc_reg_read(uc, UC_ARM64_REG_X0, &r_x0));
|
|
OK(uc_reg_read(uc, UC_ARM64_REG_PC, &r_pc));
|
|
TEST_CHECK(r_x0 == 0x7FFFFFFFFFFFFF00);
|
|
TEST_CHECK(r_pc == 0x7FFFFFFFFFFFFF00);
|
|
|
|
OK(uc_close(uc));
|
|
}
|
|
|
|
static void test_arm64_block_sync_pc_cb(uc_engine *uc, uint64_t addr,
|
|
uint32_t size, void *data)
|
|
{
|
|
uint64_t pc;
|
|
OK(uc_reg_read(uc, UC_ARM64_REG_PC, (void *)&pc));
|
|
TEST_CHECK(pc == addr);
|
|
uint64_t val = code_start;
|
|
bool first = *(bool *)data;
|
|
if (first) {
|
|
OK(uc_reg_write(uc, UC_ARM64_REG_PC, (void *)&val));
|
|
*(bool *)data = false;
|
|
}
|
|
}
|
|
|
|
static void test_arm64_block_sync_pc(void)
|
|
{
|
|
uc_engine *uc;
|
|
// add x0, x0, #1234;bl t;t:mov x1, #5678;
|
|
const char code[] = "\x00\x48\x13\x91\x01\x00\x00\x94\xc1\xc5\x82\xd2";
|
|
uc_hook hk;
|
|
uint64_t x0;
|
|
bool data = true;
|
|
|
|
uc_common_setup(&uc, UC_ARCH_ARM64, UC_MODE_ARM, code, sizeof(code) - 1,
|
|
UC_CPU_ARM64_A72);
|
|
OK(uc_hook_add(uc, &hk, UC_HOOK_BLOCK, test_arm64_block_sync_pc_cb,
|
|
(void *)&data, code_start + 8, code_start + 12));
|
|
|
|
x0 = 0;
|
|
OK(uc_reg_write(uc, UC_ARM64_REG_X0, (void *)&x0));
|
|
OK(uc_emu_start(uc, code_start, code_start + sizeof(code) - 1, 0, 0));
|
|
|
|
OK(uc_reg_read(uc, UC_ARM64_REG_X0, (void *)&x0));
|
|
|
|
TEST_CHECK(x0 == (1234 * 2));
|
|
|
|
OK(uc_hook_del(uc, hk));
|
|
OK(uc_close(uc));
|
|
}
|
|
|
|
static bool
|
|
test_arm64_block_invalid_mem_read_write_sync_cb(uc_engine *uc, int type,
|
|
uint64_t address, int size,
|
|
int64_t value, void *user_data)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static void test_arm64_block_invalid_mem_read_write_sync(void)
|
|
{
|
|
uc_engine *uc;
|
|
// mov x0, #1
|
|
// mov x1, #2
|
|
// ldr x0, [x1]
|
|
const char code[] = "\x20\x00\x80\xd2\x41\x00\x80\xd2\x20\x00\x40\xf9";
|
|
uint64_t r_pc, r_x0, r_x1;
|
|
uc_hook hk;
|
|
|
|
uc_common_setup(&uc, UC_ARCH_ARM64, UC_MODE_ARM, code, sizeof(code) - 1,
|
|
UC_CPU_ARM64_A72);
|
|
|
|
OK(uc_hook_add(uc, &hk, UC_HOOK_MEM_READ,
|
|
test_arm64_block_invalid_mem_read_write_sync_cb, NULL, 1,
|
|
0));
|
|
|
|
uc_assert_err(
|
|
UC_ERR_READ_UNMAPPED,
|
|
uc_emu_start(uc, code_start, code_start + sizeof(code) - 1, 0, 0));
|
|
|
|
OK(uc_reg_read(uc, UC_ARM64_REG_PC, &r_pc));
|
|
OK(uc_reg_read(uc, UC_ARM64_REG_X0, &r_x0));
|
|
OK(uc_reg_read(uc, UC_ARM64_REG_X1, &r_x1));
|
|
|
|
TEST_CHECK(r_pc == code_start + 8);
|
|
TEST_CHECK(r_x0 == 1);
|
|
TEST_CHECK(r_x1 == 2);
|
|
|
|
OK(uc_close(uc));
|
|
}
|
|
|
|
static void test_arm64_mmu(void)
|
|
{
|
|
uc_engine *uc;
|
|
char *data;
|
|
char tlbe[8];
|
|
uint64_t x0, x1, x2;
|
|
/*
|
|
* Not exact the binary, but aarch64-linux-gnu-as generate this code and
|
|
reference sometimes data after ttb0_base.
|
|
* // Read data from physical address
|
|
* ldr X0, =0x40000000
|
|
* ldr X1, [X0]
|
|
|
|
* // Initialize translation table control registers
|
|
* ldr X0, =0x180803F20
|
|
* msr TCR_EL1, X0
|
|
* ldr X0, =0xFFFFFFFF
|
|
* msr MAIR_EL1, X0
|
|
|
|
* // Set translation table
|
|
* adr X0, ttb0_base
|
|
* msr TTBR0_EL1, X0
|
|
|
|
* // Enable caches and the MMU
|
|
* mrs X0, SCTLR_EL1
|
|
* orr X0, X0, #(0x1 << 2) // The C bit (data cache).
|
|
* orr X0, X0, #(0x1 << 12) // The I bit (instruction cache)
|
|
* orr X0, X0, #0x1 // The M bit (MMU).
|
|
* msr SCTLR_EL1, X0
|
|
* dsb SY
|
|
* isb
|
|
|
|
* // Read the same memory area through virtual address
|
|
* ldr X0, =0x80000000
|
|
* ldr X2, [X0]
|
|
*
|
|
* // Stop
|
|
* b .
|
|
*/
|
|
char code[] = "\x00\x81\x00\x58\x01\x00\x40\xf9\x00\x81\x00\x58\x40\x20\x18"
|
|
"\xd5\x00\x81\x00\x58\x00\xa2\x18\xd5\x40\x7f\x00\x10\x00\x20"
|
|
"\x18\xd5\x00\x10\x38\xd5\x00\x00\x7e\xb2\x00\x00\x74\xb2\x00"
|
|
"\x00\x40\xb2\x00\x10\x18\xd5\x9f\x3f\x03\xd5\xdf\x3f\x03\xd5"
|
|
"\xe0\x7f\x00\x58\x02\x00\x40\xf9\x00\x00\x00\x14\x1f\x20\x03"
|
|
"\xd5\x1f\x20\x03\xd5\x1F\x20\x03\xD5\x1F\x20\x03\xD5";
|
|
|
|
data = malloc(0x1000);
|
|
TEST_CHECK(data != NULL);
|
|
|
|
OK(uc_open(UC_ARCH_ARM64, UC_MODE_ARM, &uc));
|
|
OK(uc_ctl_tlb_mode(uc, UC_TLB_CPU));
|
|
OK(uc_mem_map(uc, 0, 0x2000, UC_PROT_ALL));
|
|
OK(uc_mem_write(uc, 0, code, sizeof(code) - 1));
|
|
|
|
// generate tlb entries
|
|
tlbe[0] = 0x41;
|
|
tlbe[1] = 0x07;
|
|
tlbe[2] = 0;
|
|
tlbe[3] = 0;
|
|
tlbe[4] = 0;
|
|
tlbe[5] = 0;
|
|
tlbe[6] = 0;
|
|
tlbe[7] = 0;
|
|
OK(uc_mem_write(uc, 0x1000, tlbe, sizeof(tlbe)));
|
|
tlbe[3] = 0x40;
|
|
OK(uc_mem_write(uc, 0x1008, tlbe, sizeof(tlbe)));
|
|
OK(uc_mem_write(uc, 0x1010, tlbe, sizeof(tlbe)));
|
|
OK(uc_mem_write(uc, 0x1018, tlbe, sizeof(tlbe)));
|
|
|
|
// mentioned data referenced by the asm generated my aarch64-linux-gnu-as
|
|
tlbe[0] = 0;
|
|
tlbe[1] = 0;
|
|
OK(uc_mem_write(uc, 0x1020, tlbe, sizeof(tlbe)));
|
|
tlbe[0] = 0x20;
|
|
tlbe[1] = 0x3f;
|
|
tlbe[2] = 0x80;
|
|
tlbe[3] = 0x80;
|
|
tlbe[4] = 0x1;
|
|
OK(uc_mem_write(uc, 0x1028, tlbe, sizeof(tlbe)));
|
|
tlbe[0] = 0xff;
|
|
tlbe[1] = 0xff;
|
|
tlbe[2] = 0xff;
|
|
tlbe[3] = 0xff;
|
|
tlbe[4] = 0x00;
|
|
OK(uc_mem_write(uc, 0x1030, tlbe, sizeof(tlbe)));
|
|
tlbe[0] = 0x00;
|
|
tlbe[1] = 0x00;
|
|
tlbe[2] = 0x00;
|
|
tlbe[3] = 0x80;
|
|
OK(uc_mem_write(uc, 0x1038, tlbe, sizeof(tlbe)));
|
|
|
|
for (size_t i = 0; i < 0x1000; i++) {
|
|
data[i] = 0x44;
|
|
}
|
|
OK(uc_mem_map_ptr(uc, 0x40000000, 0x1000, UC_PROT_READ, data));
|
|
|
|
OK(uc_emu_start(uc, 0, 0x44, 0, 0));
|
|
OK(uc_reg_read(uc, UC_ARM64_REG_X0, &x0));
|
|
OK(uc_reg_read(uc, UC_ARM64_REG_X1, &x1));
|
|
OK(uc_reg_read(uc, UC_ARM64_REG_X2, &x2));
|
|
|
|
TEST_CHECK(x0 == 0x80000000);
|
|
TEST_CHECK(x1 == 0x4444444444444444);
|
|
TEST_CHECK(x2 == 0x4444444444444444);
|
|
free(data);
|
|
OK(uc_close(uc));
|
|
}
|
|
|
|
static void test_arm64_pc_wrap(void)
|
|
{
|
|
uc_engine *uc;
|
|
// add x1 x2
|
|
char add_x1_x2[] = "\x20\x00\x02\x8b";
|
|
// add x1 x3
|
|
char add_x1_x3[] = "\x20\x00\x03\x8b";
|
|
uint64_t x0, x1, x2, x3;
|
|
uint64_t pc = 0xFFFFFFFFFFFFFFFCULL;
|
|
uint64_t page = 0xFFFFFFFFFFFFF000ULL;
|
|
|
|
OK(uc_open(UC_ARCH_ARM64, UC_MODE_ARM, &uc));
|
|
OK(uc_mem_map(uc, page, 4096, UC_PROT_READ | UC_PROT_EXEC));
|
|
OK(uc_mem_write(uc, pc, add_x1_x2, sizeof(add_x1_x2) - 1));
|
|
|
|
x1 = 1;
|
|
x2 = 2;
|
|
OK(uc_reg_write(uc, UC_ARM64_REG_X1, &x1));
|
|
OK(uc_reg_write(uc, UC_ARM64_REG_X2, &x2));
|
|
|
|
OK(uc_emu_start(uc, pc, pc + 4, 0, 1));
|
|
|
|
OK(uc_mem_unmap(uc, page, 4096));
|
|
|
|
OK(uc_reg_read(uc, UC_ARM64_REG_X0, &x0));
|
|
|
|
TEST_CHECK((x0 == 1 + 2));
|
|
|
|
OK(uc_mem_map(uc, page, 4096, UC_PROT_READ | UC_PROT_EXEC));
|
|
OK(uc_mem_write(uc, pc, add_x1_x3, sizeof(add_x1_x3) - 1));
|
|
|
|
x1 = 5;
|
|
x2 = 0;
|
|
x3 = 5;
|
|
OK(uc_reg_write(uc, UC_ARM64_REG_X1, &x1));
|
|
OK(uc_reg_write(uc, UC_ARM64_REG_X2, &x2));
|
|
OK(uc_reg_write(uc, UC_ARM64_REG_X3, &x3));
|
|
|
|
OK(uc_emu_start(uc, pc, pc + 4, 0, 1));
|
|
|
|
OK(uc_mem_unmap(uc, page, 4096));
|
|
|
|
OK(uc_reg_read(uc, UC_ARM64_REG_X0, &x0));
|
|
|
|
TEST_CHECK((x0 == 5 + 5));
|
|
|
|
OK(uc_close(uc));
|
|
}
|
|
|
|
static void test_arm64_mem_prot_regress_hook_mem(uc_engine *uc,
|
|
uc_mem_type type,
|
|
uint64_t address, int size,
|
|
int64_t value, void *user_data)
|
|
{
|
|
// fprintf(stderr, "%s %p %d\n", (type == UC_MEM_WRITE) ? "UC_MEM_WRITE" :
|
|
// "UC_MEM_READ", (void *)address, size);
|
|
}
|
|
|
|
static bool test_arm64_mem_prot_regress_hook_prot(uc_engine *uc,
|
|
uc_mem_type type,
|
|
uint64_t address, int size,
|
|
int64_t value,
|
|
void *user_data)
|
|
{
|
|
// fprintf(stderr, "%s %p %d\n", (type == UC_MEM_WRITE_PROT) ?
|
|
// "UC_MEM_WRITE_PROT" : ((type == UC_MEM_FETCH_PROT) ? "UC_MEM_FETCH_PROT"
|
|
// : "UC_MEM_READ_PROT"), (void *)address, size);
|
|
return false;
|
|
}
|
|
|
|
static bool test_arm64_mem_prot_regress_hook_unm(uc_engine *uc,
|
|
uc_mem_type type,
|
|
uint64_t address, int size,
|
|
int64_t value, void *user_data)
|
|
{
|
|
// fprintf(stderr, "%s %p %d\n", (type == UC_MEM_WRITE_UNMAPPED) ?
|
|
// "UC_MEM_WRITE_UNMAPPED" : ((type == UC_MEM_FETCH_UNMAPPED) ?
|
|
// "UC_MEM_FETCH_UNMAPPED" : "UC_MEM_READ_UNMAPPED"), (void *)address,
|
|
// size);
|
|
return false;
|
|
}
|
|
|
|
// https://github.com/unicorn-engine/unicorn/issues/2078
|
|
static void test_arm64_mem_prot_regress(void)
|
|
{
|
|
const uint8_t code[] = {
|
|
0x08, 0x40, 0x5e, 0x78, // ldurh w8, [x0, #-0x1c]
|
|
};
|
|
|
|
uc_engine *uc;
|
|
OK(uc_open(UC_ARCH_ARM64, UC_MODE_ARM, &uc));
|
|
|
|
OK(uc_mem_map(uc, 0, 0x4000, UC_PROT_READ | UC_PROT_EXEC));
|
|
OK(uc_mem_map(uc, 0x4000, 0xC000, UC_PROT_READ | UC_PROT_WRITE));
|
|
OK(uc_mem_write(uc, 0, code, sizeof(code)));
|
|
uc_hook hh_mem;
|
|
OK(uc_hook_add(uc, &hh_mem, UC_HOOK_MEM_READ | UC_HOOK_MEM_WRITE,
|
|
test_arm64_mem_prot_regress_hook_mem, NULL, 1, 0));
|
|
|
|
uc_hook hh_prot;
|
|
OK(uc_hook_add(uc, &hh_prot, UC_HOOK_MEM_PROT,
|
|
test_arm64_mem_prot_regress_hook_prot, NULL, 1, 0));
|
|
|
|
uc_hook hh_unm;
|
|
OK(uc_hook_add(uc, &hh_unm, UC_HOOK_MEM_UNMAPPED,
|
|
test_arm64_mem_prot_regress_hook_unm, NULL, 1, 0));
|
|
|
|
const uint64_t value = 0x801b;
|
|
OK(uc_reg_write(uc, UC_ARM64_REG_X0, &value));
|
|
|
|
OK(uc_emu_start(uc, 0, sizeof(code), 0, 0));
|
|
|
|
OK(uc_close(uc));
|
|
}
|
|
|
|
static bool test_arm64_mem_read_write_cb(uc_engine *uc, int type,
|
|
uint64_t address, int size,
|
|
int64_t value, void *user_data)
|
|
{
|
|
uint64_t *count = (uint64_t *)user_data;
|
|
switch (type) {
|
|
case UC_MEM_READ:
|
|
count[0]++;
|
|
break;
|
|
case UC_MEM_WRITE:
|
|
count[1]++;
|
|
break;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
static void test_arm64_mem_hook_read_write(void)
|
|
{
|
|
uc_engine *uc;
|
|
// ldp x1, x2, [sp]
|
|
// stp x1, x2,[sp]
|
|
// ldp x1, x2, [sp]
|
|
// stp x1, x2,[sp]
|
|
const char code[] = {0xe1, 0x0b, 0x40, 0xa9, 0xe1, 0x0b, 0x00, 0xa9,
|
|
0xe1, 0x0b, 0x40, 0xa9, 0xe1, 0x0b, 0x00, 0xa9};
|
|
uint64_t r_sp;
|
|
r_sp = 0x16db6a040;
|
|
uc_hook hk;
|
|
uint64_t counter[2] = {0, 0};
|
|
|
|
uc_common_setup(&uc, UC_ARCH_ARM64, UC_MODE_ARM, code, sizeof(code),
|
|
UC_CPU_ARM64_A72);
|
|
|
|
uc_reg_write(uc, UC_ARM64_REG_SP, &r_sp);
|
|
uc_mem_map(uc, 0x16db68000, 1024 * 16, UC_PROT_ALL);
|
|
|
|
OK(uc_hook_add(uc, &hk, UC_HOOK_MEM_READ, test_arm64_mem_read_write_cb,
|
|
counter, 1, 0));
|
|
OK(uc_hook_add(uc, &hk, UC_HOOK_MEM_WRITE, test_arm64_mem_read_write_cb,
|
|
counter, 1, 0));
|
|
|
|
uc_assert_err(UC_ERR_OK, uc_emu_start(uc, code_start,
|
|
code_start + sizeof(code), 0, 0));
|
|
|
|
TEST_CHECK(counter[0] == 4 && counter[1] == 4);
|
|
OK(uc_close(uc));
|
|
}
|
|
|
|
static void test_arm64_pc_guarantee(void)
|
|
{
|
|
uc_engine *uc;
|
|
// ks.asm("mov x0, #1; mov x1, #2; ldr x0, [x1]")
|
|
const char code[] = "\x20\x00\x80\xd2\x41\x00\x80\xd2\x20\x00\x40\xf9";
|
|
uint64_t rip;
|
|
|
|
uc_common_setup(&uc, UC_ARCH_ARM64, UC_MODE_ARM, code, sizeof(code),
|
|
UC_CPU_ARM64_A72);
|
|
|
|
uc_assert_err(UC_ERR_READ_UNMAPPED, uc_emu_start(uc, code_start,
|
|
code_start + sizeof(code) - 1, 0, 0));
|
|
|
|
OK(uc_reg_read(uc, UC_ARM64_REG_PC, (void*)&rip));
|
|
TEST_CHECK(rip == code_start + 8);
|
|
OK(uc_close(uc));
|
|
}
|
|
|
|
TEST_LIST = {{"test_arm64_until", test_arm64_until},
|
|
{"test_arm64_code_patching", test_arm64_code_patching},
|
|
{"test_arm64_code_patching_count", test_arm64_code_patching_count},
|
|
{"test_arm64_v8_pac", test_arm64_v8_pac},
|
|
{"test_arm64_read_sctlr", test_arm64_read_sctlr},
|
|
{"test_arm64_mrs_hook", test_arm64_mrs_hook},
|
|
{"test_arm64_correct_address_in_small_jump_hook",
|
|
test_arm64_correct_address_in_small_jump_hook},
|
|
{"test_arm64_correct_address_in_long_jump_hook",
|
|
test_arm64_correct_address_in_long_jump_hook},
|
|
{"test_arm64_block_sync_pc", test_arm64_block_sync_pc},
|
|
{"test_arm64_block_invalid_mem_read_write_sync",
|
|
test_arm64_block_invalid_mem_read_write_sync},
|
|
{"test_arm64_mmu", test_arm64_mmu},
|
|
{"test_arm64_pc_wrap", test_arm64_pc_wrap},
|
|
{"test_arm64_mem_prot_regress", test_arm64_mem_prot_regress},
|
|
{"test_arm64_mem_hook_read_write", test_arm64_mem_hook_read_write},
|
|
{"test_arm64_pc_guarantee", test_arm64_pc_guarantee},
|
|
{NULL, NULL}};
|