mirror of
https://github.com/yuzu-emu/unicorn.git
synced 2025-01-09 09:25:39 +00:00
Merge branch 'bugfix-x86' of https://github.com/vardyh/unicorn into vardyh-bugfix-x86
This commit is contained in:
commit
9d52d580d2
|
@ -6005,9 +6005,15 @@ static target_ulong disas_insn(CPUX86State *env, DisasContext *s,
|
||||||
ot = mo_b_d(b, dflag);
|
ot = mo_b_d(b, dflag);
|
||||||
modrm = cpu_ldub_code(env, s->pc++);
|
modrm = cpu_ldub_code(env, s->pc++);
|
||||||
mod = (modrm >> 6) & 3;
|
mod = (modrm >> 6) & 3;
|
||||||
|
reg = ((modrm >> 3) & 7) | rex_r;
|
||||||
if (mod != 3) {
|
if (mod != 3) {
|
||||||
|
if (reg != 0)
|
||||||
|
goto illegal_op;
|
||||||
s->rip_offset = insn_const_size(ot);
|
s->rip_offset = insn_const_size(ot);
|
||||||
gen_lea_modrm(env, s, modrm);
|
gen_lea_modrm(env, s, modrm);
|
||||||
|
} else {
|
||||||
|
if (reg != 0 && reg != 7)
|
||||||
|
goto illegal_op;
|
||||||
}
|
}
|
||||||
val = insn_get(env, s, ot);
|
val = insn_get(env, s, ot);
|
||||||
tcg_gen_movi_tl(tcg_ctx, *cpu_T[0], val);
|
tcg_gen_movi_tl(tcg_ctx, *cpu_T[0], val);
|
||||||
|
|
|
@ -745,6 +745,64 @@ static void test_i386_context_save(void)
|
||||||
uc_close(uc);
|
uc_close(uc);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void test_i386_invalid_c6c7(void)
|
||||||
|
{
|
||||||
|
uc_engine *uc;
|
||||||
|
uc_err err;
|
||||||
|
uint8_t codebuf[16] = { 0 };
|
||||||
|
uint8_t opcodes[] = { 0xc6, 0xc7 };
|
||||||
|
bool valid_masks[4][8] = {
|
||||||
|
{ true, false, false, false, false, false, false, false },
|
||||||
|
{ true, false, false, false, false, false, false, false },
|
||||||
|
{ true, false, false, false, false, false, false, false },
|
||||||
|
{ true, false, false, false, false, false, false, true },
|
||||||
|
};
|
||||||
|
int i, j, k;
|
||||||
|
|
||||||
|
printf("===================================\n");
|
||||||
|
printf("Emulate i386 C6/C7 opcodes\n");
|
||||||
|
|
||||||
|
// Initialize emulator in X86-32bit mode
|
||||||
|
err = uc_open(UC_ARCH_X86, UC_MODE_32, &uc);
|
||||||
|
if (err) {
|
||||||
|
printf("Failed on uc_open() with error returned: %u\n", err);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// map 2MB memory for this emulation
|
||||||
|
uc_mem_map(uc, ADDRESS, 2 * 1024 * 1024, UC_PROT_ALL);
|
||||||
|
|
||||||
|
for (i = 0; i < 2; ++i) {
|
||||||
|
// set opcode
|
||||||
|
codebuf[0] = opcodes[i];
|
||||||
|
|
||||||
|
for (j = 0; j < 4; ++j) {
|
||||||
|
for (k = 0; k < 8; ++k) {
|
||||||
|
// set Mod bits
|
||||||
|
codebuf[1] = (uint8_t) (j << 6);
|
||||||
|
// set Reg bits
|
||||||
|
codebuf[1] |= (uint8_t) (k << 3);
|
||||||
|
|
||||||
|
// perform validation
|
||||||
|
if (uc_mem_write(uc, ADDRESS, codebuf, sizeof(codebuf))) {
|
||||||
|
printf("Failed to write emulation code to memory, quit!\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
err = uc_emu_start(uc, ADDRESS, ADDRESS + sizeof(codebuf), 0, 0);
|
||||||
|
if ((err != UC_ERR_INSN_INVALID) ^ valid_masks[j][k]) {
|
||||||
|
printf("Unexpected uc_emu_start() error returned %u: %s\n",
|
||||||
|
err, uc_strerror(err));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
printf(">>> Emulation done.\n");
|
||||||
|
|
||||||
|
uc_close(uc);
|
||||||
|
}
|
||||||
|
|
||||||
static void test_x86_64(void)
|
static void test_x86_64(void)
|
||||||
{
|
{
|
||||||
uc_engine *uc;
|
uc_engine *uc;
|
||||||
|
@ -981,6 +1039,7 @@ int main(int argc, char **argv, char **envp)
|
||||||
test_i386_invalid_mem_read();
|
test_i386_invalid_mem_read();
|
||||||
test_i386_invalid_mem_write();
|
test_i386_invalid_mem_write();
|
||||||
test_i386_jump_invalid();
|
test_i386_jump_invalid();
|
||||||
|
test_i386_invalid_c6c7();
|
||||||
}
|
}
|
||||||
else if (!strcmp(argv[1], "-64")) {
|
else if (!strcmp(argv[1], "-64")) {
|
||||||
test_x86_64();
|
test_x86_64();
|
||||||
|
@ -1001,6 +1060,7 @@ int main(int argc, char **argv, char **envp)
|
||||||
test_i386_invalid_mem_read();
|
test_i386_invalid_mem_read();
|
||||||
test_i386_invalid_mem_write();
|
test_i386_invalid_mem_write();
|
||||||
test_i386_jump_invalid();
|
test_i386_jump_invalid();
|
||||||
|
test_i386_invalid_c6c7();
|
||||||
test_x86_64();
|
test_x86_64();
|
||||||
test_x86_64_syscall();
|
test_x86_64_syscall();
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue