mupen64plus-oldsvn/r4300/x86/gr4300.c

2067 lines
54 KiB
C

/**
* Mupen64 - gr4300.c
* Copyright (C) 2002 Hacktarux
*
* Mupen64 homepage: http://mupen64.emulation64.com
* email address: hacktarux@yahoo.fr
*
* If you want to contribute to the project please contact
* me first (maybe someone is already making what you are
* planning to do).
*
*
* This program is free software; you can redistribute it and/
* or modify it under the terms of the GNU General Public Li-
* cence as published by the Free Software Foundation; either
* version 2 of the Licence, or any later version.
*
* This program is distributed in the hope that it will be use-
* ful, but WITHOUT ANY WARRANTY; without even the implied war-
* ranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
* See the GNU General Public Licence for more details.
*
* You should have received a copy of the GNU General Public
* Licence along with this program; if not, write to the Free
* Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139,
* USA.
*
**/
#include "assemble.h"
#include "../r4300.h"
#include "../macros.h"
#include "../../memory/memory.h"
#include "../interupt.h"
#include "../ops.h"
#include "../recomph.h"
#include "regcache.h"
#include "../exception.h"
#include "interpret.h"
extern unsigned int op;
precomp_instr fake_instr;
static int eax, ebx, ecx, edx, esp, ebp, esi, edi;
int branch_taken;
void gennotcompiled()
{
free_all_registers();
simplify_access();
if (dst->addr == 0xa4000040)
{
sub_reg32_imm32(ESP, 0xC);
mov_m32_reg32((unsigned int*)(&return_address), ESP);
sub_m32_imm32((unsigned int*)(&return_address), 4);
}
mov_m32_imm32((unsigned int*)(&PC), (unsigned int)(dst));
mov_reg32_imm32(EAX, (unsigned int)NOTCOMPILED);
call_reg32(EAX);
}
void genlink_subblock()
{
free_all_registers();
jmp(dst->addr+4);
}
void gendebug()
{
free_all_registers();
mov_m32_reg32((unsigned int*)&eax, EAX);
mov_m32_reg32((unsigned int*)&ebx, EBX);
mov_m32_reg32((unsigned int*)&ecx, ECX);
mov_m32_reg32((unsigned int*)&edx, EDX);
mov_m32_reg32((unsigned int*)&esp, ESP);
mov_m32_reg32((unsigned int*)&ebp, EBP);
mov_m32_reg32((unsigned int*)&esi, ESI);
mov_m32_reg32((unsigned int*)&edi, EDI);
mov_m32_imm32((unsigned int*)(&PC), (unsigned int)(dst));
mov_m32_imm32((unsigned int*)(&op), (unsigned int)(src));
mov_reg32_imm32(EAX, (unsigned int)debug);
call_reg32(EAX);
mov_reg32_m32(EAX, (unsigned int*)&eax);
mov_reg32_m32(EBX, (unsigned int*)&ebx);
mov_reg32_m32(ECX, (unsigned int*)&ecx);
mov_reg32_m32(EDX, (unsigned int*)&edx);
mov_reg32_m32(ESP, (unsigned int*)&esp);
mov_reg32_m32(EBP, (unsigned int*)&ebp);
mov_reg32_m32(ESI, (unsigned int*)&esi);
mov_reg32_m32(EDI, (unsigned int*)&edi);
}
void gencallinterp(unsigned long addr, int jump)
{
free_all_registers();
simplify_access();
if (jump)
mov_m32_imm32((unsigned int*)(&dyna_interp), 1);
mov_m32_imm32((unsigned int*)(&PC), (unsigned int)(dst));
mov_reg32_imm32(EAX, addr);
call_reg32(EAX);
if (jump)
{
mov_m32_imm32((unsigned int*)(&dyna_interp), 0);
mov_reg32_imm32(EAX, (unsigned int)dyna_jump);
call_reg32(EAX);
}
}
void genupdate_count(unsigned int addr)
{
#ifndef COMPARE_CORE
#ifndef DBG
mov_reg32_imm32(EAX, addr);
sub_reg32_m32(EAX, (unsigned int*)(&last_addr));
shr_reg32_imm8(EAX, 1);
add_m32_reg32((unsigned int*)(&Count), EAX);
#else
mov_m32_imm32((unsigned int*)(&PC), (unsigned int)(dst+1));
mov_reg32_imm32(EAX, (unsigned int)update_count);
call_reg32(EAX);
#endif
#else
mov_m32_imm32((unsigned int*)(&PC), (unsigned int)(dst+1));
mov_reg32_imm32(EAX, (unsigned int)update_count);
call_reg32(EAX);
#endif
}
void gendelayslot()
{
mov_m32_imm32((void*)(&delay_slot), 1);
recompile_opcode();
free_all_registers();
genupdate_count(dst->addr+4);
mov_m32_imm32((void*)(&delay_slot), 0);
}
void genni()
{
#ifdef EMU64_DEBUG
gencallinterp((unsigned int)NI, 0);
#endif
}
void genreserved()
{
#ifdef EMU64_DEBUG
gencallinterp((unsigned int)RESERVED, 0);
#endif
}
void genfin_block()
{
gencallinterp((unsigned int)FIN_BLOCK, 0);
}
void gencheck_interupt(unsigned int instr_structure)
{
mov_eax_memoffs32((void*)(&next_interupt));
cmp_reg32_m32(EAX, (void*)&Count);
ja_rj(17);
mov_m32_imm32((unsigned int*)(&PC), instr_structure); // 10
mov_reg32_imm32(EAX, (unsigned int)gen_interupt); // 5
call_reg32(EAX); // 2
}
void gencheck_interupt_out(unsigned int addr)
{
mov_eax_memoffs32((void*)(&next_interupt));
cmp_reg32_m32(EAX, (void*)&Count);
ja_rj(27);
mov_m32_imm32((unsigned int*)(&fake_instr.addr), addr);
mov_m32_imm32((unsigned int*)(&PC), (unsigned int)(&fake_instr));
mov_reg32_imm32(EAX, (unsigned int)gen_interupt);
call_reg32(EAX);
}
void gencheck_interupt_reg() // addr is in EAX
{
mov_reg32_m32(EBX, (void*)&next_interupt);
cmp_reg32_m32(EBX, (void*)&Count);
ja_rj(22);
mov_memoffs32_eax((unsigned int*)(&fake_instr.addr)); // 5
mov_m32_imm32((unsigned int*)(&PC), (unsigned int)(&fake_instr)); // 10
mov_reg32_imm32(EAX, (unsigned int)gen_interupt); // 5
call_reg32(EAX); // 2
}
void gennop()
{
}
void genj()
{
#ifdef INTERPRET_J
gencallinterp((unsigned int)J, 1);
#else
unsigned int naddr;
if (((dst->addr & 0xFFF) == 0xFFC &&
(dst->addr < 0x80000000 || dst->addr >= 0xC0000000))||no_compiled_jump)
{
gencallinterp((unsigned int)J, 1);
return;
}
gendelayslot();
naddr = ((dst-1)->f.j.inst_index<<2) | (dst->addr & 0xF0000000);
mov_m32_imm32((void*)(&last_addr), naddr);
gencheck_interupt((unsigned int)&actual->block[(naddr-actual->start)/4]);
jmp(naddr);
#endif
}
void genj_out()
{
#ifdef INTERPRET_J_OUT
gencallinterp((unsigned int)J_OUT, 1);
#else
unsigned int naddr;
if (((dst->addr & 0xFFF) == 0xFFC &&
(dst->addr < 0x80000000 || dst->addr >= 0xC0000000))||no_compiled_jump)
{
gencallinterp((unsigned int)J_OUT, 1);
return;
}
gendelayslot();
naddr = ((dst-1)->f.j.inst_index<<2) | (dst->addr & 0xF0000000);
mov_m32_imm32((void*)(&last_addr), naddr);
gencheck_interupt_out(naddr);
mov_m32_imm32(&jump_to_address, naddr);
mov_m32_imm32((unsigned int*)(&PC), (unsigned int)(dst+1));
mov_reg32_imm32(EAX, (unsigned int)jump_to_func);
call_reg32(EAX);
#endif
}
void genj_idle()
{
#ifdef INTERPRET_J_IDLE
gencallinterp((unsigned int)J_IDLE, 1);
#else
if (((dst->addr & 0xFFF) == 0xFFC &&
(dst->addr < 0x80000000 || dst->addr >= 0xC0000000))||no_compiled_jump)
{
gencallinterp((unsigned int)J_IDLE, 1);
return;
}
mov_eax_memoffs32((unsigned int *)(&next_interupt));
sub_reg32_m32(EAX, (unsigned int *)(&Count));
cmp_reg32_imm8(EAX, 3);
jbe_rj(11);
and_eax_imm32(0xFFFFFFFC);
add_m32_reg32((unsigned int *)(&Count), EAX);
genj();
#endif
}
void genjal()
{
#ifdef INTERPRET_JAL
gencallinterp((unsigned int)JAL, 1);
#else
unsigned int naddr;
if (((dst->addr & 0xFFF) == 0xFFC &&
(dst->addr < 0x80000000 || dst->addr >= 0xC0000000))||no_compiled_jump)
{
gencallinterp((unsigned int)JAL, 1);
return;
}
gendelayslot();
mov_m32_imm32((unsigned int *)(reg + 31), dst->addr + 4);
if (((dst->addr + 4) & 0x80000000))
mov_m32_imm32((unsigned int *)(&reg[31])+1, 0xFFFFFFFF);
else
mov_m32_imm32((unsigned int *)(&reg[31])+1, 0);
naddr = ((dst-1)->f.j.inst_index<<2) | (dst->addr & 0xF0000000);
mov_m32_imm32((void*)(&last_addr), naddr);
gencheck_interupt((unsigned int)&actual->block[(naddr-actual->start)/4]);
jmp(naddr);
#endif
}
void genjal_out()
{
#ifdef INTERPRET_JAL_OUT
gencallinterp((unsigned int)JAL_OUT, 1);
#else
unsigned int naddr;
if (((dst->addr & 0xFFF) == 0xFFC &&
(dst->addr < 0x80000000 || dst->addr >= 0xC0000000))||no_compiled_jump)
{
gencallinterp((unsigned int)JAL_OUT, 1);
return;
}
gendelayslot();
mov_m32_imm32((unsigned int *)(reg + 31), dst->addr + 4);
if (((dst->addr + 4) & 0x80000000))
mov_m32_imm32((unsigned int *)(&reg[31])+1, 0xFFFFFFFF);
else
mov_m32_imm32((unsigned int *)(&reg[31])+1, 0);
naddr = ((dst-1)->f.j.inst_index<<2) | (dst->addr & 0xF0000000);
mov_m32_imm32((void*)(&last_addr), naddr);
gencheck_interupt_out(naddr);
mov_m32_imm32(&jump_to_address, naddr);
mov_m32_imm32((unsigned int*)(&PC), (unsigned int)(dst+1));
mov_reg32_imm32(EAX, (unsigned int)jump_to_func);
call_reg32(EAX);
#endif
}
void genjal_idle()
{
#ifdef INTERPRET_JAL_IDLE
gencallinterp((unsigned int)JAL_IDLE, 1);
#else
if (((dst->addr & 0xFFF) == 0xFFC &&
(dst->addr < 0x80000000 || dst->addr >= 0xC0000000))||no_compiled_jump)
{
gencallinterp((unsigned int)JAL_IDLE, 1);
return;
}
mov_eax_memoffs32((unsigned int *)(&next_interupt));
sub_reg32_m32(EAX, (unsigned int *)(&Count));
cmp_reg32_imm8(EAX, 3);
jbe_rj(11);
and_eax_imm32(0xFFFFFFFC);
add_m32_reg32((unsigned int *)(&Count), EAX);
genjal();
#endif
}
void genbeq_test()
{
int rs_64bit = is64((unsigned int *)dst->f.i.rs);
int rt_64bit = is64((unsigned int *)dst->f.i.rt);
if (!rs_64bit && !rt_64bit)
{
int rs = allocate_register((unsigned int *)dst->f.i.rs);
int rt = allocate_register((unsigned int *)dst->f.i.rt);
cmp_reg32_reg32(rs, rt);
jne_rj(12);
mov_m32_imm32((unsigned int *)(&branch_taken), 1); // 10
jmp_imm_short(10); // 2
mov_m32_imm32((unsigned int *)(&branch_taken), 0); // 10
}
else if (rs_64bit == -1)
{
int rt1 = allocate_64_register1((unsigned int *)dst->f.i.rt);
int rt2 = allocate_64_register2((unsigned int *)dst->f.i.rt);
cmp_reg32_m32(rt1, (unsigned int *)dst->f.i.rs);
jne_rj(20);
cmp_reg32_m32(rt2, ((unsigned int *)dst->f.i.rs)+1); // 6
jne_rj(12); // 2
mov_m32_imm32((unsigned int *)(&branch_taken), 1); // 10
jmp_imm_short(10); // 2
mov_m32_imm32((unsigned int *)(&branch_taken), 0); // 10
}
else if (rt_64bit == -1)
{
int rs1 = allocate_64_register1((unsigned int *)dst->f.i.rs);
int rs2 = allocate_64_register2((unsigned int *)dst->f.i.rs);
cmp_reg32_m32(rs1, (unsigned int *)dst->f.i.rt);
jne_rj(20);
cmp_reg32_m32(rs2, ((unsigned int *)dst->f.i.rt)+1); // 6
jne_rj(12); // 2
mov_m32_imm32((unsigned int *)(&branch_taken), 1); // 10
jmp_imm_short(10); // 2
mov_m32_imm32((unsigned int *)(&branch_taken), 0); // 10
}
else
{
int rs1, rs2, rt1, rt2;
if (!rs_64bit)
{
rt1 = allocate_64_register1((unsigned int *)dst->f.i.rt);
rt2 = allocate_64_register2((unsigned int *)dst->f.i.rt);
rs1 = allocate_64_register1((unsigned int *)dst->f.i.rs);
rs2 = allocate_64_register2((unsigned int *)dst->f.i.rs);
}
else
{
rs1 = allocate_64_register1((unsigned int *)dst->f.i.rs);
rs2 = allocate_64_register2((unsigned int *)dst->f.i.rs);
rt1 = allocate_64_register1((unsigned int *)dst->f.i.rt);
rt2 = allocate_64_register2((unsigned int *)dst->f.i.rt);
}
cmp_reg32_reg32(rs1, rt1);
jne_rj(16);
cmp_reg32_reg32(rs2, rt2); // 2
jne_rj(12); // 2
mov_m32_imm32((unsigned int *)(&branch_taken), 1); // 10
jmp_imm_short(10); // 2
mov_m32_imm32((unsigned int *)(&branch_taken), 0); // 10
}
}
void gentest()
{
unsigned int temp, temp2;
cmp_m32_imm32((unsigned int *)(&branch_taken), 0);
je_near_rj(0);
temp = code_length;
mov_m32_imm32((void*)(&last_addr), dst->addr + (dst-1)->f.i.immediate*4);
gencheck_interupt((unsigned int)(dst + (dst-1)->f.i.immediate));
jmp(dst->addr + (dst-1)->f.i.immediate*4);
temp2 = code_length;
code_length = temp-4;
put32(temp2 - temp);
code_length = temp2;
mov_m32_imm32((void*)(&last_addr), dst->addr + 4);
gencheck_interupt((unsigned int)(dst + 1));
jmp(dst->addr + 4);
}
void genbeq()
{
#ifdef INTERPRET_BEQ
gencallinterp((unsigned int)BEQ, 1);
#else
if (((dst->addr & 0xFFF) == 0xFFC &&
(dst->addr < 0x80000000 || dst->addr >= 0xC0000000))||no_compiled_jump)
{
gencallinterp((unsigned int)BEQ, 1);
return;
}
genbeq_test();
gendelayslot();
gentest();
#endif
}
void gentest_out()
{
unsigned int temp, temp2;
cmp_m32_imm32((unsigned int *)(&branch_taken), 0);
je_near_rj(0);
temp = code_length;
mov_m32_imm32((void*)(&last_addr), dst->addr + (dst-1)->f.i.immediate*4);
gencheck_interupt_out(dst->addr + (dst-1)->f.i.immediate*4);
mov_m32_imm32(&jump_to_address, dst->addr + (dst-1)->f.i.immediate*4);
mov_m32_imm32((unsigned int*)(&PC), (unsigned int)(dst+1));
mov_reg32_imm32(EAX, (unsigned int)jump_to_func);
call_reg32(EAX);
temp2 = code_length;
code_length = temp-4;
put32(temp2 - temp);
code_length = temp2;
mov_m32_imm32((void*)(&last_addr), dst->addr + 4);
gencheck_interupt((unsigned int)(dst + 1));
jmp(dst->addr + 4);
}
void genbeq_out()
{
#ifdef INTERPRET_BEQ_OUT
gencallinterp((unsigned int)BEQ_OUT, 1);
#else
if (((dst->addr & 0xFFF) == 0xFFC &&
(dst->addr < 0x80000000 || dst->addr >= 0xC0000000))||no_compiled_jump)
{
gencallinterp((unsigned int)BEQ_OUT, 1);
return;
}
genbeq_test();
gendelayslot();
gentest_out();
#endif
}
void gentest_idle()
{
unsigned int temp, temp2;
int reg;
reg = lru_register();
free_register(reg);
cmp_m32_imm32((unsigned int *)(&branch_taken), 0);
je_near_rj(0);
temp = code_length;
mov_reg32_m32(reg, (unsigned int *)(&next_interupt));
sub_reg32_m32(reg, (unsigned int *)(&Count));
cmp_reg32_imm8(reg, 3);
jbe_rj(12);
and_reg32_imm32(reg, 0xFFFFFFFC); // 6
add_m32_reg32((unsigned int *)(&Count), reg); // 6
temp2 = code_length;
code_length = temp-4;
put32(temp2 - temp);
code_length = temp2;
}
void genbeq_idle()
{
#ifdef INTERPRET_BEQ_IDLE
gencallinterp((unsigned int)BEQ_IDLE, 1);
#else
if (((dst->addr & 0xFFF) == 0xFFC &&
(dst->addr < 0x80000000 || dst->addr >= 0xC0000000))||no_compiled_jump)
{
gencallinterp((unsigned int)BEQ_IDLE, 1);
return;
}
genbeq_test();
gentest_idle();
genbeq();
#endif
}
void genbne_test()
{
int rs_64bit = is64((unsigned int *)dst->f.i.rs);
int rt_64bit = is64((unsigned int *)dst->f.i.rt);
if (!rs_64bit && !rt_64bit)
{
int rs = allocate_register((unsigned int *)dst->f.i.rs);
int rt = allocate_register((unsigned int *)dst->f.i.rt);
cmp_reg32_reg32(rs, rt);
je_rj(12);
mov_m32_imm32((unsigned int *)(&branch_taken), 1); // 10
jmp_imm_short(10); // 2
mov_m32_imm32((unsigned int *)(&branch_taken), 0); // 10
}
else if (rs_64bit == -1)
{
int rt1 = allocate_64_register1((unsigned int *)dst->f.i.rt);
int rt2 = allocate_64_register2((unsigned int *)dst->f.i.rt);
cmp_reg32_m32(rt1, (unsigned int *)dst->f.i.rs);
jne_rj(20);
cmp_reg32_m32(rt2, ((unsigned int *)dst->f.i.rs)+1); // 6
jne_rj(12); // 2
mov_m32_imm32((unsigned int *)(&branch_taken), 0); // 10
jmp_imm_short(10); // 2
mov_m32_imm32((unsigned int *)(&branch_taken), 1); // 10
}
else if (rt_64bit == -1)
{
int rs1 = allocate_64_register1((unsigned int *)dst->f.i.rs);
int rs2 = allocate_64_register2((unsigned int *)dst->f.i.rs);
cmp_reg32_m32(rs1, (unsigned int *)dst->f.i.rt);
jne_rj(20);
cmp_reg32_m32(rs2, ((unsigned int *)dst->f.i.rt)+1); // 6
jne_rj(12); // 2
mov_m32_imm32((unsigned int *)(&branch_taken), 0); // 10
jmp_imm_short(10); // 2
mov_m32_imm32((unsigned int *)(&branch_taken), 1); // 10
}
else
{
int rs1, rs2, rt1, rt2;
if (!rs_64bit)
{
rt1 = allocate_64_register1((unsigned int *)dst->f.i.rt);
rt2 = allocate_64_register2((unsigned int *)dst->f.i.rt);
rs1 = allocate_64_register1((unsigned int *)dst->f.i.rs);
rs2 = allocate_64_register2((unsigned int *)dst->f.i.rs);
}
else
{
rs1 = allocate_64_register1((unsigned int *)dst->f.i.rs);
rs2 = allocate_64_register2((unsigned int *)dst->f.i.rs);
rt1 = allocate_64_register1((unsigned int *)dst->f.i.rt);
rt2 = allocate_64_register2((unsigned int *)dst->f.i.rt);
}
cmp_reg32_reg32(rs1, rt1);
jne_rj(16);
cmp_reg32_reg32(rs2, rt2); // 2
jne_rj(12); // 2
mov_m32_imm32((unsigned int *)(&branch_taken), 0); // 10
jmp_imm_short(10); // 2
mov_m32_imm32((unsigned int *)(&branch_taken), 1); // 10
}
}
void genbne()
{
#ifdef INTERPRET_BNE
gencallinterp((unsigned int)BNE, 1);
#else
if (((dst->addr & 0xFFF) == 0xFFC &&
(dst->addr < 0x80000000 || dst->addr >= 0xC0000000))||no_compiled_jump)
{
gencallinterp((unsigned int)BNE, 1);
return;
}
genbne_test();
gendelayslot();
gentest();
#endif
}
void genbne_out()
{
#ifdef INTERPRET_BNE_OUT
gencallinterp((unsigned int)BNE_OUT, 1);
#else
if (((dst->addr & 0xFFF) == 0xFFC &&
(dst->addr < 0x80000000 || dst->addr >= 0xC0000000))||no_compiled_jump)
{
gencallinterp((unsigned int)BNE_OUT, 1);
return;
}
genbne_test();
gendelayslot();
gentest_out();
#endif
}
void genbne_idle()
{
#ifdef INTERPRET_BNE_IDLE
gencallinterp((unsigned int)BNE_IDLE, 1);
#else
if (((dst->addr & 0xFFF) == 0xFFC &&
(dst->addr < 0x80000000 || dst->addr >= 0xC0000000))||no_compiled_jump)
{
gencallinterp((unsigned int)BNE_IDLE, 1);
return;
}
genbne_test();
gentest_idle();
genbne();
#endif
}
void genblez_test()
{
int rs_64bit = is64((unsigned int *)dst->f.i.rs);
if (!rs_64bit)
{
int rs = allocate_register((unsigned int *)dst->f.i.rs);
cmp_reg32_imm32(rs, 0);
jg_rj(12);
mov_m32_imm32((unsigned int *)(&branch_taken), 1); // 10
jmp_imm_short(10); // 2
mov_m32_imm32((unsigned int *)(&branch_taken), 0); // 10
}
else if (rs_64bit == -1)
{
cmp_m32_imm32(((unsigned int *)dst->f.i.rs)+1, 0);
jg_rj(14);
jne_rj(24); // 2
cmp_m32_imm32((unsigned int *)dst->f.i.rs, 0); // 10
je_rj(12); // 2
mov_m32_imm32((unsigned int *)(&branch_taken), 0); // 10
jmp_imm_short(10); // 2
mov_m32_imm32((unsigned int *)(&branch_taken), 1); // 10
}
else
{
int rs1 = allocate_64_register1((unsigned int *)dst->f.i.rs);
int rs2 = allocate_64_register2((unsigned int *)dst->f.i.rs);
cmp_reg32_imm32(rs2, 0);
jg_rj(10);
jne_rj(20); // 2
cmp_reg32_imm32(rs1, 0); // 6
je_rj(12); // 2
mov_m32_imm32((unsigned int *)(&branch_taken), 0); // 10
jmp_imm_short(10); // 2
mov_m32_imm32((unsigned int *)(&branch_taken), 1); // 10
}
}
void genblez()
{
#ifdef INTERPRET_BLEZ
gencallinterp((unsigned int)BLEZ, 1);
#else
if (((dst->addr & 0xFFF) == 0xFFC &&
(dst->addr < 0x80000000 || dst->addr >= 0xC0000000))||no_compiled_jump)
{
gencallinterp((unsigned int)BLEZ, 1);
return;
}
genblez_test();
gendelayslot();
gentest();
#endif
}
void genblez_out()
{
#ifdef INTERPRET_BLEZ_OUT
gencallinterp((unsigned int)BLEZ_OUT, 1);
#else
if (((dst->addr & 0xFFF) == 0xFFC &&
(dst->addr < 0x80000000 || dst->addr >= 0xC0000000))||no_compiled_jump)
{
gencallinterp((unsigned int)BLEZ_OUT, 1);
return;
}
genblez_test();
gendelayslot();
gentest_out();
#endif
}
void genblez_idle()
{
#ifdef INTERPRET_BLEZ_IDLE
gencallinterp((unsigned int)BLEZ_IDLE, 1);
#else
if (((dst->addr & 0xFFF) == 0xFFC &&
(dst->addr < 0x80000000 || dst->addr >= 0xC0000000))||no_compiled_jump)
{
gencallinterp((unsigned int)BLEZ_IDLE, 1);
return;
}
genblez_test();
gentest_idle();
genblez();
#endif
}
void genbgtz_test()
{
int rs_64bit = is64((unsigned int *)dst->f.i.rs);
if (!rs_64bit)
{
int rs = allocate_register((unsigned int *)dst->f.i.rs);
cmp_reg32_imm32(rs, 0);
jle_rj(12);
mov_m32_imm32((unsigned int *)(&branch_taken), 1); // 10
jmp_imm_short(10); // 2
mov_m32_imm32((unsigned int *)(&branch_taken), 0); // 10
}
else if (rs_64bit == -1)
{
cmp_m32_imm32(((unsigned int *)dst->f.i.rs)+1, 0);
jl_rj(14);
jne_rj(24); // 2
cmp_m32_imm32((unsigned int *)dst->f.i.rs, 0); // 10
jne_rj(12); // 2
mov_m32_imm32((unsigned int *)(&branch_taken), 0); // 10
jmp_imm_short(10); // 2
mov_m32_imm32((unsigned int *)(&branch_taken), 1); // 10
}
else
{
int rs1 = allocate_64_register1((unsigned int *)dst->f.i.rs);
int rs2 = allocate_64_register2((unsigned int *)dst->f.i.rs);
cmp_reg32_imm32(rs2, 0);
jl_rj(10);
jne_rj(20); // 2
cmp_reg32_imm32(rs1, 0); // 6
jne_rj(12); // 2
mov_m32_imm32((unsigned int *)(&branch_taken), 0); // 10
jmp_imm_short(10); // 2
mov_m32_imm32((unsigned int *)(&branch_taken), 1); // 10
}
}
void genbgtz()
{
#ifdef INTERPRET_BGTZ
gencallinterp((unsigned int)BGTZ, 1);
#else
if (((dst->addr & 0xFFF) == 0xFFC &&
(dst->addr < 0x80000000 || dst->addr >= 0xC0000000))||no_compiled_jump)
{
gencallinterp((unsigned int)BGTZ, 1);
return;
}
genbgtz_test();
gendelayslot();
gentest();
#endif
}
void genbgtz_out()
{
#ifdef INTERPRET_BGTZ_OUT
gencallinterp((unsigned int)BGTZ_OUT, 1);
#else
if (((dst->addr & 0xFFF) == 0xFFC &&
(dst->addr < 0x80000000 || dst->addr >= 0xC0000000))||no_compiled_jump)
{
gencallinterp((unsigned int)BGTZ_OUT, 1);
return;
}
genbgtz_test();
gendelayslot();
gentest_out();
#endif
}
void genbgtz_idle()
{
#ifdef INTERPRET_BGTZ_IDLE
gencallinterp((unsigned int)BGTZ_IDLE, 1);
#else
if (((dst->addr & 0xFFF) == 0xFFC &&
(dst->addr < 0x80000000 || dst->addr >= 0xC0000000))||no_compiled_jump)
{
gencallinterp((unsigned int)BGTZ_IDLE, 1);
return;
}
genbgtz_test();
gentest_idle();
genbgtz();
#endif
}
void genaddi()
{
#ifdef INTERPRET_ADDI
gencallinterp((unsigned int)ADDI, 0);
#else
int rs = allocate_register((unsigned int *)dst->f.i.rs);
int rt = allocate_register_w((unsigned int *)dst->f.i.rt);
mov_reg32_reg32(rt, rs);
add_reg32_imm32(rt,(int)dst->f.i.immediate);
#endif
}
void genaddiu()
{
#ifdef INTERPRET_ADDIU
gencallinterp((unsigned int)ADDIU, 0);
#else
int rs = allocate_register((unsigned int *)dst->f.i.rs);
int rt = allocate_register_w((unsigned int *)dst->f.i.rt);
mov_reg32_reg32(rt, rs);
add_reg32_imm32(rt,(int)dst->f.i.immediate);
#endif
}
void genslti()
{
#ifdef INTERPRET_SLTI
gencallinterp((unsigned int)SLTI, 0);
#else
int rs1 = allocate_64_register1((unsigned int *)dst->f.i.rs);
int rs2 = allocate_64_register2((unsigned int *)dst->f.i.rs);
int rt = allocate_register_w((unsigned int *)dst->f.i.rt);
long long imm = (long long)dst->f.i.immediate;
cmp_reg32_imm32(rs2, (unsigned int)(imm >> 32));
jl_rj(17);
jne_rj(8); // 2
cmp_reg32_imm32(rs1, (unsigned int)imm); // 6
jl_rj(7); // 2
mov_reg32_imm32(rt, 0); // 5
jmp_imm_short(5); // 2
mov_reg32_imm32(rt, 1); // 5
#endif
}
void gensltiu()
{
#ifdef INTERPRET_SLTIU
gencallinterp((unsigned int)SLTIU, 0);
#else
int rs1 = allocate_64_register1((unsigned int *)dst->f.i.rs);
int rs2 = allocate_64_register2((unsigned int *)dst->f.i.rs);
int rt = allocate_register_w((unsigned int *)dst->f.i.rt);
long long imm = (long long)dst->f.i.immediate;
cmp_reg32_imm32(rs2, (unsigned int)(imm >> 32));
jb_rj(17);
jne_rj(8); // 2
cmp_reg32_imm32(rs1, (unsigned int)imm); // 6
jb_rj(7); // 2
mov_reg32_imm32(rt, 0); // 5
jmp_imm_short(5); // 2
mov_reg32_imm32(rt, 1); // 5
#endif
}
void genandi()
{
#ifdef INTERPRET_ANDI
gencallinterp((unsigned int)ANDI, 0);
#else
int rs = allocate_register((unsigned int *)dst->f.i.rs);
int rt = allocate_register_w((unsigned int *)dst->f.i.rt);
mov_reg32_reg32(rt, rs);
and_reg32_imm32(rt, (unsigned short)dst->f.i.immediate);
#endif
}
void genori()
{
#ifdef INTERPRET_ORI
gencallinterp((unsigned int)ORI, 0);
#else
int rs1 = allocate_64_register1((unsigned int *)dst->f.i.rs);
int rs2 = allocate_64_register2((unsigned int *)dst->f.i.rs);
int rt1 = allocate_64_register1_w((unsigned int *)dst->f.i.rt);
int rt2 = allocate_64_register2_w((unsigned int *)dst->f.i.rt);
mov_reg32_reg32(rt1, rs1);
mov_reg32_reg32(rt2, rs2);
or_reg32_imm32(rt1, (unsigned short)dst->f.i.immediate);
#endif
}
void genxori()
{
#ifdef INTERPRET_XORI
gencallinterp((unsigned int)XORI, 0);
#else
int rs1 = allocate_64_register1((unsigned int *)dst->f.i.rs);
int rs2 = allocate_64_register2((unsigned int *)dst->f.i.rs);
int rt1 = allocate_64_register1_w((unsigned int *)dst->f.i.rt);
int rt2 = allocate_64_register2_w((unsigned int *)dst->f.i.rt);
mov_reg32_reg32(rt1, rs1);
mov_reg32_reg32(rt2, rs2);
xor_reg32_imm32(rt1, (unsigned short)dst->f.i.immediate);
#endif
}
void genlui()
{
#ifdef INTERPRET_LUI
gencallinterp((unsigned int)LUI, 0);
#else
int rt = allocate_register_w((unsigned int *)dst->f.i.rt);
mov_reg32_imm32(rt, (unsigned int)dst->f.i.immediate << 16);
#endif
}
void gentestl()
{
unsigned int temp, temp2;
cmp_m32_imm32((unsigned int *)(&branch_taken), 0);
je_near_rj(0);
temp = code_length;
gendelayslot();
mov_m32_imm32((void*)(&last_addr), dst->addr + (dst-1)->f.i.immediate*4);
gencheck_interupt((unsigned int)(dst + (dst-1)->f.i.immediate));
jmp(dst->addr + (dst-1)->f.i.immediate*4);
temp2 = code_length;
code_length = temp-4;
put32(temp2 - temp);
code_length = temp2;
genupdate_count(dst->addr-4);
mov_m32_imm32((void*)(&last_addr), dst->addr + 4);
gencheck_interupt((unsigned int)(dst + 1));
jmp(dst->addr + 4);
}
void genbeql()
{
#ifdef INTERPRET_BEQL
gencallinterp((unsigned int)BEQL, 1);
#else
if (((dst->addr & 0xFFF) == 0xFFC &&
(dst->addr < 0x80000000 || dst->addr >= 0xC0000000))||no_compiled_jump)
{
gencallinterp((unsigned int)BEQL, 1);
return;
}
genbeq_test();
free_all_registers();
gentestl();
#endif
}
void gentestl_out()
{
unsigned int temp, temp2;
cmp_m32_imm32((unsigned int *)(&branch_taken), 0);
je_near_rj(0);
temp = code_length;
gendelayslot();
mov_m32_imm32((void*)(&last_addr), dst->addr + (dst-1)->f.i.immediate*4);
gencheck_interupt_out(dst->addr + (dst-1)->f.i.immediate*4);
mov_m32_imm32(&jump_to_address, dst->addr + (dst-1)->f.i.immediate*4);
mov_m32_imm32((unsigned int*)(&PC), (unsigned int)(dst+1));
mov_reg32_imm32(EAX, (unsigned int)jump_to_func);
call_reg32(EAX);
temp2 = code_length;
code_length = temp-4;
put32(temp2 - temp);
code_length = temp2;
genupdate_count(dst->addr-4);
mov_m32_imm32((void*)(&last_addr), dst->addr + 4);
gencheck_interupt((unsigned int)(dst + 1));
jmp(dst->addr + 4);
}
void genbeql_out()
{
#ifdef INTERPRET_BEQL_OUT
gencallinterp((unsigned int)BEQL_OUT, 1);
#else
if (((dst->addr & 0xFFF) == 0xFFC &&
(dst->addr < 0x80000000 || dst->addr >= 0xC0000000))||no_compiled_jump)
{
gencallinterp((unsigned int)BEQL_OUT, 1);
return;
}
genbeq_test();
free_all_registers();
gentestl_out();
#endif
}
void genbeql_idle()
{
#ifdef INTERPRET_BEQL_IDLE
gencallinterp((unsigned int)BEQL_IDLE, 1);
#else
if (((dst->addr & 0xFFF) == 0xFFC &&
(dst->addr < 0x80000000 || dst->addr >= 0xC0000000))||no_compiled_jump)
{
gencallinterp((unsigned int)BEQL_IDLE, 1);
return;
}
genbeq_test();
gentest_idle();
genbeql();
#endif
}
void genbnel()
{
#ifdef INTERPRET_BNEL
gencallinterp((unsigned int)BNEL, 1);
#else
if (((dst->addr & 0xFFF) == 0xFFC &&
(dst->addr < 0x80000000 || dst->addr >= 0xC0000000))||no_compiled_jump)
{
gencallinterp((unsigned int)BNEL, 1);
return;
}
genbne_test();
free_all_registers();
gentestl();
#endif
}
void genbnel_out()
{
#ifdef INTERPRET_BNEL_OUT
gencallinterp((unsigned int)BNEL_OUT, 1);
#else
if (((dst->addr & 0xFFF) == 0xFFC &&
(dst->addr < 0x80000000 || dst->addr >= 0xC0000000))||no_compiled_jump)
{
gencallinterp((unsigned int)BNEL_OUT, 1);
return;
}
genbne_test();
free_all_registers();
gentestl_out();
#endif
}
void genbnel_idle()
{
#ifdef INTERPRET_BNEL_IDLE
gencallinterp((unsigned int)BNEL_IDLE, 1);
#else
if (((dst->addr & 0xFFF) == 0xFFC &&
(dst->addr < 0x80000000 || dst->addr >= 0xC0000000))||no_compiled_jump)
{
gencallinterp((unsigned int)BNEL_IDLE, 1);
return;
}
genbne_test();
gentest_idle();
genbnel();
#endif
}
void genblezl()
{
#ifdef INTERPRET_BLEZL
gencallinterp((unsigned int)BLEZL, 1);
#else
if (((dst->addr & 0xFFF) == 0xFFC &&
(dst->addr < 0x80000000 || dst->addr >= 0xC0000000))||no_compiled_jump)
{
gencallinterp((unsigned int)BLEZL, 1);
return;
}
genblez_test();
free_all_registers();
gentestl();
#endif
}
void genblezl_out()
{
#ifdef INTERPRET_BLEZL_OUT
gencallinterp((unsigned int)BLEZL_OUT, 1);
#else
if (((dst->addr & 0xFFF) == 0xFFC &&
(dst->addr < 0x80000000 || dst->addr >= 0xC0000000))||no_compiled_jump)
{
gencallinterp((unsigned int)BLEZL_OUT, 1);
return;
}
genblez_test();
free_all_registers();
gentestl_out();
#endif
}
void genblezl_idle()
{
#ifdef INTERPRET_BLEZL_IDLE
gencallinterp((unsigned int)BLEZL_IDLE, 1);
#else
if (((dst->addr & 0xFFF) == 0xFFC &&
(dst->addr < 0x80000000 || dst->addr >= 0xC0000000))||no_compiled_jump)
{
gencallinterp((unsigned int)BLEZL_IDLE, 1);
return;
}
genblez_test();
gentest_idle();
genblezl();
#endif
}
void genbgtzl()
{
#ifdef INTERPRET_BGTZL
gencallinterp((unsigned int)BGTZL, 1);
#else
if (((dst->addr & 0xFFF) == 0xFFC &&
(dst->addr < 0x80000000 || dst->addr >= 0xC0000000))||no_compiled_jump)
{
gencallinterp((unsigned int)BGTZL, 1);
return;
}
genbgtz_test();
free_all_registers();
gentestl();
#endif
}
void genbgtzl_out()
{
#ifdef INTERPRET_BGTZL_OUT
gencallinterp((unsigned int)BGTZL_OUT, 1);
#else
if (((dst->addr & 0xFFF) == 0xFFC &&
(dst->addr < 0x80000000 || dst->addr >= 0xC0000000))||no_compiled_jump)
{
gencallinterp((unsigned int)BGTZL_OUT, 1);
return;
}
genbgtz_test();
free_all_registers();
gentestl_out();
#endif
}
void genbgtzl_idle()
{
#ifdef INTERPRET_BGTZL_IDLE
gencallinterp((unsigned int)BGTZL_IDLE, 1);
#else
if (((dst->addr & 0xFFF) == 0xFFC &&
(dst->addr < 0x80000000 || dst->addr >= 0xC0000000))||no_compiled_jump)
{
gencallinterp((unsigned int)BGTZL_IDLE, 1);
return;
}
genbgtz_test();
gentest_idle();
genbgtzl();
#endif
}
void gendaddi()
{
#ifdef INTERPRET_DADDI
gencallinterp((unsigned int)DADDI, 0);
#else
int rs1 = allocate_64_register1((unsigned int *)dst->f.i.rs);
int rs2 = allocate_64_register2((unsigned int *)dst->f.i.rs);
int rt1 = allocate_64_register1_w((unsigned int *)dst->f.i.rt);
int rt2 = allocate_64_register2_w((unsigned int *)dst->f.i.rt);
mov_reg32_reg32(rt1, rs1);
mov_reg32_reg32(rt2, rs2);
add_reg32_imm32(rt1, dst->f.i.immediate);
adc_reg32_imm32(rt2, (int)dst->f.i.immediate>>31);
#endif
}
void gendaddiu()
{
#ifdef INTERPRET_DADDIU
gencallinterp((unsigned int)DADDIU, 0);
#else
int rs1 = allocate_64_register1((unsigned int *)dst->f.i.rs);
int rs2 = allocate_64_register2((unsigned int *)dst->f.i.rs);
int rt1 = allocate_64_register1_w((unsigned int *)dst->f.i.rt);
int rt2 = allocate_64_register2_w((unsigned int *)dst->f.i.rt);
mov_reg32_reg32(rt1, rs1);
mov_reg32_reg32(rt2, rs2);
add_reg32_imm32(rt1, dst->f.i.immediate);
adc_reg32_imm32(rt2, (int)dst->f.i.immediate>>31);
#endif
}
void genldl()
{
gencallinterp((unsigned int)LDL, 0);
}
void genldr()
{
gencallinterp((unsigned int)LDR, 0);
}
void genlb()
{
#ifdef INTERPRET_LB
gencallinterp((unsigned int)LB, 0);
#else
free_all_registers();
simplify_access();
mov_eax_memoffs32((unsigned int *)dst->f.i.rs);
add_eax_imm32((int)dst->f.i.immediate);
mov_reg32_reg32(EBX, EAX);
if(fast_memory)
{
and_eax_imm32(0xDF800000);
cmp_eax_imm32(0x80000000);
}
else
{
shr_reg32_imm8(EAX, 16);
mov_reg32_preg32x4pimm32(EAX, EAX, (unsigned int)readmemb);
cmp_reg32_imm32(EAX, (unsigned int)read_rdramb);
}
je_rj(47);
mov_m32_imm32((void *)(&PC), (unsigned int)(dst+1)); // 10
mov_m32_reg32((unsigned int *)(&address), EBX); // 6
mov_m32_imm32((unsigned int *)(&rdword), (unsigned int)dst->f.i.rt); // 10
shr_reg32_imm8(EBX, 16); // 3
mov_reg32_preg32x4pimm32(EBX, EBX, (unsigned int)readmemb); // 7
call_reg32(EBX); // 2
movsx_reg32_m8(EAX, (unsigned char *)dst->f.i.rt); // 7
jmp_imm_short(16); // 2
and_reg32_imm32(EBX, 0x7FFFFF); // 6
xor_reg8_imm8(BL, 3); // 3
movsx_reg32_8preg32pimm32(EAX, EBX, (unsigned int)rdram); // 7
set_register_state(EAX, (unsigned int*)dst->f.i.rt, 1);
#endif
}
void genlh()
{
#ifdef INTERPRET_LH
gencallinterp((unsigned int)LH, 0);
#else
free_all_registers();
simplify_access();
mov_eax_memoffs32((unsigned int *)dst->f.i.rs);
add_eax_imm32((int)dst->f.i.immediate);
mov_reg32_reg32(EBX, EAX);
if(fast_memory)
{
and_eax_imm32(0xDF800000);
cmp_eax_imm32(0x80000000);
}
else
{
shr_reg32_imm8(EAX, 16);
mov_reg32_preg32x4pimm32(EAX, EAX, (unsigned int)readmemh);
cmp_reg32_imm32(EAX, (unsigned int)read_rdramh);
}
je_rj(47);
mov_m32_imm32((void *)(&PC), (unsigned int)(dst+1)); // 10
mov_m32_reg32((unsigned int *)(&address), EBX); // 6
mov_m32_imm32((unsigned int *)(&rdword), (unsigned int)dst->f.i.rt); // 10
shr_reg32_imm8(EBX, 16); // 3
mov_reg32_preg32x4pimm32(EBX, EBX, (unsigned int)readmemh); // 7
call_reg32(EBX); // 2
movsx_reg32_m16(EAX, (unsigned short *)dst->f.i.rt); // 7
jmp_imm_short(16); // 2
and_reg32_imm32(EBX, 0x7FFFFF); // 6
xor_reg8_imm8(BL, 2); // 3
movsx_reg32_16preg32pimm32(EAX, EBX, (unsigned int)rdram); // 7
set_register_state(EAX, (unsigned int*)dst->f.i.rt, 1);
#endif
}
void genlwl()
{
gencallinterp((unsigned int)LWL, 0);
}
void genlw()
{
#ifdef INTERPRET_LW
gencallinterp((unsigned int)LW, 0);
#else
free_all_registers();
simplify_access();
mov_eax_memoffs32((unsigned int *)dst->f.i.rs);
add_eax_imm32((int)dst->f.i.immediate);
mov_reg32_reg32(EBX, EAX);
if(fast_memory)
{
and_eax_imm32(0xDF800000);
cmp_eax_imm32(0x80000000);
}
else
{
shr_reg32_imm8(EAX, 16);
mov_reg32_preg32x4pimm32(EAX, EAX, (unsigned int)readmem);
cmp_reg32_imm32(EAX, (unsigned int)read_rdram);
}
je_rj(45);
mov_m32_imm32((void *)(&PC), (unsigned int)(dst+1)); // 10
mov_m32_reg32((unsigned int *)(&address), EBX); // 6
mov_m32_imm32((unsigned int *)(&rdword), (unsigned int)dst->f.i.rt); // 10
shr_reg32_imm8(EBX, 16); // 3
mov_reg32_preg32x4pimm32(EBX, EBX, (unsigned int)readmem); // 7
call_reg32(EBX); // 2
mov_eax_memoffs32((unsigned int *)(dst->f.i.rt)); // 5
jmp_imm_short(12); // 2
and_reg32_imm32(EBX, 0x7FFFFF); // 6
mov_reg32_preg32pimm32(EAX, EBX, (unsigned int)rdram); // 6
set_register_state(EAX, (unsigned int*)dst->f.i.rt, 1);
#endif
}
void genlbu()
{
#ifdef INTERPRET_LBU
gencallinterp((unsigned int)LBU, 0);
#else
free_all_registers();
simplify_access();
mov_eax_memoffs32((unsigned int *)dst->f.i.rs);
add_eax_imm32((int)dst->f.i.immediate);
mov_reg32_reg32(EBX, EAX);
if(fast_memory)
{
and_eax_imm32(0xDF800000);
cmp_eax_imm32(0x80000000);
}
else
{
shr_reg32_imm8(EAX, 16);
mov_reg32_preg32x4pimm32(EAX, EAX, (unsigned int)readmemb);
cmp_reg32_imm32(EAX, (unsigned int)read_rdramb);
}
je_rj(46);
mov_m32_imm32((void *)(&PC), (unsigned int)(dst+1)); // 10
mov_m32_reg32((unsigned int *)(&address), EBX); // 6
mov_m32_imm32((unsigned int *)(&rdword), (unsigned int)dst->f.i.rt); // 10
shr_reg32_imm8(EBX, 16); // 3
mov_reg32_preg32x4pimm32(EBX, EBX, (unsigned int)readmemb); // 7
call_reg32(EBX); // 2
mov_reg32_m32(EAX, (unsigned int *)dst->f.i.rt); // 6
jmp_imm_short(15); // 2
and_reg32_imm32(EBX, 0x7FFFFF); // 6
xor_reg8_imm8(BL, 3); // 3
mov_reg32_preg32pimm32(EAX, EBX, (unsigned int)rdram); // 6
and_eax_imm32(0xFF);
set_register_state(EAX, (unsigned int*)dst->f.i.rt, 1);
#endif
}
void genlhu()
{
#ifdef INTERPRET_LHU
gencallinterp((unsigned int)LHU, 0);
#else
free_all_registers();
simplify_access();
mov_eax_memoffs32((unsigned int *)dst->f.i.rs);
add_eax_imm32((int)dst->f.i.immediate);
mov_reg32_reg32(EBX, EAX);
if(fast_memory)
{
and_eax_imm32(0xDF800000);
cmp_eax_imm32(0x80000000);
}
else
{
shr_reg32_imm8(EAX, 16);
mov_reg32_preg32x4pimm32(EAX, EAX, (unsigned int)readmemh);
cmp_reg32_imm32(EAX, (unsigned int)read_rdramh);
}
je_rj(46);
mov_m32_imm32((void *)(&PC), (unsigned int)(dst+1)); // 10
mov_m32_reg32((unsigned int *)(&address), EBX); // 6
mov_m32_imm32((unsigned int *)(&rdword), (unsigned int)dst->f.i.rt); // 10
shr_reg32_imm8(EBX, 16); // 3
mov_reg32_preg32x4pimm32(EBX, EBX, (unsigned int)readmemh); // 7
call_reg32(EBX); // 2
mov_reg32_m32(EAX, (unsigned int *)dst->f.i.rt); // 6
jmp_imm_short(15); // 2
and_reg32_imm32(EBX, 0x7FFFFF); // 6
xor_reg8_imm8(BL, 2); // 3
mov_reg32_preg32pimm32(EAX, EBX, (unsigned int)rdram); // 6
and_eax_imm32(0xFFFF);
set_register_state(EAX, (unsigned int*)dst->f.i.rt, 1);
#endif
}
void genlwr()
{
gencallinterp((unsigned int)LWR, 0);
}
void genlwu()
{
#ifdef INTERPRET_LWU
gencallinterp((unsigned int)LWU, 0);
#else
free_all_registers();
simplify_access();
mov_eax_memoffs32((unsigned int *)dst->f.i.rs);
add_eax_imm32((int)dst->f.i.immediate);
mov_reg32_reg32(EBX, EAX);
if(fast_memory)
{
and_eax_imm32(0xDF800000);
cmp_eax_imm32(0x80000000);
}
else
{
shr_reg32_imm8(EAX, 16);
mov_reg32_preg32x4pimm32(EAX, EAX, (unsigned int)readmem);
cmp_reg32_imm32(EAX, (unsigned int)read_rdram);
}
je_rj(45);
mov_m32_imm32((void *)(&PC), (unsigned int)(dst+1)); // 10
mov_m32_reg32((unsigned int *)(&address), EBX); // 6
mov_m32_imm32((unsigned int *)(&rdword), (unsigned int)dst->f.i.rt); // 10
shr_reg32_imm8(EBX, 16); // 3
mov_reg32_preg32x4pimm32(EBX, EBX, (unsigned int)readmem); // 7
call_reg32(EBX); // 2
mov_eax_memoffs32((unsigned int *)(dst->f.i.rt)); // 5
jmp_imm_short(12); // 2
and_reg32_imm32(EBX, 0x7FFFFF); // 6
mov_reg32_preg32pimm32(EAX, EBX, (unsigned int)rdram); // 6
xor_reg32_reg32(EBX, EBX);
set_64_register_state(EAX, EBX, (unsigned int*)dst->f.i.rt, 1);
#endif
}
void gensb()
{
#ifdef INTERPRET_SB
gencallinterp((unsigned int)SB, 0);
#else
free_all_registers();
simplify_access();
mov_reg8_m8(CL, (unsigned char *)dst->f.i.rt);
mov_eax_memoffs32((unsigned int *)dst->f.i.rs);
add_eax_imm32((int)dst->f.i.immediate);
mov_reg32_reg32(EBX, EAX);
if(fast_memory)
{
and_eax_imm32(0xDF800000);
cmp_eax_imm32(0x80000000);
}
else
{
shr_reg32_imm8(EAX, 16);
mov_reg32_preg32x4pimm32(EAX, EAX, (unsigned int)writememb);
cmp_reg32_imm32(EAX, (unsigned int)write_rdramb);
}
je_rj(41);
mov_m32_imm32((void *)(&PC), (unsigned int)(dst+1)); // 10
mov_m32_reg32((unsigned int *)(&address), EBX); // 6
mov_m8_reg8((unsigned char *)(&byte), CL); // 6
shr_reg32_imm8(EBX, 16); // 3
mov_reg32_preg32x4pimm32(EBX, EBX, (unsigned int)writememb); // 7
call_reg32(EBX); // 2
mov_eax_memoffs32((unsigned int *)(&address)); // 5
jmp_imm_short(17); // 2
mov_reg32_reg32(EAX, EBX); // 2
and_reg32_imm32(EBX, 0x7FFFFF); // 6
xor_reg8_imm8(BL, 3); // 3
mov_preg32pimm32_reg8(EBX, (unsigned int)rdram, CL); // 6
mov_reg32_reg32(EBX, EAX);
shr_reg32_imm8(EBX, 12);
cmp_preg32pimm32_imm8(EBX, (unsigned int)invalid_code, 0);
jne_rj(54);
mov_reg32_reg32(ECX, EBX); // 2
shl_reg32_imm8(EBX, 2); // 3
mov_reg32_preg32pimm32(EBX, EBX, (unsigned int)blocks); // 6
mov_reg32_preg32pimm32(EBX, EBX, (int)&actual->block - (int)actual); // 6
and_eax_imm32(0xFFF); // 5
shr_reg32_imm8(EAX, 2); // 3
mov_reg32_imm32(EDX, sizeof(precomp_instr)); // 5
mul_reg32(EDX); // 2
mov_reg32_preg32preg32pimm32(EAX, EAX, EBX, (int)&dst->ops - (int)dst); // 7
cmp_reg32_imm32(EAX, (unsigned int)NOTCOMPILED); // 6
je_rj(7); // 2
mov_preg32pimm32_imm8(ECX, (unsigned int)invalid_code, 1); // 7
#endif
}
void gensh()
{
#ifdef INTERPRET_SH
gencallinterp((unsigned int)SH, 0);
#else
free_all_registers();
simplify_access();
mov_reg16_m16(CX, (unsigned short *)dst->f.i.rt);
mov_eax_memoffs32((unsigned int *)dst->f.i.rs);
add_eax_imm32((int)dst->f.i.immediate);
mov_reg32_reg32(EBX, EAX);
if(fast_memory)
{
and_eax_imm32(0xDF800000);
cmp_eax_imm32(0x80000000);
}
else
{
shr_reg32_imm8(EAX, 16);
mov_reg32_preg32x4pimm32(EAX, EAX, (unsigned int)writememh);
cmp_reg32_imm32(EAX, (unsigned int)write_rdramh);
}
je_rj(42);
mov_m32_imm32((void *)(&PC), (unsigned int)(dst+1)); // 10
mov_m32_reg32((unsigned int *)(&address), EBX); // 6
mov_m16_reg16((unsigned short *)(&hword), CX); // 7
shr_reg32_imm8(EBX, 16); // 3
mov_reg32_preg32x4pimm32(EBX, EBX, (unsigned int)writememh); // 7
call_reg32(EBX); // 2
mov_eax_memoffs32((unsigned int *)(&address)); // 5
jmp_imm_short(18); // 2
mov_reg32_reg32(EAX, EBX); // 2
and_reg32_imm32(EBX, 0x7FFFFF); // 6
xor_reg8_imm8(BL, 2); // 3
mov_preg32pimm32_reg16(EBX, (unsigned int)rdram, CX); // 7
mov_reg32_reg32(EBX, EAX);
shr_reg32_imm8(EBX, 12);
cmp_preg32pimm32_imm8(EBX, (unsigned int)invalid_code, 0);
jne_rj(54);
mov_reg32_reg32(ECX, EBX); // 2
shl_reg32_imm8(EBX, 2); // 3
mov_reg32_preg32pimm32(EBX, EBX, (unsigned int)blocks); // 6
mov_reg32_preg32pimm32(EBX, EBX, (int)&actual->block - (int)actual); // 6
and_eax_imm32(0xFFF); // 5
shr_reg32_imm8(EAX, 2); // 3
mov_reg32_imm32(EDX, sizeof(precomp_instr)); // 5
mul_reg32(EDX); // 2
mov_reg32_preg32preg32pimm32(EAX, EAX, EBX, (int)&dst->ops - (int)dst); // 7
cmp_reg32_imm32(EAX, (unsigned int)NOTCOMPILED); // 6
je_rj(7); // 2
mov_preg32pimm32_imm8(ECX, (unsigned int)invalid_code, 1); // 7
#endif
}
void genswl()
{
gencallinterp((unsigned int)SWL, 0);
}
void gensw()
{
#ifdef INTERPRET_SW
gencallinterp((unsigned int)SW, 0);
#else
free_all_registers();
simplify_access();
mov_reg32_m32(ECX, (unsigned int *)dst->f.i.rt);
mov_eax_memoffs32((unsigned int *)dst->f.i.rs);
add_eax_imm32((int)dst->f.i.immediate);
mov_reg32_reg32(EBX, EAX);
if(fast_memory)
{
and_eax_imm32(0xDF800000);
cmp_eax_imm32(0x80000000);
}
else
{
shr_reg32_imm8(EAX, 16);
mov_reg32_preg32x4pimm32(EAX, EAX, (unsigned int)writemem);
cmp_reg32_imm32(EAX, (unsigned int)write_rdram);
}
je_rj(41);
mov_m32_imm32((void *)(&PC), (unsigned int)(dst+1)); // 10
mov_m32_reg32((unsigned int *)(&address), EBX); // 6
mov_m32_reg32((unsigned int *)(&word), ECX); // 6
shr_reg32_imm8(EBX, 16); // 3
mov_reg32_preg32x4pimm32(EBX, EBX, (unsigned int)writemem); // 7
call_reg32(EBX); // 2
mov_eax_memoffs32((unsigned int *)(&address)); // 5
jmp_imm_short(14); // 2
mov_reg32_reg32(EAX, EBX); // 2
and_reg32_imm32(EBX, 0x7FFFFF); // 6
mov_preg32pimm32_reg32(EBX, (unsigned int)rdram, ECX); // 6
mov_reg32_reg32(EBX, EAX);
shr_reg32_imm8(EBX, 12);
cmp_preg32pimm32_imm8(EBX, (unsigned int)invalid_code, 0);
jne_rj(54);
mov_reg32_reg32(ECX, EBX); // 2
shl_reg32_imm8(EBX, 2); // 3
mov_reg32_preg32pimm32(EBX, EBX, (unsigned int)blocks); // 6
mov_reg32_preg32pimm32(EBX, EBX, (int)&actual->block - (int)actual); // 6
and_eax_imm32(0xFFF); // 5
shr_reg32_imm8(EAX, 2); // 3
mov_reg32_imm32(EDX, sizeof(precomp_instr)); // 5
mul_reg32(EDX); // 2
mov_reg32_preg32preg32pimm32(EAX, EAX, EBX, (int)&dst->ops - (int)dst); // 7
cmp_reg32_imm32(EAX, (unsigned int)NOTCOMPILED); // 6
je_rj(7); // 2
mov_preg32pimm32_imm8(ECX, (unsigned int)invalid_code, 1); // 7
#endif
}
void gensdl()
{
gencallinterp((unsigned int)SDL, 0);
}
void gensdr()
{
gencallinterp((unsigned int)SDR, 0);
}
void genswr()
{
gencallinterp((unsigned int)SWR, 0);
}
void gencheck_cop1_unusable()
{
unsigned int temp, temp2;
free_all_registers();
simplify_access();
test_m32_imm32((unsigned int*)&Status, 0x20000000);
jne_rj(0);
temp = code_length;
gencallinterp((unsigned int)check_cop1_unusable, 0);
temp2 = code_length;
code_length = temp - 1;
put8(temp2 - temp);
code_length = temp2;
}
void genlwc1()
{
#ifdef INTERPRET_LWC1
gencallinterp((unsigned int)LWC1, 0);
#else
gencheck_cop1_unusable();
mov_eax_memoffs32((unsigned int *)(&reg[dst->f.lf.base]));
add_eax_imm32((int)dst->f.lf.offset);
mov_reg32_reg32(EBX, EAX);
if(fast_memory)
{
and_eax_imm32(0xDF800000);
cmp_eax_imm32(0x80000000);
}
else
{
shr_reg32_imm8(EAX, 16);
mov_reg32_preg32x4pimm32(EAX, EAX, (unsigned int)readmem);
cmp_reg32_imm32(EAX, (unsigned int)read_rdram);
}
je_rj(42);
mov_m32_imm32((void *)(&PC), (unsigned int)(dst+1)); // 10
mov_m32_reg32((unsigned int *)(&address), EBX); // 6
mov_reg32_m32(EDX, (unsigned int*)(&reg_cop1_simple[dst->f.lf.ft])); // 6
mov_m32_reg32((unsigned int *)(&rdword), EDX); // 6
shr_reg32_imm8(EBX, 16); // 3
mov_reg32_preg32x4pimm32(EBX, EBX, (unsigned int)readmem); // 7
call_reg32(EBX); // 2
jmp_imm_short(20); // 2
and_reg32_imm32(EBX, 0x7FFFFF); // 6
mov_reg32_preg32pimm32(EAX, EBX, (unsigned int)rdram); // 6
mov_reg32_m32(EBX, (unsigned int*)(&reg_cop1_simple[dst->f.lf.ft])); // 6
mov_preg32_reg32(EBX, EAX); // 2
#endif
}
void genldc1()
{
#ifdef INTERPRET_LDC1
gencallinterp((unsigned int)LDC1, 0);
#else
gencheck_cop1_unusable();
mov_eax_memoffs32((unsigned int *)(&reg[dst->f.lf.base]));
add_eax_imm32((int)dst->f.lf.offset);
mov_reg32_reg32(EBX, EAX);
if(fast_memory)
{
and_eax_imm32(0xDF800000);
cmp_eax_imm32(0x80000000);
}
else
{
shr_reg32_imm8(EAX, 16);
mov_reg32_preg32x4pimm32(EAX, EAX, (unsigned int)readmemd);
cmp_reg32_imm32(EAX, (unsigned int)read_rdramd);
}
je_rj(42);
mov_m32_imm32((void *)(&PC), (unsigned int)(dst+1)); // 10
mov_m32_reg32((unsigned int *)(&address), EBX); // 6
mov_reg32_m32(EDX, (unsigned int*)(&reg_cop1_double[dst->f.lf.ft])); // 6
mov_m32_reg32((unsigned int *)(&rdword), EDX); // 6
shr_reg32_imm8(EBX, 16); // 3
mov_reg32_preg32x4pimm32(EBX, EBX, (unsigned int)readmemd); // 7
call_reg32(EBX); // 2
jmp_imm_short(32); // 2
and_reg32_imm32(EBX, 0x7FFFFF); // 6
mov_reg32_preg32pimm32(EAX, EBX, ((unsigned int)rdram)+4); // 6
mov_reg32_preg32pimm32(ECX, EBX, ((unsigned int)rdram)); // 6
mov_reg32_m32(EBX, (unsigned int*)(&reg_cop1_double[dst->f.lf.ft])); // 6
mov_preg32_reg32(EBX, EAX); // 2
mov_preg32pimm32_reg32(EBX, 4, ECX); // 6
#endif
}
void gencache()
{
}
void genld()
{
#ifdef INTERPRET_LD
gencallinterp((unsigned int)LD, 0);
#else
free_all_registers();
simplify_access();
mov_eax_memoffs32((unsigned int *)dst->f.i.rs);
add_eax_imm32((int)dst->f.i.immediate);
mov_reg32_reg32(EBX, EAX);
if(fast_memory)
{
and_eax_imm32(0xDF800000);
cmp_eax_imm32(0x80000000);
}
else
{
shr_reg32_imm8(EAX, 16);
mov_reg32_preg32x4pimm32(EAX, EAX, (unsigned int)readmemd);
cmp_reg32_imm32(EAX, (unsigned int)read_rdramd);
}
je_rj(51);
mov_m32_imm32((void *)(&PC), (unsigned int)(dst+1)); // 10
mov_m32_reg32((unsigned int *)(&address), EBX); // 6
mov_m32_imm32((unsigned int *)(&rdword), (unsigned int)dst->f.i.rt); // 10
shr_reg32_imm8(EBX, 16); // 3
mov_reg32_preg32x4pimm32(EBX, EBX, (unsigned int)readmemd); // 7
call_reg32(EBX); // 2
mov_eax_memoffs32((unsigned int *)(dst->f.i.rt)); // 5
mov_reg32_m32(ECX, (unsigned int *)(dst->f.i.rt)+1); // 6
jmp_imm_short(18); // 2
and_reg32_imm32(EBX, 0x7FFFFF); // 6
mov_reg32_preg32pimm32(EAX, EBX, ((unsigned int)rdram)+4); // 6
mov_reg32_preg32pimm32(ECX, EBX, ((unsigned int)rdram)); // 6
set_64_register_state(EAX, ECX, (unsigned int*)dst->f.i.rt, 1);
#endif
}
void genswc1()
{
#ifdef INTERPRET_SWC1
gencallinterp((unsigned int)SWC1, 0);
#else
gencheck_cop1_unusable();
mov_reg32_m32(EDX, (unsigned int*)(&reg_cop1_simple[dst->f.lf.ft]));
mov_reg32_preg32(ECX, EDX);
mov_eax_memoffs32((unsigned int *)(&reg[dst->f.lf.base]));
add_eax_imm32((int)dst->f.lf.offset);
mov_reg32_reg32(EBX, EAX);
if(fast_memory)
{
and_eax_imm32(0xDF800000);
cmp_eax_imm32(0x80000000);
}
else
{
shr_reg32_imm8(EAX, 16);
mov_reg32_preg32x4pimm32(EAX, EAX, (unsigned int)writemem);
cmp_reg32_imm32(EAX, (unsigned int)write_rdram);
}
je_rj(41);
mov_m32_imm32((void *)(&PC), (unsigned int)(dst+1)); // 10
mov_m32_reg32((unsigned int *)(&address), EBX); // 6
mov_m32_reg32((unsigned int *)(&word), ECX); // 6
shr_reg32_imm8(EBX, 16); // 3
mov_reg32_preg32x4pimm32(EBX, EBX, (unsigned int)writemem); // 7
call_reg32(EBX); // 2
mov_eax_memoffs32((unsigned int *)(&address)); // 5
jmp_imm_short(14); // 2
mov_reg32_reg32(EAX, EBX); // 2
and_reg32_imm32(EBX, 0x7FFFFF); // 6
mov_preg32pimm32_reg32(EBX, (unsigned int)rdram, ECX); // 6
mov_reg32_reg32(EBX, EAX);
shr_reg32_imm8(EBX, 12);
cmp_preg32pimm32_imm8(EBX, (unsigned int)invalid_code, 0);
jne_rj(54);
mov_reg32_reg32(ECX, EBX); // 2
shl_reg32_imm8(EBX, 2); // 3
mov_reg32_preg32pimm32(EBX, EBX, (unsigned int)blocks); // 6
mov_reg32_preg32pimm32(EBX, EBX, (int)&actual->block - (int)actual); // 6
and_eax_imm32(0xFFF); // 5
shr_reg32_imm8(EAX, 2); // 3
mov_reg32_imm32(EDX, sizeof(precomp_instr)); // 5
mul_reg32(EDX); // 2
mov_reg32_preg32preg32pimm32(EAX, EAX, EBX, (int)&dst->ops - (int)dst); // 7
cmp_reg32_imm32(EAX, (unsigned int)NOTCOMPILED); // 6
je_rj(7); // 2
mov_preg32pimm32_imm8(ECX, (unsigned int)invalid_code, 1); // 7
#endif
}
void gensdc1()
{
#ifdef INTERPRET_SDC1
gencallinterp((unsigned int)SDC1, 0);
#else
gencheck_cop1_unusable();
mov_reg32_m32(ESI, (unsigned int*)(&reg_cop1_double[dst->f.lf.ft]));
mov_reg32_preg32(ECX, ESI);
mov_reg32_preg32pimm32(EDX, ESI, 4);
mov_eax_memoffs32((unsigned int *)(&reg[dst->f.lf.base]));
add_eax_imm32((int)dst->f.lf.offset);
mov_reg32_reg32(EBX, EAX);
if(fast_memory)
{
and_eax_imm32(0xDF800000);
cmp_eax_imm32(0x80000000);
}
else
{
shr_reg32_imm8(EAX, 16);
mov_reg32_preg32x4pimm32(EAX, EAX, (unsigned int)writememd);
cmp_reg32_imm32(EAX, (unsigned int)write_rdramd);
}
je_rj(47);
mov_m32_imm32((void *)(&PC), (unsigned int)(dst+1)); // 10
mov_m32_reg32((unsigned int *)(&address), EBX); // 6
mov_m32_reg32((unsigned int *)(&dword), ECX); // 6
mov_m32_reg32((unsigned int *)(&dword)+1, EDX); // 6
shr_reg32_imm8(EBX, 16); // 3
mov_reg32_preg32x4pimm32(EBX, EBX, (unsigned int)writememd); // 7
call_reg32(EBX); // 2
mov_eax_memoffs32((unsigned int *)(&address)); // 5
jmp_imm_short(20); // 2
mov_reg32_reg32(EAX, EBX); // 2
and_reg32_imm32(EBX, 0x7FFFFF); // 6
mov_preg32pimm32_reg32(EBX, ((unsigned int)rdram)+4, ECX); // 6
mov_preg32pimm32_reg32(EBX, ((unsigned int)rdram)+0, EDX); // 6
mov_reg32_reg32(EBX, EAX);
shr_reg32_imm8(EBX, 12);
cmp_preg32pimm32_imm8(EBX, (unsigned int)invalid_code, 0);
jne_rj(54);
mov_reg32_reg32(ECX, EBX); // 2
shl_reg32_imm8(EBX, 2); // 3
mov_reg32_preg32pimm32(EBX, EBX, (unsigned int)blocks); // 6
mov_reg32_preg32pimm32(EBX, EBX, (int)&actual->block - (int)actual); // 6
and_eax_imm32(0xFFF); // 5
shr_reg32_imm8(EAX, 2); // 3
mov_reg32_imm32(EDX, sizeof(precomp_instr)); // 5
mul_reg32(EDX); // 2
mov_reg32_preg32preg32pimm32(EAX, EAX, EBX, (int)&dst->ops - (int)dst); // 7
cmp_reg32_imm32(EAX, (unsigned int)NOTCOMPILED); // 6
je_rj(7); // 2
mov_preg32pimm32_imm8(ECX, (unsigned int)invalid_code, 1); // 7
#endif
}
void gensd()
{
#ifdef INTERPRET_SD
gencallinterp((unsigned int)SD, 0);
#else
free_all_registers();
simplify_access();
mov_reg32_m32(ECX, (unsigned int *)dst->f.i.rt);
mov_reg32_m32(EDX, ((unsigned int *)dst->f.i.rt)+1);
mov_eax_memoffs32((unsigned int *)dst->f.i.rs);
add_eax_imm32((int)dst->f.i.immediate);
mov_reg32_reg32(EBX, EAX);
if(fast_memory)
{
and_eax_imm32(0xDF800000);
cmp_eax_imm32(0x80000000);
}
else
{
shr_reg32_imm8(EAX, 16);
mov_reg32_preg32x4pimm32(EAX, EAX, (unsigned int)writememd);
cmp_reg32_imm32(EAX, (unsigned int)write_rdramd);
}
je_rj(47);
mov_m32_imm32((void *)(&PC), (unsigned int)(dst+1)); // 10
mov_m32_reg32((unsigned int *)(&address), EBX); // 6
mov_m32_reg32((unsigned int *)(&dword), ECX); // 6
mov_m32_reg32((unsigned int *)(&dword)+1, EDX); // 6
shr_reg32_imm8(EBX, 16); // 3
mov_reg32_preg32x4pimm32(EBX, EBX, (unsigned int)writememd); // 7
call_reg32(EBX); // 2
mov_eax_memoffs32((unsigned int *)(&address)); // 5
jmp_imm_short(20); // 2
mov_reg32_reg32(EAX, EBX); // 2
and_reg32_imm32(EBX, 0x7FFFFF); // 6
mov_preg32pimm32_reg32(EBX, ((unsigned int)rdram)+4, ECX); // 6
mov_preg32pimm32_reg32(EBX, ((unsigned int)rdram)+0, EDX); // 6
mov_reg32_reg32(EBX, EAX);
shr_reg32_imm8(EBX, 12);
cmp_preg32pimm32_imm8(EBX, (unsigned int)invalid_code, 0);
jne_rj(54);
mov_reg32_reg32(ECX, EBX); // 2
shl_reg32_imm8(EBX, 2); // 3
mov_reg32_preg32pimm32(EBX, EBX, (unsigned int)blocks); // 6
mov_reg32_preg32pimm32(EBX, EBX, (int)&actual->block - (int)actual); // 6
and_eax_imm32(0xFFF); // 5
shr_reg32_imm8(EAX, 2); // 3
mov_reg32_imm32(EDX, sizeof(precomp_instr)); // 5
mul_reg32(EDX); // 2
mov_reg32_preg32preg32pimm32(EAX, EAX, EBX, (int)&dst->ops - (int)dst); // 7
cmp_reg32_imm32(EAX, (unsigned int)NOTCOMPILED); // 6
je_rj(7); // 2
mov_preg32pimm32_imm8(ECX, (unsigned int)invalid_code, 1); // 7
#endif
}
void genll()
{
gencallinterp((unsigned int)LL, 0);
}
void gensc()
{
gencallinterp((unsigned int)SC, 0);
}