2009-05-21 17:48:31 +02:00
|
|
|
/* gameplaySP
|
|
|
|
*
|
|
|
|
* Copyright (C) 2006 Exophase <exophase@gmail.com>
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License as
|
|
|
|
* published by the Free Software Foundation; either version 2 of
|
|
|
|
* the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef X86_EMIT_H
|
|
|
|
#define X86_EMIT_H
|
|
|
|
|
|
|
|
u32 x86_update_gba(u32 pc);
|
|
|
|
|
|
|
|
// Although these are defined as a function, don't call them as
|
|
|
|
// such (jump to it instead)
|
|
|
|
void x86_indirect_branch_arm(u32 address);
|
|
|
|
void x86_indirect_branch_thumb(u32 address);
|
|
|
|
void x86_indirect_branch_dual(u32 address);
|
|
|
|
|
2021-03-06 21:15:22 +01:00
|
|
|
void function_cc execute_store_cpsr(u32 new_cpsr, u32 store_mask);
|
2009-05-21 17:48:31 +02:00
|
|
|
|
|
|
|
typedef enum
|
|
|
|
{
|
|
|
|
x86_reg_number_eax,
|
|
|
|
x86_reg_number_ecx,
|
|
|
|
x86_reg_number_edx,
|
|
|
|
x86_reg_number_ebx,
|
|
|
|
x86_reg_number_esp,
|
|
|
|
x86_reg_number_ebp,
|
|
|
|
x86_reg_number_esi,
|
|
|
|
x86_reg_number_edi
|
|
|
|
} x86_reg_number;
|
|
|
|
|
|
|
|
#define x86_emit_byte(value) \
|
|
|
|
*translation_ptr = value; \
|
|
|
|
translation_ptr++ \
|
|
|
|
|
|
|
|
#define x86_emit_dword(value) \
|
|
|
|
*((u32 *)translation_ptr) = value; \
|
|
|
|
translation_ptr += 4 \
|
|
|
|
|
|
|
|
typedef enum
|
|
|
|
{
|
|
|
|
x86_mod_mem = 0,
|
|
|
|
x86_mod_mem_disp8 = 1,
|
|
|
|
x86_mod_mem_disp32 = 2,
|
|
|
|
x86_mod_reg = 3
|
|
|
|
} x86_mod;
|
|
|
|
|
|
|
|
#define x86_emit_mod_rm(mod, rm, spare) \
|
|
|
|
x86_emit_byte((mod << 6) | (spare << 3) | rm) \
|
|
|
|
|
2021-09-17 20:00:37 +02:00
|
|
|
#define x86_emit_sib(scale, ridx, rbase) \
|
|
|
|
x86_emit_byte(((scale) << 6) | ((ridx) << 3) | (rbase)) \
|
|
|
|
|
2009-05-21 17:48:31 +02:00
|
|
|
#define x86_emit_mem_op(dest, base, offset) \
|
|
|
|
if(offset == 0) \
|
|
|
|
{ \
|
|
|
|
x86_emit_mod_rm(x86_mod_mem, base, dest); \
|
|
|
|
} \
|
2021-09-17 20:00:37 +02:00
|
|
|
else if(((s32)offset < 127) && ((s32)offset > -128)) \
|
2009-05-21 17:48:31 +02:00
|
|
|
{ \
|
|
|
|
x86_emit_mod_rm(x86_mod_mem_disp8, base, dest); \
|
|
|
|
x86_emit_byte((s8)offset); \
|
|
|
|
} \
|
|
|
|
else \
|
|
|
|
{ \
|
|
|
|
x86_emit_mod_rm(x86_mod_mem_disp32, base, dest); \
|
|
|
|
x86_emit_dword(offset); \
|
|
|
|
} \
|
|
|
|
|
2021-09-17 20:00:37 +02:00
|
|
|
#define x86_emit_mem_sib_op(dest, base, ridx, scale, offset) \
|
|
|
|
if(offset == 0) \
|
|
|
|
{ \
|
|
|
|
x86_emit_mod_rm(x86_mod_mem, 0x4, dest); \
|
|
|
|
x86_emit_sib(scale, ridx, base); \
|
|
|
|
} \
|
|
|
|
else if(((s32)offset < 127) && ((s32)offset > -128)) \
|
|
|
|
{ \
|
|
|
|
x86_emit_mod_rm(x86_mod_mem_disp8, 0x4, dest); \
|
|
|
|
x86_emit_sib(scale, ridx, base); \
|
|
|
|
x86_emit_byte((s8)offset); \
|
|
|
|
} \
|
|
|
|
else \
|
|
|
|
{ \
|
|
|
|
x86_emit_mod_rm(x86_mod_mem_disp32, 0x4, dest); \
|
|
|
|
x86_emit_sib(scale, ridx, base); \
|
|
|
|
x86_emit_dword(offset); \
|
|
|
|
} \
|
|
|
|
|
2009-05-21 17:48:31 +02:00
|
|
|
#define x86_emit_reg_op(dest, source) \
|
|
|
|
x86_emit_mod_rm(x86_mod_reg, source, dest) \
|
|
|
|
|
|
|
|
|
|
|
|
typedef enum
|
|
|
|
{
|
|
|
|
x86_opcode_mov_rm_reg = 0x89,
|
|
|
|
x86_opcode_mov_reg_rm = 0x8B,
|
|
|
|
x86_opcode_mov_reg_imm = 0xB8,
|
|
|
|
x86_opcode_mov_rm_imm = 0x00C7,
|
|
|
|
x86_opcode_ror_reg_imm = 0x01C1,
|
|
|
|
x86_opcode_shl_reg_imm = 0x04C1,
|
|
|
|
x86_opcode_shr_reg_imm = 0x05C1,
|
|
|
|
x86_opcode_sar_reg_imm = 0x07C1,
|
2021-08-31 21:18:22 +02:00
|
|
|
x86_opcode_ror_reg_rm = 0x01D3,
|
|
|
|
x86_opcode_rcr_reg_rm = 0x03D3,
|
|
|
|
x86_opcode_shl_reg_rm = 0x04D3,
|
|
|
|
x86_opcode_shr_reg_rm = 0x05D3,
|
|
|
|
x86_opcode_sar_reg_rm = 0x07D3,
|
|
|
|
x86_opcode_rcr_reg1 = 0x03D1,
|
2009-05-21 17:48:31 +02:00
|
|
|
x86_opcode_call_offset = 0xE8,
|
|
|
|
x86_opcode_ret = 0xC3,
|
|
|
|
x86_opcode_test_rm_imm = 0x00F7,
|
|
|
|
x86_opcode_test_reg_rm = 0x85,
|
2021-08-31 00:42:47 +02:00
|
|
|
x86_opcode_not_rm = 0x02F7,
|
2009-05-21 17:48:31 +02:00
|
|
|
x86_opcode_mul_eax_rm = 0x04F7,
|
|
|
|
x86_opcode_imul_eax_rm = 0x05F7,
|
|
|
|
x86_opcode_idiv_eax_rm = 0x07F7,
|
|
|
|
x86_opcode_add_rm_imm = 0x0081,
|
|
|
|
x86_opcode_and_rm_imm = 0x0481,
|
|
|
|
x86_opcode_sub_rm_imm = 0x0581,
|
|
|
|
x86_opcode_xor_rm_imm = 0x0681,
|
|
|
|
x86_opcode_add_reg_rm = 0x03,
|
|
|
|
x86_opcode_adc_reg_rm = 0x13,
|
2021-08-31 00:42:47 +02:00
|
|
|
x86_opcode_and_reg_rm = 0x23,
|
2009-05-21 17:48:31 +02:00
|
|
|
x86_opcode_or_reg_rm = 0x0B,
|
|
|
|
x86_opcode_sub_reg_rm = 0x2B,
|
2021-08-31 00:42:47 +02:00
|
|
|
x86_opcode_sbb_reg_rm = 0x1B,
|
2009-05-21 17:48:31 +02:00
|
|
|
x86_opcode_xor_reg_rm = 0x33,
|
|
|
|
x86_opcode_cmp_reg_rm = 0x39,
|
2021-08-31 21:18:22 +02:00
|
|
|
x86_opcode_cmp_rm_imm = 0x0781,
|
2009-05-21 17:48:31 +02:00
|
|
|
x86_opcode_lea_reg_rm = 0x8D,
|
|
|
|
x86_opcode_j = 0x80,
|
2021-09-03 01:01:37 +02:00
|
|
|
x86_opcode_cdq = 0x99,
|
|
|
|
x86_opcode_jmp = 0xE9,
|
|
|
|
x86_opcode_jmp_reg = 0x04FF,
|
|
|
|
x86_opcode_ext = 0x0F
|
|
|
|
} x86_opcodes;
|
|
|
|
|
|
|
|
typedef enum
|
|
|
|
{
|
2021-08-31 00:42:47 +02:00
|
|
|
x86_opcode_seto = 0x90,
|
|
|
|
x86_opcode_setc = 0x92,
|
|
|
|
x86_opcode_setnc = 0x93,
|
|
|
|
x86_opcode_setz = 0x94,
|
2021-08-31 21:18:22 +02:00
|
|
|
x86_opcode_setnz = 0x95,
|
2021-08-31 00:42:47 +02:00
|
|
|
x86_opcode_sets = 0x98,
|
2021-08-31 21:18:22 +02:00
|
|
|
x86_opcode_setns = 0x99,
|
2021-09-03 01:01:37 +02:00
|
|
|
} x86_ext_opcodes;
|
2009-05-21 17:48:31 +02:00
|
|
|
|
|
|
|
typedef enum
|
|
|
|
{
|
|
|
|
x86_condition_code_o = 0x00,
|
|
|
|
x86_condition_code_no = 0x01,
|
|
|
|
x86_condition_code_c = 0x02,
|
|
|
|
x86_condition_code_nc = 0x03,
|
|
|
|
x86_condition_code_z = 0x04,
|
|
|
|
x86_condition_code_nz = 0x05,
|
|
|
|
x86_condition_code_na = 0x06,
|
|
|
|
x86_condition_code_a = 0x07,
|
|
|
|
x86_condition_code_s = 0x08,
|
|
|
|
x86_condition_code_ns = 0x09,
|
|
|
|
x86_condition_code_p = 0x0A,
|
|
|
|
x86_condition_code_np = 0x0B,
|
|
|
|
x86_condition_code_l = 0x0C,
|
|
|
|
x86_condition_code_nl = 0x0D,
|
|
|
|
x86_condition_code_ng = 0x0E,
|
|
|
|
x86_condition_code_g = 0x0F
|
|
|
|
} x86_condition_codes;
|
|
|
|
|
|
|
|
#define x86_relative_offset(source, offset, next) \
|
2021-11-06 12:17:50 +01:00
|
|
|
((u32)((uintptr_t)offset - ((uintptr_t)source + next))) \
|
2009-05-21 17:48:31 +02:00
|
|
|
|
|
|
|
#define x86_unequal_operands(op_a, op_b) \
|
|
|
|
(x86_reg_number_##op_a != x86_reg_number_##op_b) \
|
|
|
|
|
|
|
|
#define x86_emit_opcode_1b_reg(opcode, dest, source) \
|
|
|
|
{ \
|
|
|
|
x86_emit_byte(x86_opcode_##opcode); \
|
|
|
|
x86_emit_reg_op(x86_reg_number_##dest, x86_reg_number_##source); \
|
|
|
|
} \
|
|
|
|
|
|
|
|
#define x86_emit_opcode_1b_mem(opcode, dest, base, offset) \
|
|
|
|
{ \
|
|
|
|
x86_emit_byte(x86_opcode_##opcode); \
|
|
|
|
x86_emit_mem_op(x86_reg_number_##dest, x86_reg_number_##base, offset); \
|
|
|
|
} \
|
|
|
|
|
2021-09-17 20:00:37 +02:00
|
|
|
#define x86_emit_opcode_1b_mem_sib(opcode, dest, base, ridx, scale, offset) \
|
|
|
|
{ \
|
|
|
|
x86_emit_byte(x86_opcode_##opcode); \
|
|
|
|
x86_emit_mem_sib_op(x86_reg_number_##dest, x86_reg_number_##base, \
|
|
|
|
x86_reg_number_##ridx, scale, offset); \
|
|
|
|
} \
|
|
|
|
|
2009-05-21 17:48:31 +02:00
|
|
|
#define x86_emit_opcode_1b(opcode, reg) \
|
|
|
|
x86_emit_byte(x86_opcode_##opcode | x86_reg_number_##reg) \
|
|
|
|
|
|
|
|
#define x86_emit_opcode_1b_ext_reg(opcode, dest) \
|
|
|
|
x86_emit_byte(x86_opcode_##opcode & 0xFF); \
|
|
|
|
x86_emit_reg_op(x86_opcode_##opcode >> 8, x86_reg_number_##dest) \
|
|
|
|
|
|
|
|
#define x86_emit_opcode_1b_ext_mem(opcode, base, offset) \
|
|
|
|
x86_emit_byte(x86_opcode_##opcode & 0xFF); \
|
|
|
|
x86_emit_mem_op(x86_opcode_##opcode >> 8, x86_reg_number_##base, offset) \
|
|
|
|
|
|
|
|
#define x86_emit_mov_reg_mem(dest, base, offset) \
|
|
|
|
x86_emit_opcode_1b_mem(mov_reg_rm, dest, base, offset) \
|
|
|
|
|
2021-09-17 20:00:37 +02:00
|
|
|
#define x86_emit_mov_reg_mem_idx(dest, base, scale, index, offset) \
|
|
|
|
x86_emit_opcode_1b_mem_sib(mov_reg_rm, dest, base, index, scale, offset) \
|
|
|
|
|
2021-11-02 20:50:41 +01:00
|
|
|
#define x86_emit_mov_mem_idx_reg(dest, base, scale, index, offset) \
|
|
|
|
x86_emit_opcode_1b_mem_sib(mov_rm_reg, dest, base, index, scale, offset) \
|
|
|
|
|
2009-05-21 17:48:31 +02:00
|
|
|
#define x86_emit_mov_mem_reg(source, base, offset) \
|
|
|
|
x86_emit_opcode_1b_mem(mov_rm_reg, source, base, offset) \
|
|
|
|
|
2021-08-31 00:42:47 +02:00
|
|
|
#define x86_emit_setcc_mem(ccode, base, offset) \
|
|
|
|
x86_emit_byte(x86_opcode_ext); \
|
|
|
|
x86_emit_opcode_1b_mem(set##ccode, eax, base, offset); \
|
|
|
|
|
|
|
|
#define x86_emit_add_reg_mem(dst, base, offset) \
|
|
|
|
x86_emit_opcode_1b_mem(add_reg_rm, dst, base, offset); \
|
|
|
|
|
2021-09-17 19:18:50 +02:00
|
|
|
#define x86_emit_or_reg_mem(dst, base, offset) \
|
|
|
|
x86_emit_opcode_1b_mem(or_reg_rm, dst, base, offset); \
|
|
|
|
|
|
|
|
#define x86_emit_xor_reg_mem(dst, base, offset) \
|
|
|
|
x86_emit_opcode_1b_mem(xor_reg_rm, dst, base, offset); \
|
|
|
|
|
|
|
|
#define x86_emit_cmp_reg_mem(rega, base, offset) \
|
|
|
|
x86_emit_opcode_1b_mem(cmp_reg_rm, rega, base, offset); \
|
|
|
|
|
|
|
|
#define x86_emit_test_reg_mem(rega, base, offset) \
|
|
|
|
x86_emit_opcode_1b_mem(test_reg_rm, rega, base, offset); \
|
|
|
|
|
2009-05-21 17:48:31 +02:00
|
|
|
#define x86_emit_mov_reg_reg(dest, source) \
|
|
|
|
if(x86_unequal_operands(dest, source)) \
|
|
|
|
{ \
|
|
|
|
x86_emit_opcode_1b_reg(mov_reg_rm, dest, source) \
|
|
|
|
} \
|
|
|
|
|
|
|
|
#define x86_emit_mov_reg_imm(dest, imm) \
|
|
|
|
x86_emit_opcode_1b(mov_reg_imm, dest); \
|
|
|
|
x86_emit_dword(imm) \
|
|
|
|
|
|
|
|
#define x86_emit_mov_mem_imm(imm, base, offset) \
|
|
|
|
x86_emit_opcode_1b_ext_mem(mov_rm_imm, base, offset); \
|
|
|
|
x86_emit_dword(imm) \
|
|
|
|
|
2021-11-07 19:19:19 +01:00
|
|
|
#define x86_emit_and_mem_imm(imm, base, offset) \
|
|
|
|
x86_emit_opcode_1b_ext_mem(and_rm_imm, base, offset); \
|
|
|
|
x86_emit_dword(imm) \
|
|
|
|
|
2009-05-21 17:48:31 +02:00
|
|
|
#define x86_emit_shl_reg_imm(dest, imm) \
|
|
|
|
x86_emit_opcode_1b_ext_reg(shl_reg_imm, dest); \
|
|
|
|
x86_emit_byte(imm) \
|
|
|
|
|
|
|
|
#define x86_emit_shr_reg_imm(dest, imm) \
|
|
|
|
x86_emit_opcode_1b_ext_reg(shr_reg_imm, dest); \
|
|
|
|
x86_emit_byte(imm) \
|
|
|
|
|
|
|
|
#define x86_emit_sar_reg_imm(dest, imm) \
|
|
|
|
x86_emit_opcode_1b_ext_reg(sar_reg_imm, dest); \
|
|
|
|
x86_emit_byte(imm) \
|
|
|
|
|
|
|
|
#define x86_emit_ror_reg_imm(dest, imm) \
|
|
|
|
x86_emit_opcode_1b_ext_reg(ror_reg_imm, dest); \
|
|
|
|
x86_emit_byte(imm) \
|
|
|
|
|
|
|
|
#define x86_emit_add_reg_reg(dest, source) \
|
|
|
|
x86_emit_opcode_1b_reg(add_reg_rm, dest, source) \
|
|
|
|
|
|
|
|
#define x86_emit_adc_reg_reg(dest, source) \
|
|
|
|
x86_emit_opcode_1b_reg(adc_reg_rm, dest, source) \
|
|
|
|
|
|
|
|
#define x86_emit_sub_reg_reg(dest, source) \
|
|
|
|
x86_emit_opcode_1b_reg(sub_reg_rm, dest, source) \
|
|
|
|
|
2021-08-31 00:42:47 +02:00
|
|
|
#define x86_emit_sbb_reg_reg(dest, source) \
|
|
|
|
x86_emit_opcode_1b_reg(sbb_reg_rm, dest, source) \
|
|
|
|
|
|
|
|
#define x86_emit_and_reg_reg(dest, source) \
|
|
|
|
x86_emit_opcode_1b_reg(and_reg_rm, dest, source) \
|
|
|
|
|
2009-05-21 17:48:31 +02:00
|
|
|
#define x86_emit_or_reg_reg(dest, source) \
|
|
|
|
x86_emit_opcode_1b_reg(or_reg_rm, dest, source) \
|
|
|
|
|
|
|
|
#define x86_emit_xor_reg_reg(dest, source) \
|
|
|
|
x86_emit_opcode_1b_reg(xor_reg_rm, dest, source) \
|
|
|
|
|
|
|
|
#define x86_emit_add_reg_imm(dest, imm) \
|
|
|
|
if(imm != 0) \
|
|
|
|
{ \
|
|
|
|
x86_emit_opcode_1b_ext_reg(add_rm_imm, dest); \
|
|
|
|
x86_emit_dword(imm); \
|
|
|
|
} \
|
|
|
|
|
|
|
|
#define x86_emit_sub_reg_imm(dest, imm) \
|
|
|
|
if(imm != 0) \
|
|
|
|
{ \
|
|
|
|
x86_emit_opcode_1b_ext_reg(sub_rm_imm, dest); \
|
|
|
|
x86_emit_dword(imm); \
|
|
|
|
} \
|
|
|
|
|
|
|
|
#define x86_emit_and_reg_imm(dest, imm) \
|
|
|
|
x86_emit_opcode_1b_ext_reg(and_rm_imm, dest); \
|
|
|
|
x86_emit_dword(imm) \
|
|
|
|
|
|
|
|
#define x86_emit_xor_reg_imm(dest, imm) \
|
|
|
|
x86_emit_opcode_1b_ext_reg(xor_rm_imm, dest); \
|
|
|
|
x86_emit_dword(imm) \
|
|
|
|
|
|
|
|
#define x86_emit_test_reg_imm(dest, imm) \
|
|
|
|
x86_emit_opcode_1b_ext_reg(test_rm_imm, dest); \
|
|
|
|
x86_emit_dword(imm) \
|
|
|
|
|
|
|
|
#define x86_emit_cmp_reg_reg(dest, source) \
|
|
|
|
x86_emit_opcode_1b_reg(cmp_reg_rm, dest, source) \
|
|
|
|
|
|
|
|
#define x86_emit_test_reg_reg(dest, source) \
|
|
|
|
x86_emit_opcode_1b_reg(test_reg_rm, dest, source) \
|
|
|
|
|
|
|
|
#define x86_emit_cmp_reg_imm(dest, imm) \
|
|
|
|
x86_emit_opcode_1b_ext_reg(cmp_rm_imm, dest); \
|
|
|
|
x86_emit_dword(imm) \
|
|
|
|
|
2021-08-31 21:18:22 +02:00
|
|
|
#define x86_emit_rot_reg_reg(type, dest) \
|
|
|
|
x86_emit_opcode_1b_ext_reg(type##_reg_rm, dest) \
|
|
|
|
|
|
|
|
#define x86_emit_rot_reg1(type, dest) \
|
|
|
|
x86_emit_opcode_1b_ext_reg(type##_reg1, dest) \
|
|
|
|
|
|
|
|
#define x86_emit_shr_reg_reg(dest) \
|
|
|
|
x86_emit_opcode_1b_ext_reg(shr_reg_rm, dest) \
|
|
|
|
|
|
|
|
#define x86_emit_sar_reg_reg(dest) \
|
|
|
|
x86_emit_opcode_1b_ext_reg(sar_reg_rm, dest) \
|
|
|
|
|
|
|
|
#define x86_emit_shl_reg_reg(dest) \
|
|
|
|
x86_emit_opcode_1b_ext_reg(shl_reg_rm, dest) \
|
|
|
|
|
2009-05-21 17:48:31 +02:00
|
|
|
#define x86_emit_mul_eax_reg(source) \
|
|
|
|
x86_emit_opcode_1b_ext_reg(mul_eax_rm, source) \
|
|
|
|
|
|
|
|
#define x86_emit_imul_eax_reg(source) \
|
|
|
|
x86_emit_opcode_1b_ext_reg(imul_eax_rm, source) \
|
|
|
|
|
|
|
|
#define x86_emit_idiv_eax_reg(source) \
|
|
|
|
x86_emit_opcode_1b_ext_reg(idiv_eax_rm, source) \
|
|
|
|
|
2021-08-31 00:42:47 +02:00
|
|
|
#define x86_emit_not_reg(srcdst) \
|
|
|
|
x86_emit_opcode_1b_ext_reg(not_rm, srcdst) \
|
|
|
|
|
2021-09-03 01:01:37 +02:00
|
|
|
#define x86_emit_cdq() \
|
|
|
|
x86_emit_byte(x86_opcode_cdq) \
|
2009-05-21 17:48:31 +02:00
|
|
|
|
|
|
|
#define x86_emit_call_offset(relative_offset) \
|
|
|
|
x86_emit_byte(x86_opcode_call_offset); \
|
|
|
|
x86_emit_dword(relative_offset) \
|
|
|
|
|
|
|
|
#define x86_emit_ret() \
|
|
|
|
x86_emit_byte(x86_opcode_ret) \
|
|
|
|
|
|
|
|
#define x86_emit_lea_reg_mem(dest, base, offset) \
|
|
|
|
x86_emit_opcode_1b_mem(lea_reg_rm, dest, base, offset) \
|
|
|
|
|
|
|
|
#define x86_emit_j_filler(condition_code, writeback_location) \
|
|
|
|
x86_emit_byte(x86_opcode_ext); \
|
|
|
|
x86_emit_byte(x86_opcode_j | condition_code); \
|
|
|
|
(writeback_location) = translation_ptr; \
|
|
|
|
translation_ptr += 4 \
|
|
|
|
|
|
|
|
#define x86_emit_j_offset(condition_code, offset) \
|
|
|
|
x86_emit_byte(x86_opcode_ext); \
|
|
|
|
x86_emit_byte(x86_opcode_j | condition_code); \
|
|
|
|
x86_emit_dword(offset) \
|
|
|
|
|
|
|
|
#define x86_emit_jmp_filler(writeback_location) \
|
|
|
|
x86_emit_byte(x86_opcode_jmp); \
|
|
|
|
(writeback_location) = translation_ptr; \
|
|
|
|
translation_ptr += 4 \
|
|
|
|
|
|
|
|
#define x86_emit_jmp_offset(offset) \
|
|
|
|
x86_emit_byte(x86_opcode_jmp); \
|
|
|
|
x86_emit_dword(offset) \
|
|
|
|
|
|
|
|
#define x86_emit_jmp_reg(source) \
|
|
|
|
x86_emit_opcode_1b_ext_reg(jmp_reg, source) \
|
|
|
|
|
2021-11-02 20:11:46 +01:00
|
|
|
#define reg_base ebx // Saved register
|
|
|
|
#define reg_cycles ebp // Saved register
|
2009-05-21 17:48:31 +02:00
|
|
|
#define reg_a0 eax
|
|
|
|
#define reg_a1 edx
|
|
|
|
#define reg_a2 ecx
|
2021-11-04 20:22:42 +01:00
|
|
|
#define reg_t0 esi
|
2009-05-21 17:48:31 +02:00
|
|
|
#define reg_rv eax
|
|
|
|
|
2021-11-06 12:17:50 +01:00
|
|
|
#if defined(_WIN64)
|
|
|
|
#define reg_arg0 ecx
|
|
|
|
#define reg_arg1 edx
|
|
|
|
#elif defined(__x86_64__) || defined(__amd64__)
|
|
|
|
#define reg_arg0 edi
|
|
|
|
#define reg_arg1 esi
|
|
|
|
#else
|
|
|
|
#define reg_arg0 eax
|
|
|
|
#define reg_arg1 edx
|
|
|
|
#endif
|
|
|
|
|
2021-09-17 20:00:37 +02:00
|
|
|
/* Offsets from reg_base, see stub.S */
|
|
|
|
#define SPSR_BASE_OFF 0xA9100
|
|
|
|
|
2021-08-31 21:18:22 +02:00
|
|
|
#define generate_test_imm(ireg, imm) \
|
|
|
|
x86_emit_test_reg_imm(reg_##ireg, imm); \
|
|
|
|
|
2021-09-17 19:18:50 +02:00
|
|
|
#define generate_test_memreg(ireg_ref, arm_reg_src) \
|
|
|
|
x86_emit_test_reg_mem(reg_##ireg_ref, reg_base, arm_reg_src * 4) \
|
|
|
|
|
|
|
|
#define generate_cmp_memreg(ireg_ref, arm_reg_src) \
|
|
|
|
x86_emit_cmp_reg_mem(reg_##ireg_ref, reg_base, arm_reg_src * 4) \
|
|
|
|
|
2021-08-31 21:18:22 +02:00
|
|
|
#define generate_cmp_imm(ireg, imm) \
|
|
|
|
x86_emit_cmp_reg_imm(reg_##ireg, imm) \
|
|
|
|
|
2021-09-17 19:18:50 +02:00
|
|
|
#define generate_cmp_reg(ireg, ireg2) \
|
|
|
|
x86_emit_cmp_reg_reg(reg_##ireg, reg_##ireg2) \
|
|
|
|
|
2021-08-31 00:42:47 +02:00
|
|
|
#define generate_update_flag(condcode, regnum) \
|
|
|
|
x86_emit_setcc_mem(condcode, reg_base, regnum * 4) \
|
|
|
|
|
2021-09-17 20:00:37 +02:00
|
|
|
#define generate_load_spsr(ireg, idxr) \
|
|
|
|
x86_emit_mov_reg_mem_idx(reg_##ireg, reg_base, 2, reg_##idxr, SPSR_BASE_OFF);
|
|
|
|
|
2021-11-02 20:50:41 +01:00
|
|
|
#define generate_store_spsr(ireg, idxr) \
|
|
|
|
x86_emit_mov_mem_idx_reg(reg_##ireg, reg_base, 2, reg_##idxr, SPSR_BASE_OFF);
|
|
|
|
|
2009-05-21 17:48:31 +02:00
|
|
|
#define generate_load_reg(ireg, reg_index) \
|
|
|
|
x86_emit_mov_reg_mem(reg_##ireg, reg_base, reg_index * 4); \
|
|
|
|
|
|
|
|
#define generate_load_pc(ireg, new_pc) \
|
2021-05-07 20:41:54 +02:00
|
|
|
x86_emit_mov_reg_imm(reg_##ireg, (new_pc)) \
|
2009-05-21 17:48:31 +02:00
|
|
|
|
|
|
|
#define generate_load_imm(ireg, imm) \
|
|
|
|
x86_emit_mov_reg_imm(reg_##ireg, imm) \
|
|
|
|
|
|
|
|
#define generate_store_reg(ireg, reg_index) \
|
2021-11-02 20:50:41 +01:00
|
|
|
x86_emit_mov_mem_reg(reg_##ireg, reg_base, (reg_index) * 4) \
|
|
|
|
|
|
|
|
#define generate_store_reg_i32(imm32, reg_index) \
|
|
|
|
x86_emit_mov_mem_imm((imm32), reg_base, (reg_index) * 4) \
|
2009-05-21 17:48:31 +02:00
|
|
|
|
|
|
|
#define generate_shift_left(ireg, imm) \
|
|
|
|
x86_emit_shl_reg_imm(reg_##ireg, imm) \
|
|
|
|
|
2021-08-31 21:18:22 +02:00
|
|
|
#define generate_shift_left_var(ireg) \
|
|
|
|
x86_emit_shl_reg_reg(reg_##ireg) \
|
|
|
|
|
2009-05-21 17:48:31 +02:00
|
|
|
#define generate_shift_right(ireg, imm) \
|
|
|
|
x86_emit_shr_reg_imm(reg_##ireg, imm) \
|
|
|
|
|
2021-08-31 21:18:22 +02:00
|
|
|
#define generate_shift_right_var(ireg) \
|
|
|
|
x86_emit_shr_reg_reg(reg_##ireg) \
|
|
|
|
|
2009-05-21 17:48:31 +02:00
|
|
|
#define generate_shift_right_arithmetic(ireg, imm) \
|
|
|
|
x86_emit_sar_reg_imm(reg_##ireg, imm) \
|
|
|
|
|
2021-08-31 21:18:22 +02:00
|
|
|
#define generate_shift_right_arithmetic_var(ireg) \
|
|
|
|
x86_emit_sar_reg_reg(reg_##ireg) \
|
|
|
|
|
2009-05-21 17:48:31 +02:00
|
|
|
#define generate_rotate_right(ireg, imm) \
|
|
|
|
x86_emit_ror_reg_imm(reg_##ireg, imm) \
|
|
|
|
|
2021-08-31 21:18:22 +02:00
|
|
|
#define generate_rotate_right_var(ireg) \
|
|
|
|
x86_emit_rot_reg_reg(ror, reg_##ireg) \
|
|
|
|
|
|
|
|
#define generate_rcr(ireg) \
|
|
|
|
x86_emit_rot_reg_reg(rcr, reg_##ireg) \
|
|
|
|
|
|
|
|
#define generate_rcr1(ireg) \
|
|
|
|
x86_emit_rot_reg1(rcr, reg_##ireg) \
|
|
|
|
|
2021-11-07 19:19:19 +01:00
|
|
|
#define generate_and_mem(imm, ireg_base, offset) \
|
|
|
|
x86_emit_and_mem_imm(imm, reg_##ireg_base, (offset)) \
|
|
|
|
|
2021-08-31 00:42:47 +02:00
|
|
|
#define generate_and(ireg_dest, ireg_src) \
|
|
|
|
x86_emit_and_reg_reg(reg_##ireg_dest, reg_##ireg_src) \
|
|
|
|
|
2009-05-21 17:48:31 +02:00
|
|
|
#define generate_add(ireg_dest, ireg_src) \
|
|
|
|
x86_emit_add_reg_reg(reg_##ireg_dest, reg_##ireg_src) \
|
|
|
|
|
2021-08-31 00:42:47 +02:00
|
|
|
#define generate_adc(ireg_dest, ireg_src) \
|
|
|
|
x86_emit_adc_reg_reg(reg_##ireg_dest, reg_##ireg_src) \
|
|
|
|
|
|
|
|
#define generate_add_memreg(ireg_dest, arm_reg_src) \
|
|
|
|
x86_emit_add_reg_mem(reg_##ireg_dest, reg_base, arm_reg_src * 4) \
|
|
|
|
|
2009-05-21 17:48:31 +02:00
|
|
|
#define generate_sub(ireg_dest, ireg_src) \
|
|
|
|
x86_emit_sub_reg_reg(reg_##ireg_dest, reg_##ireg_src) \
|
|
|
|
|
2021-08-31 00:42:47 +02:00
|
|
|
#define generate_sbb(ireg_dest, ireg_src) \
|
|
|
|
x86_emit_sbb_reg_reg(reg_##ireg_dest, reg_##ireg_src) \
|
|
|
|
|
2009-05-21 17:48:31 +02:00
|
|
|
#define generate_or(ireg_dest, ireg_src) \
|
|
|
|
x86_emit_or_reg_reg(reg_##ireg_dest, reg_##ireg_src) \
|
|
|
|
|
2021-09-17 19:18:50 +02:00
|
|
|
#define generate_or_mem(ireg_dest, arm_reg_src) \
|
|
|
|
x86_emit_or_reg_mem(reg_##ireg_dest, reg_base, arm_reg_src * 4) \
|
|
|
|
|
2009-05-21 17:48:31 +02:00
|
|
|
#define generate_xor(ireg_dest, ireg_src) \
|
|
|
|
x86_emit_xor_reg_reg(reg_##ireg_dest, reg_##ireg_src) \
|
|
|
|
|
2021-09-17 19:18:50 +02:00
|
|
|
#define generate_xor_mem(ireg_dest, arm_reg_src) \
|
|
|
|
x86_emit_xor_reg_mem(reg_##ireg_dest, reg_base, arm_reg_src * 4) \
|
|
|
|
|
2009-05-21 17:48:31 +02:00
|
|
|
#define generate_add_imm(ireg, imm) \
|
|
|
|
x86_emit_add_reg_imm(reg_##ireg, imm) \
|
|
|
|
|
|
|
|
#define generate_sub_imm(ireg, imm) \
|
|
|
|
x86_emit_sub_reg_imm(reg_##ireg, imm) \
|
|
|
|
|
|
|
|
#define generate_xor_imm(ireg, imm) \
|
|
|
|
x86_emit_xor_reg_imm(reg_##ireg, imm) \
|
|
|
|
|
|
|
|
#define generate_add_reg_reg_imm(ireg_dest, ireg_src, imm) \
|
|
|
|
x86_emit_lea_reg_mem(reg_##ireg_dest, reg_##ireg_src, imm) \
|
|
|
|
|
|
|
|
#define generate_and_imm(ireg, imm) \
|
|
|
|
x86_emit_and_reg_imm(reg_##ireg, imm) \
|
|
|
|
|
|
|
|
#define generate_mov(ireg_dest, ireg_src) \
|
|
|
|
x86_emit_mov_reg_reg(reg_##ireg_dest, reg_##ireg_src) \
|
|
|
|
|
2021-08-31 00:42:47 +02:00
|
|
|
#define generate_not(ireg) \
|
|
|
|
x86_emit_not_reg(reg_##ireg) \
|
|
|
|
|
2009-05-21 17:48:31 +02:00
|
|
|
#define generate_multiply(ireg) \
|
|
|
|
x86_emit_imul_eax_reg(reg_##ireg) \
|
|
|
|
|
|
|
|
#define generate_multiply_s64(ireg) \
|
|
|
|
x86_emit_imul_eax_reg(reg_##ireg) \
|
|
|
|
|
|
|
|
#define generate_multiply_u64(ireg) \
|
|
|
|
x86_emit_mul_eax_reg(reg_##ireg) \
|
|
|
|
|
|
|
|
#define generate_multiply_s64_add(ireg_src, ireg_lo, ireg_hi) \
|
|
|
|
x86_emit_imul_eax_reg(reg_##ireg_src); \
|
|
|
|
x86_emit_add_reg_reg(reg_a0, reg_##ireg_lo); \
|
|
|
|
x86_emit_adc_reg_reg(reg_a1, reg_##ireg_hi) \
|
|
|
|
|
|
|
|
#define generate_multiply_u64_add(ireg_src, ireg_lo, ireg_hi) \
|
|
|
|
x86_emit_mul_eax_reg(reg_##ireg_src); \
|
|
|
|
x86_emit_add_reg_reg(reg_a0, reg_##ireg_lo); \
|
|
|
|
x86_emit_adc_reg_reg(reg_a1, reg_##ireg_hi) \
|
|
|
|
|
|
|
|
|
|
|
|
#define generate_function_call(function_location) \
|
|
|
|
x86_emit_call_offset(x86_relative_offset(translation_ptr, \
|
|
|
|
function_location, 4)); \
|
|
|
|
|
|
|
|
#define generate_exit_block() \
|
|
|
|
x86_emit_ret(); \
|
|
|
|
|
|
|
|
#define generate_cycle_update() \
|
|
|
|
x86_emit_sub_reg_imm(reg_cycles, cycle_count); \
|
|
|
|
cycle_count = 0 \
|
|
|
|
|
|
|
|
#define generate_branch_patch_conditional(dest, offset) \
|
|
|
|
*((u32 *)(dest)) = x86_relative_offset(dest, offset, 4) \
|
|
|
|
|
|
|
|
#define generate_branch_patch_unconditional(dest, offset) \
|
|
|
|
*((u32 *)(dest)) = x86_relative_offset(dest, offset, 4) \
|
|
|
|
|
|
|
|
#define generate_branch_no_cycle_update(writeback_location, new_pc) \
|
|
|
|
if(pc == idle_loop_target_pc) \
|
|
|
|
{ \
|
2021-12-21 19:59:33 +01:00
|
|
|
generate_load_imm(cycles, 0); \
|
2009-05-21 17:48:31 +02:00
|
|
|
x86_emit_mov_reg_imm(eax, new_pc); \
|
|
|
|
generate_function_call(x86_update_gba); \
|
|
|
|
x86_emit_jmp_filler(writeback_location); \
|
|
|
|
} \
|
|
|
|
else \
|
|
|
|
{ \
|
|
|
|
x86_emit_test_reg_reg(reg_cycles, reg_cycles); \
|
|
|
|
x86_emit_j_offset(x86_condition_code_ns, 10); \
|
|
|
|
x86_emit_mov_reg_imm(eax, new_pc); \
|
|
|
|
generate_function_call(x86_update_gba); \
|
|
|
|
x86_emit_jmp_filler(writeback_location); \
|
|
|
|
} \
|
|
|
|
|
|
|
|
#define generate_branch_cycle_update(writeback_location, new_pc) \
|
|
|
|
generate_cycle_update(); \
|
|
|
|
generate_branch_no_cycle_update(writeback_location, new_pc) \
|
|
|
|
|
|
|
|
// a0 holds the destination
|
|
|
|
|
|
|
|
#define generate_indirect_branch_cycle_update(type) \
|
|
|
|
generate_cycle_update(); \
|
2021-11-06 12:17:50 +01:00
|
|
|
x86_emit_call_offset(x86_relative_offset(translation_ptr, \
|
2009-05-21 17:48:31 +02:00
|
|
|
x86_indirect_branch_##type, 4)) \
|
|
|
|
|
|
|
|
#define generate_indirect_branch_no_cycle_update(type) \
|
2021-11-06 12:17:50 +01:00
|
|
|
x86_emit_call_offset(x86_relative_offset(translation_ptr, \
|
2009-05-21 17:48:31 +02:00
|
|
|
x86_indirect_branch_##type, 4)) \
|
|
|
|
|
2021-07-26 21:41:07 +02:00
|
|
|
#define block_prologue_size 0
|
|
|
|
#define generate_block_prologue()
|
|
|
|
#define generate_block_extra_vars_arm()
|
|
|
|
#define generate_block_extra_vars_thumb()
|
2009-05-21 17:48:31 +02:00
|
|
|
|
2021-07-26 21:41:07 +02:00
|
|
|
#define generate_indirect_branch_arm() \
|
2009-05-21 17:48:31 +02:00
|
|
|
{ \
|
|
|
|
if(condition == 0x0E) \
|
|
|
|
{ \
|
|
|
|
generate_indirect_branch_cycle_update(arm); \
|
|
|
|
} \
|
|
|
|
else \
|
|
|
|
{ \
|
|
|
|
generate_indirect_branch_no_cycle_update(arm); \
|
|
|
|
} \
|
|
|
|
} \
|
2021-07-26 21:41:07 +02:00
|
|
|
|
|
|
|
#define generate_indirect_branch_dual() \
|
2009-05-21 17:48:31 +02:00
|
|
|
{ \
|
|
|
|
if(condition == 0x0E) \
|
|
|
|
{ \
|
|
|
|
generate_indirect_branch_cycle_update(dual); \
|
|
|
|
} \
|
|
|
|
else \
|
|
|
|
{ \
|
|
|
|
generate_indirect_branch_no_cycle_update(dual); \
|
|
|
|
} \
|
|
|
|
} \
|
|
|
|
|
|
|
|
|
|
|
|
#define get_shift_imm() \
|
|
|
|
u32 shift = (opcode >> 7) & 0x1F \
|
|
|
|
|
|
|
|
#define generate_shift_reg(ireg, name, flags_op) \
|
|
|
|
generate_load_reg_pc(ireg, rm, 12); \
|
|
|
|
generate_load_reg(a1, ((opcode >> 8) & 0x0F)); \
|
2021-09-01 19:34:43 +02:00
|
|
|
generate_and_imm(a1, 0xFF); \
|
2021-08-31 21:18:22 +02:00
|
|
|
generate_##name##_##flags_op##_reg(ireg); \
|
2009-05-21 17:48:31 +02:00
|
|
|
|
2021-04-03 00:37:42 +02:00
|
|
|
#ifdef TRACE_INSTRUCTIONS
|
2021-08-15 22:48:43 +02:00
|
|
|
void function_cc trace_instruction(u32 pc, u32 mode)
|
2021-04-03 00:37:42 +02:00
|
|
|
{
|
2021-08-15 22:48:43 +02:00
|
|
|
if (mode)
|
|
|
|
printf("Executed arm %x\n", pc);
|
|
|
|
else
|
|
|
|
printf("Executed thumb %x\n", pc);
|
2021-12-11 11:27:59 +01:00
|
|
|
#ifdef TRACE_REGISTERS
|
|
|
|
print_regs();
|
|
|
|
#endif
|
2021-04-03 00:37:42 +02:00
|
|
|
}
|
|
|
|
|
2021-08-15 22:48:43 +02:00
|
|
|
#define emit_trace_instruction(pc, mode) \
|
2021-11-06 12:17:50 +01:00
|
|
|
x86_emit_mov_reg_imm(reg_arg0, pc); \
|
|
|
|
x86_emit_mov_reg_imm(reg_arg1, mode); \
|
2021-08-15 22:48:43 +02:00
|
|
|
generate_function_call(trace_instruction);
|
2021-04-03 00:37:42 +02:00
|
|
|
#define emit_trace_arm_instruction(pc) \
|
2021-08-15 22:48:43 +02:00
|
|
|
emit_trace_instruction(pc, 1)
|
|
|
|
#define emit_trace_thumb_instruction(pc) \
|
|
|
|
emit_trace_instruction(pc, 0)
|
2021-04-03 00:37:42 +02:00
|
|
|
#else
|
|
|
|
#define emit_trace_thumb_instruction(pc)
|
|
|
|
#define emit_trace_arm_instruction(pc)
|
|
|
|
#endif
|
|
|
|
|
2021-12-13 18:31:01 +01:00
|
|
|
#define check_generate_n_flag (flag_status & 0x08)
|
|
|
|
#define check_generate_z_flag (flag_status & 0x04)
|
|
|
|
#define check_generate_c_flag (flag_status & 0x02)
|
|
|
|
#define check_generate_v_flag (flag_status & 0x01)
|
2009-05-21 17:48:31 +02:00
|
|
|
|
2021-08-31 21:18:22 +02:00
|
|
|
#define generate_asr_no_flags_reg(ireg) \
|
|
|
|
{ \
|
|
|
|
u8 *jmpinst; \
|
|
|
|
generate_mov(a2, a1); \
|
|
|
|
generate_shift_right_arithmetic_var(a0); \
|
|
|
|
generate_cmp_imm(a2, 32); \
|
|
|
|
x86_emit_j_filler(x86_condition_code_l, jmpinst); \
|
|
|
|
generate_shift_right_arithmetic(a0, 31); \
|
|
|
|
generate_branch_patch_conditional(jmpinst, translation_ptr); \
|
|
|
|
generate_mov(ireg, a0); \
|
2009-05-21 17:48:31 +02:00
|
|
|
}
|
|
|
|
|
2021-08-31 21:18:22 +02:00
|
|
|
#define generate_asr_flags_reg(ireg) \
|
|
|
|
{ \
|
|
|
|
u8 *jmpinst1, *jmpinst2; \
|
|
|
|
generate_mov(a2, a1); \
|
|
|
|
generate_or(a2, a2); \
|
|
|
|
x86_emit_j_filler(x86_condition_code_z, jmpinst1); \
|
|
|
|
generate_shift_right_arithmetic_var(a0); \
|
|
|
|
generate_update_flag(c, REG_C_FLAG) \
|
|
|
|
generate_cmp_imm(a2, 32); \
|
|
|
|
x86_emit_j_filler(x86_condition_code_l, jmpinst2); \
|
|
|
|
generate_shift_right_arithmetic(a0, 16); \
|
|
|
|
generate_shift_right_arithmetic(a0, 16); \
|
|
|
|
generate_update_flag(c, REG_C_FLAG) \
|
|
|
|
generate_branch_patch_conditional(jmpinst1, translation_ptr); \
|
|
|
|
generate_branch_patch_conditional(jmpinst2, translation_ptr); \
|
|
|
|
generate_mov(ireg, a0); \
|
2009-05-21 17:48:31 +02:00
|
|
|
}
|
|
|
|
|
2021-08-31 21:18:22 +02:00
|
|
|
#define generate_lsl_no_flags_reg(ireg) \
|
|
|
|
{ \
|
|
|
|
u8 *jmpinst; \
|
|
|
|
generate_mov(a2, a1); \
|
|
|
|
generate_shift_left_var(a0); \
|
|
|
|
generate_cmp_imm(a2, 32); \
|
|
|
|
x86_emit_j_filler(x86_condition_code_l, jmpinst); \
|
|
|
|
generate_load_imm(a0, 0); \
|
|
|
|
generate_branch_patch_conditional(jmpinst, translation_ptr); \
|
|
|
|
generate_mov(ireg, a0); \
|
2009-05-21 17:48:31 +02:00
|
|
|
}
|
|
|
|
|
2021-08-31 21:18:22 +02:00
|
|
|
#define generate_lsl_flags_reg(ireg) \
|
|
|
|
{ \
|
|
|
|
u8 *jmpinst1, *jmpinst2; \
|
|
|
|
generate_mov(a2, a1); \
|
|
|
|
generate_or(a2, a2); \
|
|
|
|
x86_emit_j_filler(x86_condition_code_z, jmpinst1); \
|
|
|
|
generate_sub_imm(a2, 1); \
|
|
|
|
generate_shift_left_var(a0); \
|
|
|
|
generate_or(a0, a0); \
|
|
|
|
generate_update_flag(s, REG_C_FLAG) \
|
|
|
|
generate_shift_left(a0, 1); \
|
|
|
|
generate_cmp_imm(a2, 32); \
|
|
|
|
x86_emit_j_filler(x86_condition_code_l, jmpinst2); \
|
|
|
|
generate_load_imm(a0, 0); \
|
|
|
|
generate_store_reg(a0, REG_C_FLAG) \
|
|
|
|
generate_branch_patch_conditional(jmpinst1, translation_ptr); \
|
|
|
|
generate_branch_patch_conditional(jmpinst2, translation_ptr); \
|
|
|
|
generate_mov(ireg, a0); \
|
2009-05-21 17:48:31 +02:00
|
|
|
}
|
|
|
|
|
2021-08-31 21:18:22 +02:00
|
|
|
#define generate_lsr_no_flags_reg(ireg) \
|
|
|
|
{ \
|
|
|
|
u8 *jmpinst; \
|
|
|
|
generate_mov(a2, a1); \
|
|
|
|
generate_shift_right_var(a0); \
|
|
|
|
generate_cmp_imm(a2, 32); \
|
|
|
|
x86_emit_j_filler(x86_condition_code_l, jmpinst); \
|
|
|
|
generate_xor(a0, a0); \
|
|
|
|
generate_branch_patch_conditional(jmpinst, translation_ptr); \
|
|
|
|
generate_mov(ireg, a0); \
|
2009-05-21 17:48:31 +02:00
|
|
|
}
|
|
|
|
|
2021-08-31 21:18:22 +02:00
|
|
|
#define generate_lsr_flags_reg(ireg) \
|
|
|
|
{ \
|
|
|
|
u8 *jmpinst1, *jmpinst2; \
|
|
|
|
generate_mov(a2, a1); \
|
|
|
|
generate_or(a2, a2); \
|
|
|
|
x86_emit_j_filler(x86_condition_code_z, jmpinst1); \
|
|
|
|
generate_sub_imm(a2, 1); \
|
|
|
|
generate_shift_right_var(a0); \
|
|
|
|
generate_test_imm(a0, 1); \
|
|
|
|
generate_update_flag(nz, REG_C_FLAG) \
|
|
|
|
generate_shift_right(a0, 1); \
|
|
|
|
generate_cmp_imm(a2, 32); \
|
|
|
|
x86_emit_j_filler(x86_condition_code_l, jmpinst2); \
|
|
|
|
generate_xor(a0, a0); \
|
|
|
|
generate_store_reg(a0, REG_C_FLAG) \
|
|
|
|
generate_branch_patch_conditional(jmpinst1, translation_ptr); \
|
|
|
|
generate_branch_patch_conditional(jmpinst2, translation_ptr); \
|
|
|
|
generate_mov(ireg, a0); \
|
2009-05-21 17:48:31 +02:00
|
|
|
}
|
|
|
|
|
2021-08-31 21:18:22 +02:00
|
|
|
#define generate_ror_no_flags_reg(ireg) \
|
|
|
|
generate_mov(a2, a1); \
|
|
|
|
generate_rotate_right_var(a0); \
|
|
|
|
generate_mov(ireg, a0);
|
2009-05-21 17:48:31 +02:00
|
|
|
|
2021-08-31 21:18:22 +02:00
|
|
|
#define generate_ror_flags_reg(ireg) \
|
|
|
|
{ \
|
|
|
|
u8 *jmpinst; \
|
|
|
|
generate_mov(a2, a1); \
|
|
|
|
generate_or(a2, a2); \
|
|
|
|
x86_emit_j_filler(x86_condition_code_z, jmpinst); \
|
|
|
|
generate_sub_imm(a2, 1); \
|
|
|
|
generate_rotate_right_var(a0); \
|
|
|
|
generate_test_imm(a0, 1); \
|
|
|
|
generate_update_flag(nz, REG_C_FLAG) \
|
|
|
|
generate_rotate_right(a0, 1); \
|
|
|
|
generate_branch_patch_conditional(jmpinst, translation_ptr); \
|
|
|
|
generate_mov(ireg, a0); \
|
2009-05-21 17:48:31 +02:00
|
|
|
}
|
|
|
|
|
2021-08-31 21:18:22 +02:00
|
|
|
// Shift right sets the CF of the shifted-out bit, use it with setc
|
|
|
|
#define generate_rrx_flags(ireg) \
|
|
|
|
generate_load_imm(a2, 0xffffffff); \
|
|
|
|
generate_add_memreg(a2, REG_C_FLAG); \
|
|
|
|
generate_rcr1(a0); \
|
|
|
|
generate_update_flag(c, REG_C_FLAG) \
|
|
|
|
generate_mov(ireg, a0);
|
2009-05-21 17:48:31 +02:00
|
|
|
|
2021-08-31 21:18:22 +02:00
|
|
|
#define generate_rrx(ireg) \
|
|
|
|
generate_load_reg(a2, REG_C_FLAG); \
|
|
|
|
generate_shift_right(ireg, 1); \
|
|
|
|
generate_shift_left(a2, 31); \
|
|
|
|
generate_or(ireg, a2); \
|
2009-05-21 17:48:31 +02:00
|
|
|
|
|
|
|
#define generate_shift_imm_lsl_no_flags(ireg) \
|
|
|
|
generate_load_reg_pc(ireg, rm, 8); \
|
|
|
|
if(shift != 0) \
|
|
|
|
{ \
|
|
|
|
generate_shift_left(ireg, shift); \
|
|
|
|
} \
|
|
|
|
|
|
|
|
#define generate_shift_imm_lsr_no_flags(ireg) \
|
|
|
|
if(shift != 0) \
|
|
|
|
{ \
|
|
|
|
generate_load_reg_pc(ireg, rm, 8); \
|
|
|
|
generate_shift_right(ireg, shift); \
|
|
|
|
} \
|
|
|
|
else \
|
|
|
|
{ \
|
|
|
|
generate_load_imm(ireg, 0); \
|
|
|
|
} \
|
|
|
|
|
|
|
|
#define generate_shift_imm_asr_no_flags(ireg) \
|
|
|
|
generate_load_reg_pc(ireg, rm, 8); \
|
|
|
|
if(shift != 0) \
|
|
|
|
{ \
|
|
|
|
generate_shift_right_arithmetic(ireg, shift); \
|
|
|
|
} \
|
|
|
|
else \
|
|
|
|
{ \
|
|
|
|
generate_shift_right_arithmetic(ireg, 31); \
|
|
|
|
} \
|
|
|
|
|
|
|
|
#define generate_shift_imm_ror_no_flags(ireg) \
|
|
|
|
if(shift != 0) \
|
|
|
|
{ \
|
|
|
|
generate_load_reg_pc(ireg, rm, 8); \
|
|
|
|
generate_rotate_right(ireg, shift); \
|
|
|
|
} \
|
|
|
|
else \
|
|
|
|
{ \
|
2021-08-31 21:18:22 +02:00
|
|
|
generate_load_reg_pc(ireg, rm, 8); \
|
|
|
|
generate_rrx(ireg); \
|
2009-05-21 17:48:31 +02:00
|
|
|
} \
|
|
|
|
|
|
|
|
#define generate_shift_imm_lsl_flags(ireg) \
|
|
|
|
generate_load_reg_pc(ireg, rm, 8); \
|
|
|
|
if(shift != 0) \
|
|
|
|
{ \
|
|
|
|
generate_shift_left(ireg, shift); \
|
2021-09-01 19:34:43 +02:00
|
|
|
generate_update_flag(c, REG_C_FLAG); \
|
2009-05-21 17:48:31 +02:00
|
|
|
} \
|
|
|
|
|
|
|
|
#define generate_shift_imm_lsr_flags(ireg) \
|
2021-09-01 19:34:43 +02:00
|
|
|
generate_load_reg_pc(ireg, rm, 8); \
|
2009-05-21 17:48:31 +02:00
|
|
|
if(shift != 0) \
|
|
|
|
{ \
|
|
|
|
generate_shift_right(ireg, shift); \
|
2021-09-01 19:34:43 +02:00
|
|
|
generate_update_flag(c, REG_C_FLAG); \
|
2009-05-21 17:48:31 +02:00
|
|
|
} \
|
|
|
|
else \
|
|
|
|
{ \
|
2021-09-01 19:34:43 +02:00
|
|
|
generate_shift_right(ireg, 31); \
|
|
|
|
generate_store_reg(ireg, REG_C_FLAG); \
|
2009-05-21 17:48:31 +02:00
|
|
|
generate_load_imm(ireg, 0); \
|
|
|
|
} \
|
|
|
|
|
|
|
|
#define generate_shift_imm_asr_flags(ireg) \
|
2021-09-01 19:34:43 +02:00
|
|
|
generate_load_reg_pc(ireg, rm, 8); \
|
2009-05-21 17:48:31 +02:00
|
|
|
if(shift != 0) \
|
|
|
|
{ \
|
|
|
|
generate_shift_right_arithmetic(ireg, shift); \
|
2021-09-01 19:34:43 +02:00
|
|
|
generate_update_flag(c, REG_C_FLAG); \
|
2009-05-21 17:48:31 +02:00
|
|
|
} \
|
|
|
|
else \
|
|
|
|
{ \
|
|
|
|
generate_shift_right_arithmetic(ireg, 31); \
|
2021-09-01 19:34:43 +02:00
|
|
|
generate_update_flag(nz, REG_C_FLAG); \
|
2009-05-21 17:48:31 +02:00
|
|
|
} \
|
|
|
|
|
|
|
|
#define generate_shift_imm_ror_flags(ireg) \
|
|
|
|
generate_load_reg_pc(ireg, rm, 8); \
|
|
|
|
if(shift != 0) \
|
|
|
|
{ \
|
|
|
|
generate_rotate_right(ireg, shift); \
|
2021-09-01 19:34:43 +02:00
|
|
|
generate_update_flag(c, REG_C_FLAG) \
|
2009-05-21 17:48:31 +02:00
|
|
|
} \
|
|
|
|
else \
|
|
|
|
{ \
|
2021-08-31 21:18:22 +02:00
|
|
|
generate_rrx_flags(ireg); \
|
2009-05-21 17:48:31 +02:00
|
|
|
} \
|
|
|
|
|
|
|
|
#define generate_shift_imm(ireg, name, flags_op) \
|
|
|
|
get_shift_imm(); \
|
|
|
|
generate_shift_imm_##name##_##flags_op(ireg) \
|
|
|
|
|
|
|
|
#define generate_load_rm_sh(flags_op) \
|
|
|
|
switch((opcode >> 4) & 0x07) \
|
|
|
|
{ \
|
|
|
|
/* LSL imm */ \
|
|
|
|
case 0x0: \
|
|
|
|
{ \
|
|
|
|
generate_shift_imm(a0, lsl, flags_op); \
|
|
|
|
break; \
|
|
|
|
} \
|
|
|
|
\
|
|
|
|
/* LSL reg */ \
|
|
|
|
case 0x1: \
|
|
|
|
{ \
|
|
|
|
generate_shift_reg(a0, lsl, flags_op); \
|
|
|
|
break; \
|
|
|
|
} \
|
|
|
|
\
|
|
|
|
/* LSR imm */ \
|
|
|
|
case 0x2: \
|
|
|
|
{ \
|
|
|
|
generate_shift_imm(a0, lsr, flags_op); \
|
|
|
|
break; \
|
|
|
|
} \
|
|
|
|
\
|
|
|
|
/* LSR reg */ \
|
|
|
|
case 0x3: \
|
|
|
|
{ \
|
|
|
|
generate_shift_reg(a0, lsr, flags_op); \
|
|
|
|
break; \
|
|
|
|
} \
|
|
|
|
\
|
|
|
|
/* ASR imm */ \
|
|
|
|
case 0x4: \
|
|
|
|
{ \
|
|
|
|
generate_shift_imm(a0, asr, flags_op); \
|
|
|
|
break; \
|
|
|
|
} \
|
|
|
|
\
|
|
|
|
/* ASR reg */ \
|
|
|
|
case 0x5: \
|
|
|
|
{ \
|
|
|
|
generate_shift_reg(a0, asr, flags_op); \
|
|
|
|
break; \
|
|
|
|
} \
|
|
|
|
\
|
|
|
|
/* ROR imm */ \
|
|
|
|
case 0x6: \
|
|
|
|
{ \
|
|
|
|
generate_shift_imm(a0, ror, flags_op); \
|
|
|
|
break; \
|
|
|
|
} \
|
|
|
|
\
|
|
|
|
/* ROR reg */ \
|
|
|
|
case 0x7: \
|
|
|
|
{ \
|
|
|
|
generate_shift_reg(a0, ror, flags_op); \
|
|
|
|
break; \
|
|
|
|
} \
|
|
|
|
} \
|
|
|
|
|
|
|
|
#define generate_load_offset_sh() \
|
|
|
|
switch((opcode >> 5) & 0x03) \
|
|
|
|
{ \
|
|
|
|
/* LSL imm */ \
|
|
|
|
case 0x0: \
|
|
|
|
{ \
|
|
|
|
generate_shift_imm(a1, lsl, no_flags); \
|
|
|
|
break; \
|
|
|
|
} \
|
|
|
|
\
|
|
|
|
/* LSR imm */ \
|
|
|
|
case 0x1: \
|
|
|
|
{ \
|
|
|
|
generate_shift_imm(a1, lsr, no_flags); \
|
|
|
|
break; \
|
|
|
|
} \
|
|
|
|
\
|
|
|
|
/* ASR imm */ \
|
|
|
|
case 0x2: \
|
|
|
|
{ \
|
|
|
|
generate_shift_imm(a1, asr, no_flags); \
|
|
|
|
break; \
|
|
|
|
} \
|
|
|
|
\
|
|
|
|
/* ROR imm */ \
|
|
|
|
case 0x3: \
|
|
|
|
{ \
|
|
|
|
generate_shift_imm(a1, ror, no_flags); \
|
|
|
|
break; \
|
|
|
|
} \
|
|
|
|
} \
|
|
|
|
|
|
|
|
#define extract_flags() \
|
|
|
|
reg[REG_N_FLAG] = reg[REG_CPSR] >> 31; \
|
|
|
|
reg[REG_Z_FLAG] = (reg[REG_CPSR] >> 30) & 0x01; \
|
|
|
|
reg[REG_C_FLAG] = (reg[REG_CPSR] >> 29) & 0x01; \
|
|
|
|
reg[REG_V_FLAG] = (reg[REG_CPSR] >> 28) & 0x01; \
|
|
|
|
|
2021-09-17 20:00:37 +02:00
|
|
|
#define collapse_flags(ireg, tmpreg) \
|
|
|
|
generate_load_reg(ireg, REG_V_FLAG); \
|
|
|
|
generate_load_reg(tmpreg, REG_C_FLAG); \
|
|
|
|
generate_shift_left(tmpreg, 1); \
|
|
|
|
generate_or(ireg, tmpreg); \
|
|
|
|
generate_load_reg(tmpreg, REG_Z_FLAG); \
|
|
|
|
generate_shift_left(tmpreg, 2); \
|
|
|
|
generate_or(ireg, tmpreg); \
|
|
|
|
generate_load_reg(tmpreg, REG_N_FLAG); \
|
|
|
|
generate_shift_left(tmpreg, 3); \
|
|
|
|
generate_or(ireg, tmpreg); \
|
|
|
|
generate_load_reg(tmpreg, REG_CPSR); \
|
|
|
|
generate_shift_left(ireg, 28); \
|
|
|
|
generate_and_imm(tmpreg, 0xFF); \
|
|
|
|
generate_or(ireg, tmpreg); \
|
|
|
|
generate_store_reg(ireg, REG_CPSR); \
|
2009-05-21 17:48:31 +02:00
|
|
|
|
|
|
|
// It should be okay to still generate result flags, spsr will overwrite them.
|
|
|
|
// This is pretty infrequent (returning from interrupt handlers, et al) so
|
|
|
|
// probably not worth optimizing for.
|
|
|
|
|
|
|
|
#define generate_load_reg_pc(ireg, reg_index, pc_offset) \
|
|
|
|
if(reg_index == 15) \
|
|
|
|
{ \
|
|
|
|
generate_load_pc(ireg, pc + pc_offset); \
|
|
|
|
} \
|
|
|
|
else \
|
|
|
|
{ \
|
|
|
|
generate_load_reg(ireg, reg_index); \
|
|
|
|
} \
|
|
|
|
|
|
|
|
#define generate_store_reg_pc_no_flags(ireg, reg_index) \
|
|
|
|
generate_store_reg(ireg, reg_index); \
|
|
|
|
if(reg_index == 15) \
|
|
|
|
{ \
|
|
|
|
generate_mov(a0, ireg); \
|
|
|
|
generate_indirect_branch_arm(); \
|
|
|
|
} \
|
|
|
|
|
2021-03-06 21:15:22 +01:00
|
|
|
u32 function_cc execute_spsr_restore(u32 address)
|
2009-05-21 17:48:31 +02:00
|
|
|
{
|
2023-01-11 20:44:56 +01:00
|
|
|
if(reg[CPU_MODE] != MODE_USER && reg[CPU_MODE] != MODE_SYSTEM)
|
2009-05-21 17:48:31 +02:00
|
|
|
{
|
2023-01-11 20:44:56 +01:00
|
|
|
reg[REG_CPSR] = REG_SPSR(reg[CPU_MODE]);
|
2009-05-21 17:48:31 +02:00
|
|
|
extract_flags();
|
2023-01-11 20:44:56 +01:00
|
|
|
set_cpu_mode(cpu_modes[reg[REG_CPSR] & 0xF]);
|
2021-11-02 21:14:38 +01:00
|
|
|
|
|
|
|
if((io_registers[REG_IE] & io_registers[REG_IF]) &&
|
|
|
|
io_registers[REG_IME] && ((reg[REG_CPSR] & 0x80) == 0))
|
|
|
|
{
|
2023-01-11 20:44:56 +01:00
|
|
|
REG_MODE(MODE_IRQ)[6] = reg[REG_PC] + 4;
|
|
|
|
REG_SPSR(MODE_IRQ) = reg[REG_CPSR];
|
2021-11-02 21:14:38 +01:00
|
|
|
reg[REG_CPSR] = 0xD2;
|
|
|
|
address = 0x00000018;
|
|
|
|
set_cpu_mode(MODE_IRQ);
|
|
|
|
}
|
2009-05-21 17:48:31 +02:00
|
|
|
|
|
|
|
if(reg[REG_CPSR] & 0x20)
|
|
|
|
address |= 0x01;
|
|
|
|
}
|
|
|
|
|
|
|
|
return address;
|
|
|
|
}
|
|
|
|
|
|
|
|
#define generate_store_reg_pc_flags(ireg, reg_index) \
|
|
|
|
generate_store_reg(ireg, reg_index); \
|
|
|
|
if(reg_index == 15) \
|
|
|
|
{ \
|
2021-11-06 12:17:50 +01:00
|
|
|
generate_mov(arg0, ireg); \
|
2009-05-21 17:48:31 +02:00
|
|
|
generate_function_call(execute_spsr_restore); \
|
|
|
|
generate_indirect_branch_dual(); \
|
|
|
|
} \
|
|
|
|
|
2021-11-07 19:19:19 +01:00
|
|
|
// These generate a branch on the opposite condition on purpose.
|
|
|
|
// For ARM mode we aim to skip instructions (therefore opposite)
|
|
|
|
// In Thumb mode we skip the conditional branch in a similar way
|
2021-09-17 19:18:50 +02:00
|
|
|
#define generate_condition_eq(ireg) \
|
2021-11-07 19:19:19 +01:00
|
|
|
generate_and_mem(1, base, REG_Z_FLAG * 4); \
|
2021-09-17 19:18:50 +02:00
|
|
|
x86_emit_j_filler(x86_condition_code_z, backpatch_address) \
|
|
|
|
|
|
|
|
#define generate_condition_ne(ireg) \
|
2021-11-07 19:19:19 +01:00
|
|
|
generate_and_mem(1, base, REG_Z_FLAG * 4); \
|
2021-09-17 19:18:50 +02:00
|
|
|
x86_emit_j_filler(x86_condition_code_nz, backpatch_address) \
|
|
|
|
|
|
|
|
#define generate_condition_cs(ireg) \
|
2021-11-07 19:19:19 +01:00
|
|
|
generate_and_mem(1, base, REG_C_FLAG * 4); \
|
2021-09-17 19:18:50 +02:00
|
|
|
x86_emit_j_filler(x86_condition_code_z, backpatch_address) \
|
|
|
|
|
|
|
|
#define generate_condition_cc(ireg) \
|
2021-11-07 19:19:19 +01:00
|
|
|
generate_and_mem(1, base, REG_C_FLAG * 4); \
|
2021-09-17 19:18:50 +02:00
|
|
|
x86_emit_j_filler(x86_condition_code_nz, backpatch_address) \
|
|
|
|
|
|
|
|
#define generate_condition_mi(ireg) \
|
2021-11-07 19:19:19 +01:00
|
|
|
generate_and_mem(1, base, REG_N_FLAG * 4); \
|
2021-09-17 19:18:50 +02:00
|
|
|
x86_emit_j_filler(x86_condition_code_z, backpatch_address) \
|
|
|
|
|
|
|
|
#define generate_condition_pl(ireg) \
|
2021-11-07 19:19:19 +01:00
|
|
|
generate_and_mem(1, base, REG_N_FLAG * 4); \
|
2021-09-17 19:18:50 +02:00
|
|
|
x86_emit_j_filler(x86_condition_code_nz, backpatch_address) \
|
|
|
|
|
|
|
|
#define generate_condition_vs(ireg) \
|
2021-11-07 19:19:19 +01:00
|
|
|
generate_and_mem(1, base, REG_V_FLAG * 4); \
|
2021-09-17 19:18:50 +02:00
|
|
|
x86_emit_j_filler(x86_condition_code_z, backpatch_address) \
|
|
|
|
|
|
|
|
#define generate_condition_vc(ireg) \
|
2021-11-07 19:19:19 +01:00
|
|
|
generate_and_mem(1, base, REG_V_FLAG * 4); \
|
2021-09-17 19:18:50 +02:00
|
|
|
x86_emit_j_filler(x86_condition_code_nz, backpatch_address) \
|
|
|
|
|
|
|
|
#define generate_condition_hi(ireg) \
|
|
|
|
generate_load_reg(ireg, REG_C_FLAG); \
|
|
|
|
generate_xor_imm(ireg, 1); \
|
|
|
|
generate_or_mem(ireg, REG_Z_FLAG); \
|
|
|
|
x86_emit_j_filler(x86_condition_code_nz, backpatch_address) \
|
|
|
|
|
|
|
|
#define generate_condition_ls(ireg) \
|
|
|
|
generate_load_reg(ireg, REG_C_FLAG); \
|
|
|
|
generate_xor_imm(ireg, 1); \
|
|
|
|
generate_or_mem(ireg, REG_Z_FLAG); \
|
|
|
|
x86_emit_j_filler(x86_condition_code_z, backpatch_address) \
|
|
|
|
|
|
|
|
#define generate_condition_ge(ireg) \
|
|
|
|
generate_load_reg(ireg, REG_N_FLAG); \
|
|
|
|
generate_cmp_memreg(ireg, REG_V_FLAG); \
|
|
|
|
x86_emit_j_filler(x86_condition_code_nz, backpatch_address) \
|
|
|
|
|
|
|
|
#define generate_condition_lt(ireg) \
|
|
|
|
generate_load_reg(ireg, REG_N_FLAG); \
|
|
|
|
generate_cmp_memreg(ireg, REG_V_FLAG); \
|
|
|
|
x86_emit_j_filler(x86_condition_code_z, backpatch_address) \
|
|
|
|
|
|
|
|
#define generate_condition_gt(ireg) \
|
|
|
|
generate_load_reg(ireg, REG_N_FLAG); \
|
|
|
|
generate_xor_mem(ireg, REG_V_FLAG); \
|
|
|
|
generate_or_mem(ireg, REG_Z_FLAG); \
|
|
|
|
x86_emit_j_filler(x86_condition_code_nz, backpatch_address) \
|
|
|
|
|
|
|
|
#define generate_condition_le(ireg) \
|
|
|
|
generate_load_reg(ireg, REG_N_FLAG); \
|
|
|
|
generate_xor_mem(ireg, REG_V_FLAG); \
|
|
|
|
generate_or_mem(ireg, REG_Z_FLAG); \
|
|
|
|
x86_emit_j_filler(x86_condition_code_z, backpatch_address) \
|
|
|
|
|
|
|
|
|
|
|
|
#define generate_condition(ireg) \
|
2009-05-21 17:48:31 +02:00
|
|
|
switch(condition) \
|
|
|
|
{ \
|
|
|
|
case 0x0: \
|
2021-09-17 19:18:50 +02:00
|
|
|
generate_condition_eq(ireg); \
|
2009-05-21 17:48:31 +02:00
|
|
|
break; \
|
|
|
|
\
|
|
|
|
case 0x1: \
|
2021-09-17 19:18:50 +02:00
|
|
|
generate_condition_ne(ireg); \
|
2009-05-21 17:48:31 +02:00
|
|
|
break; \
|
|
|
|
\
|
|
|
|
case 0x2: \
|
2021-09-17 19:18:50 +02:00
|
|
|
generate_condition_cs(ireg); \
|
2009-05-21 17:48:31 +02:00
|
|
|
break; \
|
|
|
|
\
|
|
|
|
case 0x3: \
|
2021-09-17 19:18:50 +02:00
|
|
|
generate_condition_cc(ireg); \
|
2009-05-21 17:48:31 +02:00
|
|
|
break; \
|
|
|
|
\
|
|
|
|
case 0x4: \
|
2021-09-17 19:18:50 +02:00
|
|
|
generate_condition_mi(ireg); \
|
2009-05-21 17:48:31 +02:00
|
|
|
break; \
|
|
|
|
\
|
|
|
|
case 0x5: \
|
2021-09-17 19:18:50 +02:00
|
|
|
generate_condition_pl(ireg); \
|
2009-05-21 17:48:31 +02:00
|
|
|
break; \
|
|
|
|
\
|
|
|
|
case 0x6: \
|
2021-09-17 19:18:50 +02:00
|
|
|
generate_condition_vs(ireg); \
|
2009-05-21 17:48:31 +02:00
|
|
|
break; \
|
|
|
|
\
|
|
|
|
case 0x7: \
|
2021-09-17 19:18:50 +02:00
|
|
|
generate_condition_vc(ireg); \
|
2009-05-21 17:48:31 +02:00
|
|
|
break; \
|
|
|
|
\
|
|
|
|
case 0x8: \
|
2021-09-17 19:18:50 +02:00
|
|
|
generate_condition_hi(ireg); \
|
2009-05-21 17:48:31 +02:00
|
|
|
break; \
|
|
|
|
\
|
|
|
|
case 0x9: \
|
2021-09-17 19:18:50 +02:00
|
|
|
generate_condition_ls(ireg); \
|
2009-05-21 17:48:31 +02:00
|
|
|
break; \
|
|
|
|
\
|
|
|
|
case 0xA: \
|
2021-09-17 19:18:50 +02:00
|
|
|
generate_condition_ge(ireg); \
|
2009-05-21 17:48:31 +02:00
|
|
|
break; \
|
|
|
|
\
|
|
|
|
case 0xB: \
|
2021-09-17 19:18:50 +02:00
|
|
|
generate_condition_lt(ireg); \
|
2009-05-21 17:48:31 +02:00
|
|
|
break; \
|
|
|
|
\
|
|
|
|
case 0xC: \
|
2021-09-17 19:18:50 +02:00
|
|
|
generate_condition_gt(ireg); \
|
2009-05-21 17:48:31 +02:00
|
|
|
break; \
|
|
|
|
\
|
|
|
|
case 0xD: \
|
2021-09-17 19:18:50 +02:00
|
|
|
generate_condition_le(ireg); \
|
2009-05-21 17:48:31 +02:00
|
|
|
break; \
|
|
|
|
\
|
|
|
|
case 0xE: \
|
|
|
|
/* AL */ \
|
|
|
|
break; \
|
|
|
|
\
|
|
|
|
case 0xF: \
|
|
|
|
/* Reserved */ \
|
|
|
|
break; \
|
|
|
|
} \
|
|
|
|
|
|
|
|
#define generate_branch() \
|
|
|
|
{ \
|
|
|
|
if(condition == 0x0E) \
|
|
|
|
{ \
|
|
|
|
generate_branch_cycle_update( \
|
|
|
|
block_exits[block_exit_position].branch_source, \
|
|
|
|
block_exits[block_exit_position].branch_target); \
|
|
|
|
} \
|
|
|
|
else \
|
|
|
|
{ \
|
|
|
|
generate_branch_no_cycle_update( \
|
|
|
|
block_exits[block_exit_position].branch_source, \
|
|
|
|
block_exits[block_exit_position].branch_target); \
|
|
|
|
} \
|
|
|
|
block_exit_position++; \
|
|
|
|
} \
|
|
|
|
|
|
|
|
#define rm_op_reg rm
|
|
|
|
#define rm_op_imm imm
|
|
|
|
|
|
|
|
#define arm_data_proc_reg_flags() \
|
2021-08-31 00:42:47 +02:00
|
|
|
arm_decode_data_proc_reg(opcode); \
|
2009-05-21 17:48:31 +02:00
|
|
|
if(flag_status & 0x02) \
|
|
|
|
{ \
|
|
|
|
generate_load_rm_sh(flags) \
|
|
|
|
} \
|
|
|
|
else \
|
|
|
|
{ \
|
|
|
|
generate_load_rm_sh(no_flags); \
|
|
|
|
} \
|
|
|
|
|
|
|
|
#define arm_data_proc_reg() \
|
2014-12-10 01:17:37 +01:00
|
|
|
arm_decode_data_proc_reg(opcode); \
|
2009-05-21 17:48:31 +02:00
|
|
|
generate_load_rm_sh(no_flags) \
|
|
|
|
|
|
|
|
#define arm_data_proc_imm() \
|
2014-12-10 01:17:37 +01:00
|
|
|
arm_decode_data_proc_imm(opcode); \
|
2009-05-21 17:48:31 +02:00
|
|
|
ror(imm, imm, imm_ror); \
|
|
|
|
generate_load_imm(a0, imm) \
|
|
|
|
|
|
|
|
#define arm_data_proc_imm_flags() \
|
2014-12-10 01:17:37 +01:00
|
|
|
arm_decode_data_proc_imm(opcode); \
|
2009-05-21 17:48:31 +02:00
|
|
|
if((flag_status & 0x02) && (imm_ror != 0)) \
|
|
|
|
{ \
|
|
|
|
/* Generate carry flag from integer rotation */ \
|
|
|
|
generate_load_imm(a0, ((imm >> (imm_ror - 1)) & 0x01)); \
|
|
|
|
generate_store_reg(a0, REG_C_FLAG); \
|
|
|
|
} \
|
|
|
|
ror(imm, imm, imm_ror); \
|
|
|
|
generate_load_imm(a0, imm) \
|
|
|
|
|
|
|
|
|
|
|
|
#define arm_data_proc(name, type, flags_op) \
|
|
|
|
{ \
|
|
|
|
arm_data_proc_##type(); \
|
|
|
|
generate_load_reg_pc(a1, rn, 8); \
|
2021-08-31 00:42:47 +02:00
|
|
|
arm_data_proc_##name(rd, generate_store_reg_pc_##flags_op); \
|
2009-05-21 17:48:31 +02:00
|
|
|
} \
|
|
|
|
|
|
|
|
#define arm_data_proc_test(name, type) \
|
|
|
|
{ \
|
|
|
|
arm_data_proc_##type(); \
|
|
|
|
generate_load_reg_pc(a1, rn, 8); \
|
2021-08-31 00:42:47 +02:00
|
|
|
arm_data_proc_test_##name(); \
|
2009-05-21 17:48:31 +02:00
|
|
|
} \
|
|
|
|
|
|
|
|
#define arm_data_proc_unary(name, type, flags_op) \
|
|
|
|
{ \
|
|
|
|
arm_data_proc_##type(); \
|
2021-08-31 00:42:47 +02:00
|
|
|
arm_data_proc_unary_##name(rd, generate_store_reg_pc_##flags_op); \
|
2009-05-21 17:48:31 +02:00
|
|
|
} \
|
|
|
|
|
|
|
|
#define arm_data_proc_mov(type) \
|
|
|
|
{ \
|
|
|
|
arm_data_proc_##type(); \
|
|
|
|
generate_store_reg_pc_no_flags(a0, rd); \
|
|
|
|
} \
|
|
|
|
|
|
|
|
#define arm_multiply_flags_yes() \
|
2021-12-06 17:08:18 +01:00
|
|
|
generate_and(a0, a0); \
|
2021-08-31 21:18:22 +02:00
|
|
|
generate_update_flag(z, REG_Z_FLAG) \
|
|
|
|
generate_update_flag(s, REG_N_FLAG)
|
2009-05-21 17:48:31 +02:00
|
|
|
|
|
|
|
#define arm_multiply_flags_no(_dest) \
|
|
|
|
|
|
|
|
#define arm_multiply_add_no() \
|
|
|
|
|
|
|
|
#define arm_multiply_add_yes() \
|
|
|
|
generate_load_reg(a1, rn); \
|
|
|
|
generate_add(a0, a1) \
|
|
|
|
|
|
|
|
#define arm_multiply(add_op, flags) \
|
|
|
|
{ \
|
|
|
|
arm_decode_multiply(); \
|
|
|
|
generate_load_reg(a0, rm); \
|
|
|
|
generate_load_reg(a1, rs); \
|
|
|
|
generate_multiply(a1); \
|
|
|
|
arm_multiply_add_##add_op(); \
|
|
|
|
arm_multiply_flags_##flags(); \
|
2021-12-06 17:08:18 +01:00
|
|
|
generate_store_reg(a0, rd); \
|
2009-05-21 17:48:31 +02:00
|
|
|
} \
|
|
|
|
|
|
|
|
#define arm_multiply_long_flags_yes() \
|
2021-11-04 20:22:42 +01:00
|
|
|
generate_mov(t0, a1); \
|
|
|
|
generate_and(t0, t0); \
|
2021-08-31 21:18:22 +02:00
|
|
|
generate_update_flag(s, REG_N_FLAG) \
|
2021-11-04 20:22:42 +01:00
|
|
|
generate_or(t0, a0); \
|
2021-08-31 21:18:22 +02:00
|
|
|
generate_update_flag(z, REG_Z_FLAG) \
|
2009-05-21 17:48:31 +02:00
|
|
|
|
|
|
|
#define arm_multiply_long_flags_no(_dest) \
|
|
|
|
|
|
|
|
#define arm_multiply_long_add_yes(name) \
|
|
|
|
generate_load_reg(a2, rdlo); \
|
2021-11-04 20:22:42 +01:00
|
|
|
generate_load_reg(t0, rdhi); \
|
|
|
|
generate_multiply_##name(a1, a2, t0) \
|
2009-05-21 17:48:31 +02:00
|
|
|
|
|
|
|
#define arm_multiply_long_add_no(name) \
|
|
|
|
generate_multiply_##name(a1) \
|
|
|
|
|
|
|
|
#define arm_multiply_long(name, add_op, flags) \
|
|
|
|
{ \
|
|
|
|
arm_decode_multiply_long(); \
|
|
|
|
generate_load_reg(a0, rm); \
|
|
|
|
generate_load_reg(a1, rs); \
|
|
|
|
arm_multiply_long_add_##add_op(name); \
|
|
|
|
generate_store_reg(a0, rdlo); \
|
|
|
|
generate_store_reg(a1, rdhi); \
|
|
|
|
arm_multiply_long_flags_##flags(); \
|
|
|
|
} \
|
|
|
|
|
2021-09-17 20:00:37 +02:00
|
|
|
#define execute_read_cpsr(oreg) \
|
|
|
|
collapse_flags(oreg, a2)
|
2009-05-21 17:48:31 +02:00
|
|
|
|
2021-09-17 20:00:37 +02:00
|
|
|
#define execute_read_spsr(oreg) \
|
|
|
|
collapse_flags(oreg, a2); \
|
|
|
|
generate_load_reg(oreg, CPU_MODE); \
|
2023-01-11 20:44:56 +01:00
|
|
|
generate_and_imm(oreg, 0xF); \
|
2021-09-17 20:00:37 +02:00
|
|
|
generate_load_spsr(oreg, oreg); \
|
2009-05-21 17:48:31 +02:00
|
|
|
|
|
|
|
#define arm_psr_read(op_type, psr_reg) \
|
2021-09-17 20:00:37 +02:00
|
|
|
execute_read_##psr_reg(rv); \
|
2009-05-21 17:48:31 +02:00
|
|
|
generate_store_reg(rv, rd) \
|
|
|
|
|
2021-12-10 18:32:04 +01:00
|
|
|
// Does mode-change magic (including IRQ checks)
|
2021-11-06 12:17:50 +01:00
|
|
|
u32 execute_store_cpsr_body()
|
2009-05-21 17:48:31 +02:00
|
|
|
{
|
2023-01-11 20:44:56 +01:00
|
|
|
set_cpu_mode(cpu_modes[reg[REG_CPSR] & 0xF]);
|
2021-12-10 18:32:04 +01:00
|
|
|
if((io_registers[REG_IE] & io_registers[REG_IF]) &&
|
|
|
|
io_registers[REG_IME] && ((reg[REG_CPSR] & 0x80) == 0))
|
2009-05-21 17:48:31 +02:00
|
|
|
{
|
2023-01-11 20:44:56 +01:00
|
|
|
REG_MODE(MODE_IRQ)[6] = reg[REG_PC] + 4;
|
|
|
|
REG_SPSR(MODE_IRQ) = reg[REG_CPSR];
|
2021-12-10 18:32:04 +01:00
|
|
|
reg[REG_CPSR] = (reg[REG_CPSR] & 0xFFFFFF00) | 0xD2;
|
|
|
|
set_cpu_mode(MODE_IRQ);
|
|
|
|
return 0x00000018;
|
2009-05-21 17:48:31 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#define arm_psr_load_new_reg() \
|
|
|
|
generate_load_reg(a0, rm) \
|
|
|
|
|
|
|
|
#define arm_psr_load_new_imm() \
|
|
|
|
ror(imm, imm, imm_ror); \
|
|
|
|
generate_load_imm(a0, imm) \
|
|
|
|
|
2021-11-02 20:50:41 +01:00
|
|
|
#define execute_store_cpsr() \
|
2023-01-11 20:44:56 +01:00
|
|
|
generate_load_imm(a1, cpsr_masks[psr_pfield][0]); \
|
|
|
|
generate_load_imm(a2, cpsr_masks[psr_pfield][1]); \
|
2021-12-10 18:32:04 +01:00
|
|
|
generate_store_reg_i32(pc, REG_PC); \
|
2021-11-02 20:50:41 +01:00
|
|
|
generate_function_call(execute_store_cpsr) \
|
|
|
|
|
2023-01-11 20:44:56 +01:00
|
|
|
/* REG_SPSR(reg[CPU_MODE]) = (new_spsr & store_mask) | (old_spsr & (~store_mask))*/
|
2021-11-02 20:50:41 +01:00
|
|
|
#define execute_store_spsr() \
|
|
|
|
generate_load_reg(a2, CPU_MODE); \
|
2023-01-11 20:44:56 +01:00
|
|
|
generate_and_imm(a2, 0xF); \
|
2021-11-02 20:50:41 +01:00
|
|
|
generate_load_spsr(a1, a2); \
|
2023-01-11 20:44:56 +01:00
|
|
|
generate_and_imm(a0, spsr_masks[psr_pfield]); \
|
|
|
|
generate_and_imm(a1, ~spsr_masks[psr_pfield]); \
|
2021-11-02 20:50:41 +01:00
|
|
|
generate_or(a0, a1); \
|
|
|
|
generate_store_spsr(a0, a2); \
|
|
|
|
|
2009-05-21 17:48:31 +02:00
|
|
|
#define arm_psr_store(op_type, psr_reg) \
|
|
|
|
arm_psr_load_new_##op_type(); \
|
2021-11-02 20:50:41 +01:00
|
|
|
execute_store_##psr_reg(); \
|
2009-05-21 17:48:31 +02:00
|
|
|
|
|
|
|
#define arm_psr(op_type, transfer_type, psr_reg) \
|
|
|
|
{ \
|
2014-12-10 01:17:37 +01:00
|
|
|
arm_decode_psr_##op_type(opcode); \
|
2009-05-21 17:48:31 +02:00
|
|
|
arm_psr_##transfer_type(op_type, psr_reg); \
|
|
|
|
} \
|
|
|
|
|
|
|
|
#define arm_access_memory_load(mem_type) \
|
|
|
|
cycle_count += 2; \
|
2021-11-03 22:20:31 +01:00
|
|
|
generate_load_pc(a1, pc); \
|
2009-05-21 17:48:31 +02:00
|
|
|
generate_function_call(execute_load_##mem_type); \
|
|
|
|
generate_store_reg_pc_no_flags(rv, rd) \
|
|
|
|
|
|
|
|
#define arm_access_memory_store(mem_type) \
|
|
|
|
cycle_count++; \
|
|
|
|
generate_load_reg_pc(a1, rd, 12); \
|
2021-11-02 20:50:41 +01:00
|
|
|
generate_store_reg_i32(pc + 4, REG_PC); \
|
2009-05-21 17:48:31 +02:00
|
|
|
generate_function_call(execute_store_##mem_type) \
|
|
|
|
|
|
|
|
#define no_op \
|
|
|
|
|
|
|
|
#define arm_access_memory_writeback_yes(off_op) \
|
|
|
|
reg[rn] = address off_op \
|
|
|
|
|
|
|
|
#define arm_access_memory_writeback_no(off_op) \
|
|
|
|
|
|
|
|
#define load_reg_op reg[rd] \
|
|
|
|
|
|
|
|
#define store_reg_op reg_op \
|
|
|
|
|
|
|
|
#define arm_access_memory_adjust_op_up add
|
|
|
|
#define arm_access_memory_adjust_op_down sub
|
|
|
|
#define arm_access_memory_reverse_op_up sub
|
|
|
|
#define arm_access_memory_reverse_op_down add
|
|
|
|
|
|
|
|
#define arm_access_memory_reg_pre(adjust_dir_op, reverse_dir_op) \
|
|
|
|
generate_load_reg_pc(a0, rn, 8); \
|
|
|
|
generate_##adjust_dir_op(a0, a1) \
|
|
|
|
|
|
|
|
#define arm_access_memory_reg_pre_wb(adjust_dir_op, reverse_dir_op) \
|
|
|
|
arm_access_memory_reg_pre(adjust_dir_op, reverse_dir_op); \
|
|
|
|
generate_store_reg(a0, rn) \
|
|
|
|
|
|
|
|
#define arm_access_memory_reg_post(adjust_dir_op, reverse_dir_op) \
|
|
|
|
generate_load_reg(a0, rn); \
|
|
|
|
generate_##adjust_dir_op(a0, a1); \
|
|
|
|
generate_store_reg(a0, rn); \
|
|
|
|
generate_##reverse_dir_op(a0, a1) \
|
|
|
|
|
|
|
|
#define arm_access_memory_imm_pre(adjust_dir_op, reverse_dir_op) \
|
|
|
|
generate_load_reg_pc(a0, rn, 8); \
|
|
|
|
generate_##adjust_dir_op##_imm(a0, offset) \
|
|
|
|
|
|
|
|
#define arm_access_memory_imm_pre_wb(adjust_dir_op, reverse_dir_op) \
|
|
|
|
arm_access_memory_imm_pre(adjust_dir_op, reverse_dir_op); \
|
|
|
|
generate_store_reg(a0, rn) \
|
|
|
|
|
|
|
|
#define arm_access_memory_imm_post(adjust_dir_op, reverse_dir_op) \
|
|
|
|
generate_load_reg(a0, rn); \
|
|
|
|
generate_##adjust_dir_op##_imm(a0, offset); \
|
|
|
|
generate_store_reg(a0, rn); \
|
|
|
|
generate_##reverse_dir_op##_imm(a0, offset) \
|
|
|
|
|
|
|
|
|
|
|
|
#define arm_data_trans_reg(adjust_op, adjust_dir_op, reverse_dir_op) \
|
|
|
|
arm_decode_data_trans_reg(); \
|
|
|
|
generate_load_offset_sh(); \
|
|
|
|
arm_access_memory_reg_##adjust_op(adjust_dir_op, reverse_dir_op) \
|
|
|
|
|
|
|
|
#define arm_data_trans_imm(adjust_op, adjust_dir_op, reverse_dir_op) \
|
|
|
|
arm_decode_data_trans_imm(); \
|
|
|
|
arm_access_memory_imm_##adjust_op(adjust_dir_op, reverse_dir_op) \
|
|
|
|
|
|
|
|
#define arm_data_trans_half_reg(adjust_op, adjust_dir_op, reverse_dir_op) \
|
|
|
|
arm_decode_half_trans_r(); \
|
|
|
|
generate_load_reg(a1, rm); \
|
|
|
|
arm_access_memory_reg_##adjust_op(adjust_dir_op, reverse_dir_op) \
|
|
|
|
|
|
|
|
#define arm_data_trans_half_imm(adjust_op, adjust_dir_op, reverse_dir_op) \
|
|
|
|
arm_decode_half_trans_of(); \
|
|
|
|
arm_access_memory_imm_##adjust_op(adjust_dir_op, reverse_dir_op) \
|
|
|
|
|
|
|
|
#define arm_access_memory(access_type, direction, adjust_op, mem_type, \
|
|
|
|
offset_type) \
|
|
|
|
{ \
|
|
|
|
arm_data_trans_##offset_type(adjust_op, \
|
|
|
|
arm_access_memory_adjust_op_##direction, \
|
|
|
|
arm_access_memory_reverse_op_##direction); \
|
|
|
|
\
|
|
|
|
arm_access_memory_##access_type(mem_type); \
|
|
|
|
} \
|
|
|
|
|
|
|
|
#define word_bit_count(word) \
|
|
|
|
(bit_count[word >> 8] + bit_count[word & 0xFF]) \
|
|
|
|
|
|
|
|
|
|
|
|
#define arm_block_memory_load() \
|
2021-11-03 22:20:31 +01:00
|
|
|
generate_load_pc(a1, pc); \
|
|
|
|
generate_function_call(execute_load_u32); \
|
2009-05-21 17:48:31 +02:00
|
|
|
generate_store_reg(rv, i) \
|
|
|
|
|
|
|
|
#define arm_block_memory_store() \
|
|
|
|
generate_load_reg_pc(a1, i, 8); \
|
2021-11-02 23:16:47 +01:00
|
|
|
generate_function_call(execute_store_aligned_u32) \
|
2009-05-21 17:48:31 +02:00
|
|
|
|
|
|
|
#define arm_block_memory_final_load() \
|
|
|
|
arm_block_memory_load() \
|
|
|
|
|
|
|
|
#define arm_block_memory_final_store() \
|
|
|
|
generate_load_reg_pc(a1, i, 12); \
|
2021-11-02 20:50:41 +01:00
|
|
|
generate_store_reg_i32(pc + 4, REG_PC); \
|
2009-05-21 17:48:31 +02:00
|
|
|
generate_function_call(execute_store_u32) \
|
|
|
|
|
|
|
|
#define arm_block_memory_adjust_pc_store() \
|
|
|
|
|
|
|
|
#define arm_block_memory_adjust_pc_load() \
|
|
|
|
if(reg_list & 0x8000) \
|
|
|
|
{ \
|
|
|
|
generate_indirect_branch_arm(); \
|
|
|
|
} \
|
|
|
|
|
|
|
|
#define arm_block_memory_offset_down_a() \
|
2021-11-04 20:22:42 +01:00
|
|
|
generate_add_imm(a0, -((word_bit_count(reg_list) * 4) - 4)) \
|
2009-05-21 17:48:31 +02:00
|
|
|
|
|
|
|
#define arm_block_memory_offset_down_b() \
|
2021-11-04 20:22:42 +01:00
|
|
|
generate_add_imm(a0, -(word_bit_count(reg_list) * 4)) \
|
2009-05-21 17:48:31 +02:00
|
|
|
|
|
|
|
#define arm_block_memory_offset_no() \
|
|
|
|
|
|
|
|
#define arm_block_memory_offset_up() \
|
2021-11-04 20:22:42 +01:00
|
|
|
generate_add_imm(a0, 4) \
|
2009-05-21 17:48:31 +02:00
|
|
|
|
|
|
|
#define arm_block_memory_writeback_down() \
|
|
|
|
generate_load_reg(a0, rn) \
|
|
|
|
generate_add_imm(a0, -(word_bit_count(reg_list) * 4)); \
|
|
|
|
generate_store_reg(a0, rn) \
|
|
|
|
|
|
|
|
#define arm_block_memory_writeback_up() \
|
|
|
|
generate_load_reg(a0, rn); \
|
|
|
|
generate_add_imm(a0, (word_bit_count(reg_list) * 4)); \
|
|
|
|
generate_store_reg(a0, rn) \
|
|
|
|
|
|
|
|
#define arm_block_memory_writeback_no()
|
|
|
|
|
|
|
|
// Only emit writeback if the register is not in the list
|
|
|
|
|
|
|
|
#define arm_block_memory_writeback_load(writeback_type) \
|
|
|
|
if(!((reg_list >> rn) & 0x01)) \
|
|
|
|
{ \
|
|
|
|
arm_block_memory_writeback_##writeback_type(); \
|
|
|
|
} \
|
|
|
|
|
|
|
|
#define arm_block_memory_writeback_store(writeback_type) \
|
|
|
|
arm_block_memory_writeback_##writeback_type() \
|
|
|
|
|
|
|
|
#define arm_block_memory(access_type, offset_type, writeback_type, s_bit) \
|
|
|
|
{ \
|
|
|
|
arm_decode_block_trans(); \
|
|
|
|
u32 offset = 0; \
|
|
|
|
u32 i; \
|
|
|
|
\
|
2021-11-04 20:22:42 +01:00
|
|
|
generate_load_reg(a0, rn); \
|
2009-05-21 17:48:31 +02:00
|
|
|
arm_block_memory_offset_##offset_type(); \
|
2021-11-04 20:22:42 +01:00
|
|
|
generate_and_imm(a0, ~0x03); \
|
|
|
|
generate_store_reg(a0, REG_SAVE3); \
|
2009-05-21 17:48:31 +02:00
|
|
|
arm_block_memory_writeback_##access_type(writeback_type); \
|
|
|
|
\
|
|
|
|
for(i = 0; i < 16; i++) \
|
|
|
|
{ \
|
|
|
|
if((reg_list >> i) & 0x01) \
|
|
|
|
{ \
|
|
|
|
cycle_count++; \
|
2021-11-04 20:22:42 +01:00
|
|
|
generate_load_reg(a0, REG_SAVE3); \
|
|
|
|
generate_add_imm(a0, offset) \
|
2009-05-21 17:48:31 +02:00
|
|
|
if(reg_list & ~((2 << i) - 1)) \
|
|
|
|
{ \
|
|
|
|
arm_block_memory_##access_type(); \
|
|
|
|
offset += 4; \
|
|
|
|
} \
|
|
|
|
else \
|
|
|
|
{ \
|
|
|
|
arm_block_memory_final_##access_type(); \
|
|
|
|
} \
|
|
|
|
} \
|
|
|
|
} \
|
|
|
|
\
|
|
|
|
arm_block_memory_adjust_pc_##access_type(); \
|
|
|
|
} \
|
|
|
|
|
|
|
|
#define arm_swap(type) \
|
|
|
|
{ \
|
|
|
|
arm_decode_swap(); \
|
|
|
|
cycle_count += 3; \
|
|
|
|
generate_load_reg(a0, rn); \
|
2021-11-03 22:20:31 +01:00
|
|
|
generate_load_pc(a1, pc); \
|
2009-05-21 17:48:31 +02:00
|
|
|
generate_function_call(execute_load_##type); \
|
2021-11-04 20:22:42 +01:00
|
|
|
generate_mov(a2, rv); \
|
2009-05-21 17:48:31 +02:00
|
|
|
generate_load_reg(a0, rn); \
|
|
|
|
generate_load_reg(a1, rm); \
|
2021-11-04 20:22:42 +01:00
|
|
|
generate_store_reg(a2, rd); \
|
2009-05-21 17:48:31 +02:00
|
|
|
generate_function_call(execute_store_##type); \
|
|
|
|
} \
|
|
|
|
|
|
|
|
#define thumb_rn_op_reg(_rn) \
|
|
|
|
generate_load_reg(a0, _rn) \
|
|
|
|
|
|
|
|
#define thumb_rn_op_imm(_imm) \
|
|
|
|
generate_load_imm(a0, _imm) \
|
|
|
|
|
|
|
|
// Types: add_sub, add_sub_imm, alu_op, imm
|
|
|
|
// Affects N/Z/C/V flags
|
|
|
|
|
|
|
|
#define thumb_data_proc(type, name, rn_type, _rd, _rs, _rn) \
|
|
|
|
{ \
|
|
|
|
thumb_decode_##type(); \
|
|
|
|
thumb_rn_op_##rn_type(_rn); \
|
|
|
|
generate_load_reg(a1, _rs); \
|
2021-08-31 00:42:47 +02:00
|
|
|
arm_data_proc_##name(_rd, generate_store_reg); \
|
2009-05-21 17:48:31 +02:00
|
|
|
} \
|
|
|
|
|
|
|
|
#define thumb_data_proc_test(type, name, rn_type, _rs, _rn) \
|
|
|
|
{ \
|
|
|
|
thumb_decode_##type(); \
|
|
|
|
thumb_rn_op_##rn_type(_rn); \
|
|
|
|
generate_load_reg(a1, _rs); \
|
2021-08-31 00:42:47 +02:00
|
|
|
arm_data_proc_test_##name(); \
|
2009-05-21 17:48:31 +02:00
|
|
|
} \
|
|
|
|
|
|
|
|
#define thumb_data_proc_unary(type, name, rn_type, _rd, _rn) \
|
|
|
|
{ \
|
|
|
|
thumb_decode_##type(); \
|
|
|
|
thumb_rn_op_##rn_type(_rn); \
|
2021-08-31 00:42:47 +02:00
|
|
|
arm_data_proc_unary_##name(_rd, generate_store_reg); \
|
2009-05-21 17:48:31 +02:00
|
|
|
} \
|
|
|
|
|
|
|
|
#define thumb_data_proc_mov(type, rn_type, _rd, _rn) \
|
|
|
|
{ \
|
|
|
|
thumb_decode_##type(); \
|
|
|
|
thumb_rn_op_##rn_type(_rn); \
|
|
|
|
generate_store_reg(a0, _rd); \
|
|
|
|
} \
|
|
|
|
|
2021-08-31 00:42:47 +02:00
|
|
|
#define generate_store_reg_pc_thumb(ireg, rd) \
|
2009-05-21 17:48:31 +02:00
|
|
|
generate_store_reg(ireg, rd); \
|
|
|
|
if(rd == 15) \
|
|
|
|
{ \
|
|
|
|
generate_indirect_branch_cycle_update(thumb); \
|
|
|
|
} \
|
|
|
|
|
|
|
|
#define thumb_data_proc_hi(name) \
|
|
|
|
{ \
|
|
|
|
thumb_decode_hireg_op(); \
|
|
|
|
generate_load_reg_pc(a0, rs, 4); \
|
|
|
|
generate_load_reg_pc(a1, rd, 4); \
|
2021-08-31 00:42:47 +02:00
|
|
|
arm_data_proc_##name(rd, generate_store_reg_pc_thumb); \
|
2009-05-21 17:48:31 +02:00
|
|
|
} \
|
|
|
|
|
|
|
|
#define thumb_data_proc_test_hi(name) \
|
|
|
|
{ \
|
|
|
|
thumb_decode_hireg_op(); \
|
|
|
|
generate_load_reg_pc(a0, rs, 4); \
|
|
|
|
generate_load_reg_pc(a1, rd, 4); \
|
2021-08-31 00:42:47 +02:00
|
|
|
arm_data_proc_test_##name(); \
|
2009-05-21 17:48:31 +02:00
|
|
|
} \
|
|
|
|
|
|
|
|
#define thumb_data_proc_unary_hi(name) \
|
|
|
|
{ \
|
|
|
|
thumb_decode_hireg_op(); \
|
|
|
|
generate_load_reg_pc(a0, rn, 4); \
|
2021-08-31 00:42:47 +02:00
|
|
|
arm_data_proc_unary_##name(rd, generate_store_reg_pc_thumb); \
|
2009-05-21 17:48:31 +02:00
|
|
|
} \
|
|
|
|
|
|
|
|
#define thumb_data_proc_mov_hi() \
|
|
|
|
{ \
|
|
|
|
thumb_decode_hireg_op(); \
|
|
|
|
generate_load_reg_pc(a0, rs, 4); \
|
2021-08-31 00:42:47 +02:00
|
|
|
generate_store_reg_pc_thumb(a0, rd); \
|
2009-05-21 17:48:31 +02:00
|
|
|
} \
|
|
|
|
|
|
|
|
#define thumb_load_pc(_rd) \
|
|
|
|
{ \
|
|
|
|
thumb_decode_imm(); \
|
|
|
|
generate_load_pc(a0, (((pc & ~2) + 4) + (imm * 4))); \
|
|
|
|
generate_store_reg(a0, _rd); \
|
|
|
|
} \
|
|
|
|
|
|
|
|
#define thumb_load_sp(_rd) \
|
|
|
|
{ \
|
|
|
|
thumb_decode_imm(); \
|
|
|
|
generate_load_reg(a0, 13); \
|
|
|
|
generate_add_imm(a0, (imm * 4)); \
|
|
|
|
generate_store_reg(a0, _rd); \
|
|
|
|
} \
|
|
|
|
|
|
|
|
#define thumb_adjust_sp_up() \
|
|
|
|
generate_add_imm(a0, imm * 4) \
|
|
|
|
|
|
|
|
#define thumb_adjust_sp_down() \
|
|
|
|
generate_sub_imm(a0, imm * 4) \
|
|
|
|
|
|
|
|
|
|
|
|
#define thumb_adjust_sp(direction) \
|
|
|
|
{ \
|
|
|
|
thumb_decode_add_sp(); \
|
|
|
|
generate_load_reg(a0, REG_SP); \
|
|
|
|
thumb_adjust_sp_##direction(); \
|
|
|
|
generate_store_reg(a0, REG_SP); \
|
|
|
|
} \
|
|
|
|
|
|
|
|
// Decode types: shift, alu_op
|
|
|
|
// Operation types: lsl, lsr, asr, ror
|
|
|
|
// Affects N/Z/C flags
|
|
|
|
|
2021-09-01 19:34:43 +02:00
|
|
|
#define thumb_lsl_imm_op() \
|
|
|
|
if (imm) { \
|
|
|
|
generate_shift_left(a0, imm); \
|
|
|
|
generate_update_flag(c, REG_C_FLAG) \
|
|
|
|
} else { \
|
|
|
|
generate_or(a0, a0); \
|
|
|
|
} \
|
2021-12-13 18:31:01 +01:00
|
|
|
update_logical_flags() \
|
2009-05-21 17:48:31 +02:00
|
|
|
|
2021-09-01 19:34:43 +02:00
|
|
|
#define thumb_lsr_imm_op() \
|
|
|
|
if (imm) { \
|
|
|
|
generate_shift_right(a0, imm); \
|
|
|
|
generate_update_flag(c, REG_C_FLAG) \
|
|
|
|
} else { \
|
|
|
|
generate_shift_right(a0, 31); \
|
|
|
|
generate_update_flag(nz, REG_C_FLAG) \
|
|
|
|
generate_xor(a0, a0); \
|
|
|
|
} \
|
2021-12-13 18:31:01 +01:00
|
|
|
update_logical_flags() \
|
2009-05-21 17:48:31 +02:00
|
|
|
|
2021-09-01 19:34:43 +02:00
|
|
|
#define thumb_asr_imm_op() \
|
|
|
|
if (imm) { \
|
|
|
|
generate_shift_right_arithmetic(a0, imm); \
|
|
|
|
generate_update_flag(c, REG_C_FLAG) \
|
|
|
|
} else { \
|
|
|
|
generate_shift_right_arithmetic(a0, 31); \
|
|
|
|
generate_update_flag(s, REG_C_FLAG) \
|
|
|
|
} \
|
2021-12-13 18:31:01 +01:00
|
|
|
update_logical_flags() \
|
2009-05-21 17:48:31 +02:00
|
|
|
|
2021-09-01 19:34:43 +02:00
|
|
|
#define thumb_ror_imm_op() \
|
|
|
|
if (imm) { \
|
|
|
|
generate_rotate_right(a0, imm); \
|
|
|
|
generate_update_flag(c, REG_C_FLAG) \
|
|
|
|
} else { \
|
|
|
|
generate_rrx_flags(a0); \
|
|
|
|
} \
|
2021-12-13 18:31:01 +01:00
|
|
|
update_logical_flags() \
|
2009-05-21 17:48:31 +02:00
|
|
|
|
|
|
|
|
|
|
|
#define generate_shift_load_operands_reg() \
|
|
|
|
generate_load_reg(a0, rd); \
|
|
|
|
generate_load_reg(a1, rs) \
|
|
|
|
|
|
|
|
#define generate_shift_load_operands_imm() \
|
|
|
|
generate_load_reg(a0, rs); \
|
|
|
|
generate_load_imm(a1, imm) \
|
|
|
|
|
2021-09-01 19:34:43 +02:00
|
|
|
#define thumb_shift_operation_imm(op_type) \
|
|
|
|
thumb_##op_type##_imm_op()
|
|
|
|
|
|
|
|
#define thumb_shift_operation_reg(op_type) \
|
|
|
|
generate_##op_type##_flags_reg(a0); \
|
|
|
|
generate_or(a0, a0); \
|
2021-12-13 18:31:01 +01:00
|
|
|
update_logical_flags() \
|
2021-09-01 19:34:43 +02:00
|
|
|
|
2009-05-21 17:48:31 +02:00
|
|
|
#define thumb_shift(decode_type, op_type, value_type) \
|
|
|
|
{ \
|
|
|
|
thumb_decode_##decode_type(); \
|
|
|
|
generate_shift_load_operands_##value_type(); \
|
2021-09-01 19:34:43 +02:00
|
|
|
thumb_shift_operation_##value_type(op_type); \
|
2009-05-21 17:48:31 +02:00
|
|
|
generate_store_reg(rv, rd); \
|
|
|
|
} \
|
|
|
|
|
|
|
|
// Operation types: imm, mem_reg, mem_imm
|
|
|
|
|
2021-05-07 20:41:54 +02:00
|
|
|
#define thumb_load_pc_pool_const(reg_rd, value) \
|
2021-11-02 20:50:41 +01:00
|
|
|
generate_store_reg_i32(value, reg_rd) \
|
2021-05-07 20:41:54 +02:00
|
|
|
|
2009-05-21 17:48:31 +02:00
|
|
|
#define thumb_access_memory_load(mem_type, reg_rd) \
|
|
|
|
cycle_count += 2; \
|
2021-11-03 22:20:31 +01:00
|
|
|
generate_load_pc(a1, pc); \
|
2009-05-21 17:48:31 +02:00
|
|
|
generate_function_call(execute_load_##mem_type); \
|
|
|
|
generate_store_reg(rv, reg_rd) \
|
|
|
|
|
|
|
|
#define thumb_access_memory_store(mem_type, reg_rd) \
|
|
|
|
cycle_count++; \
|
|
|
|
generate_load_reg(a1, reg_rd); \
|
2021-11-02 20:50:41 +01:00
|
|
|
generate_store_reg_i32(pc + 2, REG_PC); \
|
2009-05-21 17:48:31 +02:00
|
|
|
generate_function_call(execute_store_##mem_type) \
|
|
|
|
|
|
|
|
#define thumb_access_memory_generate_address_pc_relative(offset, _rb, _ro) \
|
|
|
|
generate_load_pc(a0, (offset)) \
|
|
|
|
|
|
|
|
#define thumb_access_memory_generate_address_reg_imm_sp(offset, _rb, _ro) \
|
|
|
|
generate_load_reg(a0, _rb); \
|
|
|
|
generate_add_imm(a0, (offset * 4)) \
|
|
|
|
|
|
|
|
#define thumb_access_memory_generate_address_reg_imm(offset, _rb, _ro) \
|
|
|
|
generate_load_reg(a0, _rb); \
|
|
|
|
generate_add_imm(a0, (offset)) \
|
|
|
|
|
|
|
|
#define thumb_access_memory_generate_address_reg_reg(offset, _rb, _ro) \
|
|
|
|
generate_load_reg(a0, _rb); \
|
|
|
|
generate_load_reg(a1, _ro); \
|
|
|
|
generate_add(a0, a1) \
|
|
|
|
|
|
|
|
#define thumb_access_memory(access_type, op_type, _rd, _rb, _ro, \
|
|
|
|
address_type, offset, mem_type) \
|
|
|
|
{ \
|
|
|
|
thumb_decode_##op_type(); \
|
|
|
|
thumb_access_memory_generate_address_##address_type(offset, _rb, _ro); \
|
|
|
|
thumb_access_memory_##access_type(mem_type, _rd); \
|
|
|
|
} \
|
|
|
|
|
|
|
|
#define thumb_block_address_preadjust_up() \
|
2021-11-04 20:22:42 +01:00
|
|
|
generate_add_imm(a0, (bit_count[reg_list] * 4)) \
|
2009-05-21 17:48:31 +02:00
|
|
|
|
|
|
|
#define thumb_block_address_preadjust_down() \
|
2021-11-04 20:22:42 +01:00
|
|
|
generate_sub_imm(a0, (bit_count[reg_list] * 4)) \
|
2009-05-21 17:48:31 +02:00
|
|
|
|
|
|
|
#define thumb_block_address_preadjust_push_lr() \
|
2021-11-04 20:22:42 +01:00
|
|
|
generate_sub_imm(a0, ((bit_count[reg_list] + 1) * 4)) \
|
2009-05-21 17:48:31 +02:00
|
|
|
|
|
|
|
#define thumb_block_address_preadjust_no() \
|
|
|
|
|
|
|
|
#define thumb_block_address_postadjust_no(base_reg) \
|
2021-11-04 20:22:42 +01:00
|
|
|
generate_store_reg(a0, base_reg) \
|
2009-05-21 17:48:31 +02:00
|
|
|
|
|
|
|
#define thumb_block_address_postadjust_up(base_reg) \
|
2021-11-04 20:22:42 +01:00
|
|
|
generate_add_imm(a0, (bit_count[reg_list] * 4)); \
|
2009-05-21 17:48:31 +02:00
|
|
|
generate_store_reg(a0, base_reg) \
|
|
|
|
|
|
|
|
#define thumb_block_address_postadjust_down(base_reg) \
|
|
|
|
generate_sub_imm(a0, (bit_count[reg_list] * 4)); \
|
|
|
|
generate_store_reg(a0, base_reg) \
|
|
|
|
|
|
|
|
#define thumb_block_address_postadjust_pop_pc(base_reg) \
|
2021-11-04 20:22:42 +01:00
|
|
|
generate_add_imm(a0, ((bit_count[reg_list] + 1) * 4)); \
|
2009-05-21 17:48:31 +02:00
|
|
|
generate_store_reg(a0, base_reg) \
|
|
|
|
|
|
|
|
#define thumb_block_address_postadjust_push_lr(base_reg) \
|
2021-11-04 20:22:42 +01:00
|
|
|
generate_store_reg(a0, base_reg) \
|
2009-05-21 17:48:31 +02:00
|
|
|
|
|
|
|
#define thumb_block_memory_extra_no() \
|
|
|
|
|
|
|
|
#define thumb_block_memory_extra_up() \
|
|
|
|
|
|
|
|
#define thumb_block_memory_extra_down() \
|
|
|
|
|
|
|
|
#define thumb_block_memory_extra_pop_pc() \
|
2021-11-04 20:22:42 +01:00
|
|
|
generate_load_reg(a0, REG_SAVE3); \
|
|
|
|
generate_add_imm(a0, (bit_count[reg_list] * 4)); \
|
2021-11-03 22:20:31 +01:00
|
|
|
generate_load_pc(a1, pc); \
|
|
|
|
generate_function_call(execute_load_u32); \
|
2009-05-21 17:48:31 +02:00
|
|
|
generate_store_reg(rv, REG_PC); \
|
|
|
|
generate_indirect_branch_cycle_update(thumb) \
|
|
|
|
|
|
|
|
#define thumb_block_memory_extra_push_lr(base_reg) \
|
2021-11-04 20:22:42 +01:00
|
|
|
generate_load_reg(a0, REG_SAVE3); \
|
|
|
|
generate_add_imm(a0, (bit_count[reg_list] * 4)); \
|
2009-05-21 17:48:31 +02:00
|
|
|
generate_load_reg(a1, REG_LR); \
|
2021-11-02 23:16:47 +01:00
|
|
|
generate_function_call(execute_store_aligned_u32) \
|
2009-05-21 17:48:31 +02:00
|
|
|
|
|
|
|
#define thumb_block_memory_load() \
|
2021-11-03 22:20:31 +01:00
|
|
|
generate_load_pc(a1, pc); \
|
|
|
|
generate_function_call(execute_load_u32); \
|
2009-05-21 17:48:31 +02:00
|
|
|
generate_store_reg(rv, i) \
|
|
|
|
|
|
|
|
#define thumb_block_memory_store() \
|
|
|
|
generate_load_reg(a1, i); \
|
2021-11-02 23:16:47 +01:00
|
|
|
generate_function_call(execute_store_aligned_u32) \
|
2009-05-21 17:48:31 +02:00
|
|
|
|
|
|
|
#define thumb_block_memory_final_load() \
|
|
|
|
thumb_block_memory_load() \
|
|
|
|
|
|
|
|
#define thumb_block_memory_final_store() \
|
|
|
|
generate_load_reg(a1, i); \
|
2021-11-02 20:50:41 +01:00
|
|
|
generate_store_reg_i32(pc + 2, REG_PC); \
|
2009-05-21 17:48:31 +02:00
|
|
|
generate_function_call(execute_store_u32) \
|
|
|
|
|
|
|
|
#define thumb_block_memory_final_no(access_type) \
|
|
|
|
thumb_block_memory_final_##access_type() \
|
|
|
|
|
|
|
|
#define thumb_block_memory_final_up(access_type) \
|
|
|
|
thumb_block_memory_final_##access_type() \
|
|
|
|
|
|
|
|
#define thumb_block_memory_final_down(access_type) \
|
|
|
|
thumb_block_memory_final_##access_type() \
|
|
|
|
|
|
|
|
#define thumb_block_memory_final_push_lr(access_type) \
|
|
|
|
thumb_block_memory_##access_type() \
|
|
|
|
|
|
|
|
#define thumb_block_memory_final_pop_pc(access_type) \
|
|
|
|
thumb_block_memory_##access_type() \
|
|
|
|
|
|
|
|
#define thumb_block_memory(access_type, pre_op, post_op, base_reg) \
|
|
|
|
{ \
|
|
|
|
thumb_decode_rlist(); \
|
|
|
|
u32 i; \
|
|
|
|
u32 offset = 0; \
|
|
|
|
\
|
2021-11-04 20:22:42 +01:00
|
|
|
generate_load_reg(a0, base_reg); \
|
|
|
|
generate_and_imm(a0, ~0x03); \
|
2009-05-21 17:48:31 +02:00
|
|
|
thumb_block_address_preadjust_##pre_op(); \
|
2021-11-04 20:22:42 +01:00
|
|
|
generate_store_reg(a0, REG_SAVE3); \
|
2009-05-21 17:48:31 +02:00
|
|
|
thumb_block_address_postadjust_##post_op(base_reg); \
|
|
|
|
\
|
|
|
|
for(i = 0; i < 8; i++) \
|
|
|
|
{ \
|
|
|
|
if((reg_list >> i) & 0x01) \
|
|
|
|
{ \
|
|
|
|
cycle_count++; \
|
2021-11-04 20:22:42 +01:00
|
|
|
generate_load_reg(a0, REG_SAVE3); \
|
|
|
|
generate_add_imm(a0, offset) \
|
2009-05-21 17:48:31 +02:00
|
|
|
if(reg_list & ~((2 << i) - 1)) \
|
|
|
|
{ \
|
|
|
|
thumb_block_memory_##access_type(); \
|
|
|
|
offset += 4; \
|
|
|
|
} \
|
|
|
|
else \
|
|
|
|
{ \
|
|
|
|
thumb_block_memory_final_##post_op(access_type); \
|
|
|
|
} \
|
|
|
|
} \
|
|
|
|
} \
|
|
|
|
\
|
|
|
|
thumb_block_memory_extra_##post_op(); \
|
|
|
|
} \
|
|
|
|
|
|
|
|
|
|
|
|
#define thumb_conditional_branch(condition) \
|
|
|
|
{ \
|
|
|
|
generate_cycle_update(); \
|
2021-09-17 19:18:50 +02:00
|
|
|
generate_condition_##condition(a0); \
|
2009-05-21 17:48:31 +02:00
|
|
|
generate_branch_no_cycle_update( \
|
|
|
|
block_exits[block_exit_position].branch_source, \
|
|
|
|
block_exits[block_exit_position].branch_target); \
|
|
|
|
generate_branch_patch_conditional(backpatch_address, translation_ptr); \
|
|
|
|
block_exit_position++; \
|
|
|
|
} \
|
|
|
|
|
|
|
|
|
2021-08-31 00:42:47 +02:00
|
|
|
// Execute functions
|
2009-05-21 17:48:31 +02:00
|
|
|
|
2021-08-31 00:42:47 +02:00
|
|
|
#define update_logical_flags() \
|
2021-12-13 18:31:01 +01:00
|
|
|
if (check_generate_z_flag) { \
|
|
|
|
generate_update_flag(z, REG_Z_FLAG) \
|
|
|
|
} \
|
|
|
|
if (check_generate_n_flag) { \
|
|
|
|
generate_update_flag(s, REG_N_FLAG) \
|
|
|
|
} \
|
2009-05-21 17:48:31 +02:00
|
|
|
|
2021-08-31 00:42:47 +02:00
|
|
|
#define update_add_flags() \
|
|
|
|
update_logical_flags() \
|
2021-12-13 18:31:01 +01:00
|
|
|
if (check_generate_c_flag) { \
|
|
|
|
generate_update_flag(c, REG_C_FLAG) \
|
|
|
|
} \
|
|
|
|
if (check_generate_v_flag) { \
|
|
|
|
generate_update_flag(o, REG_V_FLAG) \
|
|
|
|
} \
|
2009-05-21 17:48:31 +02:00
|
|
|
|
2021-08-31 00:42:47 +02:00
|
|
|
#define update_sub_flags() \
|
|
|
|
update_logical_flags() \
|
2021-12-13 18:31:01 +01:00
|
|
|
if (check_generate_c_flag) { \
|
|
|
|
generate_update_flag(nc, REG_C_FLAG) \
|
|
|
|
} \
|
|
|
|
if (check_generate_v_flag) { \
|
|
|
|
generate_update_flag(o, REG_V_FLAG) \
|
|
|
|
} \
|
2009-05-21 17:48:31 +02:00
|
|
|
|
2021-08-31 00:42:47 +02:00
|
|
|
#define arm_data_proc_and(rd, storefnc) \
|
|
|
|
generate_and(a0, a1); \
|
|
|
|
storefnc(a0, rd);
|
2009-05-21 17:48:31 +02:00
|
|
|
|
2021-08-31 00:42:47 +02:00
|
|
|
#define arm_data_proc_ands(rd, storefnc) \
|
2021-12-13 18:31:01 +01:00
|
|
|
generate_and(a0, a1); \
|
|
|
|
update_logical_flags(); \
|
|
|
|
storefnc(a0, rd);
|
2009-05-21 17:48:31 +02:00
|
|
|
|
2021-08-31 00:42:47 +02:00
|
|
|
#define arm_data_proc_eor(rd, storefnc) \
|
|
|
|
generate_xor(a0, a1); \
|
|
|
|
storefnc(a0, rd);
|
2009-05-21 17:48:31 +02:00
|
|
|
|
2021-08-31 00:42:47 +02:00
|
|
|
#define arm_data_proc_eors(rd, storefnc) \
|
2021-12-13 18:31:01 +01:00
|
|
|
generate_xor(a0, a1); \
|
|
|
|
update_logical_flags(); \
|
|
|
|
storefnc(a0, rd);
|
2009-05-21 17:48:31 +02:00
|
|
|
|
2021-08-31 00:42:47 +02:00
|
|
|
#define arm_data_proc_orr(rd, storefnc) \
|
|
|
|
generate_or(a0, a1); \
|
|
|
|
storefnc(a0, rd);
|
2009-05-21 17:48:31 +02:00
|
|
|
|
2021-08-31 00:42:47 +02:00
|
|
|
#define arm_data_proc_orrs(rd, storefnc) \
|
2021-12-13 18:31:01 +01:00
|
|
|
generate_or(a0, a1); \
|
|
|
|
update_logical_flags(); \
|
|
|
|
storefnc(a0, rd);
|
2021-08-31 00:42:47 +02:00
|
|
|
|
|
|
|
#define arm_data_proc_bic(rd, storefnc) \
|
|
|
|
generate_not(a0); \
|
|
|
|
generate_and(a0, a1); \
|
|
|
|
storefnc(a0, rd);
|
|
|
|
|
|
|
|
#define arm_data_proc_bics(rd, storefnc) \
|
2021-12-13 18:31:01 +01:00
|
|
|
generate_not(a0); \
|
|
|
|
generate_and(a0, a1); \
|
|
|
|
update_logical_flags(); \
|
|
|
|
storefnc(a0, rd);
|
2021-08-31 00:42:47 +02:00
|
|
|
|
|
|
|
#define arm_data_proc_add(rd, storefnc) \
|
|
|
|
generate_add(a0, a1); \
|
|
|
|
storefnc(a0, rd);
|
|
|
|
|
|
|
|
#define arm_data_proc_adds(rd, storefnc) \
|
2021-12-13 18:31:01 +01:00
|
|
|
generate_add(a0, a1); \
|
|
|
|
update_add_flags(); \
|
|
|
|
storefnc(a0, rd);
|
2021-08-31 00:42:47 +02:00
|
|
|
|
|
|
|
// Argument ordering is inverted between arm and x86
|
|
|
|
#define arm_data_proc_sub(rd, storefnc) \
|
|
|
|
generate_sub(a1, a0); \
|
|
|
|
storefnc(a1, rd);
|
|
|
|
|
|
|
|
#define arm_data_proc_rsb(rd, storefnc) \
|
|
|
|
generate_sub(a0, a1); \
|
|
|
|
storefnc(a0, rd);
|
|
|
|
|
|
|
|
// Borrow flag in ARM is opposite to carry flag in x86
|
|
|
|
#define arm_data_proc_subs(rd, storefnc) \
|
2021-12-13 18:31:01 +01:00
|
|
|
generate_sub(a1, a0); \
|
|
|
|
update_sub_flags(); \
|
|
|
|
storefnc(a1, rd);
|
2021-08-31 00:42:47 +02:00
|
|
|
|
|
|
|
#define arm_data_proc_rsbs(rd, storefnc) \
|
2021-12-13 18:31:01 +01:00
|
|
|
generate_sub(a0, a1); \
|
|
|
|
update_sub_flags(); \
|
|
|
|
storefnc(a0, rd);
|
2021-08-31 00:42:47 +02:00
|
|
|
|
|
|
|
#define arm_data_proc_mul(rd, storefnc) \
|
|
|
|
generate_multiply(a1); \
|
|
|
|
storefnc(a0, rd);
|
|
|
|
|
|
|
|
#define arm_data_proc_muls(rd, storefnc) \
|
2021-12-06 17:08:18 +01:00
|
|
|
generate_multiply(a1); \
|
|
|
|
generate_and(a0, a0); \
|
|
|
|
update_logical_flags(); \
|
|
|
|
storefnc(a0, rd);
|
2021-08-31 00:42:47 +02:00
|
|
|
|
2021-12-13 18:31:01 +01:00
|
|
|
#define load_c_flag(tmpreg) \
|
2021-08-31 00:42:47 +02:00
|
|
|
/* Loads the flag to the right value by adding it to ~0 causing carry */ \
|
2021-12-13 18:31:01 +01:00
|
|
|
generate_load_imm(tmpreg, 0xffffffff); \
|
|
|
|
generate_add_memreg(tmpreg, REG_C_FLAG); \
|
|
|
|
|
|
|
|
#define load_inv_c_flag(tmpreg) \
|
|
|
|
/* Loads the inverse C flag (for subtraction, since ARM's inverted) */ \
|
|
|
|
generate_load_reg(tmpreg, REG_C_FLAG); \
|
|
|
|
generate_sub_imm(tmpreg, 1); \
|
|
|
|
|
|
|
|
|
|
|
|
#define arm_data_proc_adc(rd, storefnc) \
|
|
|
|
load_c_flag(a2) \
|
2021-08-31 00:42:47 +02:00
|
|
|
generate_adc(a0, a1); \
|
|
|
|
storefnc(a0, rd);
|
|
|
|
|
|
|
|
#define arm_data_proc_adcs(rd, storefnc) \
|
2021-12-13 18:31:01 +01:00
|
|
|
load_c_flag(a2) \
|
|
|
|
generate_adc(a0, a1); \
|
|
|
|
update_add_flags() \
|
|
|
|
storefnc(a0, rd);
|
2021-08-31 00:42:47 +02:00
|
|
|
|
|
|
|
#define arm_data_proc_sbc(rd, storefnc) \
|
2021-12-13 18:31:01 +01:00
|
|
|
load_inv_c_flag(a2) \
|
2021-08-31 00:42:47 +02:00
|
|
|
generate_sbb(a1, a0); \
|
|
|
|
storefnc(a1, rd);
|
|
|
|
|
|
|
|
#define arm_data_proc_sbcs(rd, storefnc) \
|
2021-12-13 18:31:01 +01:00
|
|
|
load_inv_c_flag(a2) \
|
|
|
|
generate_sbb(a1, a0); \
|
|
|
|
update_sub_flags() \
|
|
|
|
storefnc(a1, rd);
|
2021-08-31 00:42:47 +02:00
|
|
|
|
|
|
|
#define arm_data_proc_rsc(rd, storefnc) \
|
2021-12-13 18:31:01 +01:00
|
|
|
load_inv_c_flag(a2) \
|
2021-08-31 00:42:47 +02:00
|
|
|
generate_sbb(a0, a1); \
|
|
|
|
storefnc(a0, rd);
|
|
|
|
|
|
|
|
#define arm_data_proc_rscs(rd, storefnc) \
|
2021-12-13 18:31:01 +01:00
|
|
|
load_inv_c_flag(a2) \
|
|
|
|
generate_sbb(a0, a1); \
|
|
|
|
update_sub_flags() \
|
|
|
|
storefnc(a0, rd);
|
2021-08-31 00:42:47 +02:00
|
|
|
|
|
|
|
|
|
|
|
#define arm_data_proc_test_cmp() \
|
|
|
|
generate_sub(a1, a0); \
|
|
|
|
update_sub_flags()
|
|
|
|
|
|
|
|
#define arm_data_proc_test_cmn() \
|
|
|
|
generate_add(a1, a0); \
|
|
|
|
update_add_flags()
|
|
|
|
|
|
|
|
#define arm_data_proc_test_tst() \
|
|
|
|
generate_and(a0, a1); \
|
|
|
|
update_logical_flags()
|
|
|
|
|
|
|
|
#define arm_data_proc_test_teq() \
|
|
|
|
generate_xor(a0, a1); \
|
|
|
|
update_logical_flags()
|
|
|
|
|
|
|
|
#define arm_data_proc_unary_mov(rd, storefnc) \
|
|
|
|
storefnc(a0, rd);
|
|
|
|
|
|
|
|
#define arm_data_proc_unary_movs(rd, storefnc) \
|
|
|
|
arm_data_proc_unary_mov(rd, storefnc); \
|
|
|
|
generate_or(a0, a0); \
|
|
|
|
update_logical_flags()
|
|
|
|
|
|
|
|
#define arm_data_proc_unary_mvn(rd, storefnc) \
|
|
|
|
generate_not(a0); \
|
|
|
|
storefnc(a0, rd);
|
|
|
|
|
|
|
|
#define arm_data_proc_unary_mvns(rd, storefnc) \
|
|
|
|
arm_data_proc_unary_mvn(rd, storefnc); \
|
|
|
|
/* NOT does not update the flag register */ \
|
|
|
|
generate_or(a0, a0); \
|
|
|
|
update_logical_flags()
|
|
|
|
|
|
|
|
#define arm_data_proc_unary_neg(rd, storefnc) \
|
|
|
|
generate_xor(a1, a1); \
|
|
|
|
arm_data_proc_subs(rd, storefnc)
|
2009-05-21 17:48:31 +02:00
|
|
|
|
|
|
|
|
2021-03-06 21:15:22 +01:00
|
|
|
static void function_cc execute_swi(u32 pc)
|
2009-05-21 17:48:31 +02:00
|
|
|
{
|
2021-09-17 21:30:33 +02:00
|
|
|
// Open bus value after SWI
|
2022-01-26 19:03:14 +01:00
|
|
|
reg[REG_BUS_VALUE] = 0xe3a02004;
|
2023-01-11 20:44:56 +01:00
|
|
|
REG_MODE(MODE_SUPERVISOR)[6] = pc;
|
|
|
|
REG_SPSR(MODE_SUPERVISOR) = reg[REG_CPSR];
|
2021-09-17 22:22:01 +02:00
|
|
|
// Move to ARM mode, supervisor mode, disable IRQs
|
|
|
|
reg[REG_CPSR] = (reg[REG_CPSR] & ~0x3F) | 0x13 | 0x80;
|
2009-05-21 17:48:31 +02:00
|
|
|
set_cpu_mode(MODE_SUPERVISOR);
|
|
|
|
}
|
|
|
|
|
|
|
|
#define arm_conditional_block_header() \
|
2021-09-17 19:18:50 +02:00
|
|
|
generate_cycle_update(); \
|
|
|
|
generate_condition(a0); \
|
2009-05-21 17:48:31 +02:00
|
|
|
|
|
|
|
#define arm_b() \
|
|
|
|
generate_branch() \
|
|
|
|
|
|
|
|
#define arm_bl() \
|
2021-11-02 22:00:09 +01:00
|
|
|
generate_load_pc(a0, (pc + 4)); \
|
2009-05-21 17:48:31 +02:00
|
|
|
generate_store_reg(a0, REG_LR); \
|
|
|
|
generate_branch() \
|
|
|
|
|
|
|
|
#define arm_bx() \
|
2014-12-10 01:17:37 +01:00
|
|
|
arm_decode_branchx(opcode); \
|
2009-05-21 17:48:31 +02:00
|
|
|
generate_load_reg(a0, rn); \
|
|
|
|
generate_indirect_branch_dual(); \
|
|
|
|
|
|
|
|
#define arm_swi() \
|
2021-09-17 20:00:37 +02:00
|
|
|
collapse_flags(a0, a1); \
|
2021-11-06 12:17:50 +01:00
|
|
|
generate_load_pc(arg0, (pc + 4)); \
|
2009-05-21 17:48:31 +02:00
|
|
|
generate_function_call(execute_swi); \
|
|
|
|
generate_branch() \
|
|
|
|
|
|
|
|
#define thumb_b() \
|
|
|
|
generate_branch_cycle_update( \
|
|
|
|
block_exits[block_exit_position].branch_source, \
|
|
|
|
block_exits[block_exit_position].branch_target); \
|
|
|
|
block_exit_position++ \
|
|
|
|
|
|
|
|
#define thumb_bl() \
|
2021-11-02 22:00:09 +01:00
|
|
|
generate_load_pc(a0, ((pc + 2) | 0x01)); \
|
2009-05-21 17:48:31 +02:00
|
|
|
generate_store_reg(a0, REG_LR); \
|
|
|
|
generate_branch_cycle_update( \
|
|
|
|
block_exits[block_exit_position].branch_source, \
|
|
|
|
block_exits[block_exit_position].branch_target); \
|
|
|
|
block_exit_position++ \
|
|
|
|
|
|
|
|
#define thumb_blh() \
|
|
|
|
{ \
|
|
|
|
thumb_decode_branch(); \
|
2021-11-02 22:00:09 +01:00
|
|
|
generate_load_pc(a0, ((pc + 2) | 0x01)); \
|
2009-05-21 17:48:31 +02:00
|
|
|
generate_load_reg(a1, REG_LR); \
|
|
|
|
generate_store_reg(a0, REG_LR); \
|
|
|
|
generate_mov(a0, a1); \
|
|
|
|
generate_add_imm(a0, (offset * 2)); \
|
|
|
|
generate_indirect_branch_cycle_update(thumb); \
|
|
|
|
} \
|
|
|
|
|
|
|
|
#define thumb_bx() \
|
|
|
|
{ \
|
|
|
|
thumb_decode_hireg_op(); \
|
|
|
|
generate_load_reg_pc(a0, rs, 4); \
|
|
|
|
generate_indirect_branch_cycle_update(dual); \
|
|
|
|
} \
|
|
|
|
|
2021-05-05 02:20:00 +02:00
|
|
|
#define thumb_process_cheats() \
|
|
|
|
generate_function_call(process_cheats);
|
|
|
|
|
|
|
|
#define arm_process_cheats() \
|
|
|
|
generate_function_call(process_cheats);
|
|
|
|
|
2009-05-21 17:48:31 +02:00
|
|
|
#define thumb_swi() \
|
2021-09-17 20:00:37 +02:00
|
|
|
collapse_flags(a0, a1); \
|
2021-11-06 12:17:50 +01:00
|
|
|
generate_load_pc(arg0, (pc + 2)); \
|
2009-05-21 17:48:31 +02:00
|
|
|
generate_function_call(execute_swi); \
|
|
|
|
generate_branch_cycle_update( \
|
|
|
|
block_exits[block_exit_position].branch_source, \
|
|
|
|
block_exits[block_exit_position].branch_target); \
|
|
|
|
block_exit_position++ \
|
|
|
|
|
2021-09-03 01:01:37 +02:00
|
|
|
#define arm_hle_div(cpu_mode) \
|
2021-09-16 19:30:30 +02:00
|
|
|
{ \
|
|
|
|
u8 *jmpinst; \
|
2021-09-03 01:01:37 +02:00
|
|
|
generate_load_reg(a0, 0); \
|
|
|
|
generate_load_reg(a2, 1); \
|
2021-09-16 19:30:30 +02:00
|
|
|
generate_cmp_imm(a2, 0); \
|
|
|
|
x86_emit_j_filler(x86_condition_code_z, jmpinst); \
|
2021-09-03 01:01:37 +02:00
|
|
|
x86_emit_cdq(); \
|
|
|
|
x86_emit_idiv_eax_reg(ecx); \
|
|
|
|
generate_store_reg(a0, 0); \
|
|
|
|
generate_store_reg(a1, 1); \
|
|
|
|
generate_mov(a1, a0); \
|
2021-12-13 22:26:49 +01:00
|
|
|
generate_shift_right_arithmetic(a1, 31); \
|
|
|
|
generate_xor(a0, a1); \
|
|
|
|
generate_sub(a0, a1); \
|
2021-09-03 01:01:37 +02:00
|
|
|
generate_store_reg(a0, 3); \
|
2021-09-16 19:30:30 +02:00
|
|
|
generate_branch_patch_conditional(jmpinst, translation_ptr); \
|
|
|
|
}
|
2021-09-03 01:01:37 +02:00
|
|
|
|
|
|
|
#define arm_hle_div_arm(cpu_mode) \
|
2021-09-16 19:30:30 +02:00
|
|
|
{ \
|
|
|
|
u8 *jmpinst; \
|
2021-09-03 01:01:37 +02:00
|
|
|
generate_load_reg(a0, 1); \
|
|
|
|
generate_load_reg(a2, 0); \
|
2021-09-16 19:30:30 +02:00
|
|
|
generate_cmp_imm(a2, 0); \
|
|
|
|
x86_emit_j_filler(x86_condition_code_z, jmpinst); \
|
2021-09-03 01:01:37 +02:00
|
|
|
x86_emit_cdq(); \
|
|
|
|
x86_emit_idiv_eax_reg(ecx); \
|
|
|
|
generate_store_reg(a0, 0); \
|
|
|
|
generate_store_reg(a1, 1); \
|
|
|
|
generate_mov(a1, a0); \
|
2021-12-13 22:26:49 +01:00
|
|
|
generate_shift_right_arithmetic(a1, 31); \
|
|
|
|
generate_xor(a0, a1); \
|
|
|
|
generate_sub(a0, a1); \
|
2021-09-03 01:01:37 +02:00
|
|
|
generate_store_reg(a0, 3); \
|
2021-09-16 19:30:30 +02:00
|
|
|
generate_branch_patch_conditional(jmpinst, translation_ptr); \
|
|
|
|
}
|
2009-05-21 17:48:31 +02:00
|
|
|
|
|
|
|
#define generate_translation_gate(type) \
|
2021-11-02 22:00:09 +01:00
|
|
|
generate_load_pc(a0, pc); \
|
2009-05-21 17:48:31 +02:00
|
|
|
generate_indirect_branch_no_cycle_update(type) \
|
|
|
|
|
2021-11-06 12:17:50 +01:00
|
|
|
extern void* x86_table_data[9][16];
|
|
|
|
extern void* x86_table_info[9][16];
|
2021-07-28 19:12:43 +02:00
|
|
|
|
|
|
|
void init_emitter(void) {
|
|
|
|
memcpy(x86_table_info, x86_table_data, sizeof(x86_table_data));
|
2021-09-10 00:30:55 +02:00
|
|
|
|
2021-10-30 22:54:51 +02:00
|
|
|
rom_cache_watermark = INITIAL_ROM_WATERMARK;
|
2021-09-10 00:30:55 +02:00
|
|
|
init_bios_hooks();
|
2021-07-28 19:12:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
u32 function_cc execute_arm_translate_internal(u32 cycles, void *regptr);
|
|
|
|
|
2021-09-03 01:01:37 +02:00
|
|
|
u32 execute_arm_translate(u32 cycles) {
|
2021-07-28 19:12:43 +02:00
|
|
|
return execute_arm_translate_internal(cycles, ®[0]);
|
|
|
|
}
|
2021-03-16 22:57:45 +01:00
|
|
|
|
2009-05-21 17:48:31 +02:00
|
|
|
#endif
|