mirror of https://github.com/PCSX2/pcsx2.git
Made some tweaks to the EErec block manager in hopes of resolving Issue 208 [Soul Calibur 3 missing gfx], involving the willbranch3 logic path.
Emitter: Added xJcc8 / xJcc32 functions for doing modified jump targets. Also: minor cleanup to recent counters fixes (nothing relevant, just reduced out some copy-paste jobs for clarity). git-svn-id: http://pcsx2.googlecode.com/svn/trunk@1143 96395faa-99c1-11dd-bbfe-3dabce05a288
This commit is contained in:
parent
3202a42578
commit
91a4ee79cc
|
@ -85,16 +85,12 @@ static __forceinline void _rcntSet( int cntidx )
|
||||||
if (c < nextCounter)
|
if (c < nextCounter)
|
||||||
{
|
{
|
||||||
nextCounter = c;
|
nextCounter = c;
|
||||||
|
cpuSetNextBranch( nextsCounter, nextCounter ); //Need to update on counter resets/target changes
|
||||||
if((g_nextBranchCycle - nextsCounter) > (u32)nextCounter) //Need to update on counter resets/target changes
|
|
||||||
{
|
|
||||||
g_nextBranchCycle = nextsCounter + nextCounter;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Ignore target diff if target is currently disabled.
|
// Ignore target diff if target is currently disabled.
|
||||||
// (the overflow is all we care about since it goes first, and then the
|
// (the overflow is all we care about since it goes first, and then the
|
||||||
// target will be turned on afterward).
|
// target will be turned on afterward, and handled in the next event test).
|
||||||
|
|
||||||
if( counter.target & EECNT_FUTURE_TARGET )
|
if( counter.target & EECNT_FUTURE_TARGET )
|
||||||
{
|
{
|
||||||
|
@ -107,15 +103,10 @@ static __forceinline void _rcntSet( int cntidx )
|
||||||
if (c < nextCounter)
|
if (c < nextCounter)
|
||||||
{
|
{
|
||||||
nextCounter = c;
|
nextCounter = c;
|
||||||
|
cpuSetNextBranch( nextsCounter, nextCounter ); //Need to update on counter resets/target changes
|
||||||
if((g_nextBranchCycle - nextsCounter) > (u32)nextCounter) //Need to update on counter resets/target changes
|
|
||||||
{
|
|
||||||
g_nextBranchCycle = nextsCounter + nextCounter;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
//cpuSetNextBranch( nextsCounter, nextCounter );
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
static __forceinline void cpuRcntSet()
|
static __forceinline void cpuRcntSet()
|
||||||
|
|
|
@ -95,11 +95,7 @@ static void _rcntSet( int cntidx )
|
||||||
if(c < (u64)psxNextCounter)
|
if(c < (u64)psxNextCounter)
|
||||||
{
|
{
|
||||||
psxNextCounter = (u32)c;
|
psxNextCounter = (u32)c;
|
||||||
|
psxSetNextBranch( psxNextsCounter, psxNextCounter ); //Need to update on counter resets/target changes
|
||||||
if((g_psxNextBranchCycle - psxNextsCounter) > (u32)psxNextCounter) //Need to update on counter resets/target changes
|
|
||||||
{
|
|
||||||
g_psxNextBranchCycle = psxNextsCounter + psxNextCounter;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
//if((counter.mode & 0x10) == 0 || psxCounters[i].target > 0xffff) continue;
|
//if((counter.mode & 0x10) == 0 || psxCounters[i].target > 0xffff) continue;
|
||||||
|
@ -111,11 +107,7 @@ static void _rcntSet( int cntidx )
|
||||||
if(c < (u64)psxNextCounter)
|
if(c < (u64)psxNextCounter)
|
||||||
{
|
{
|
||||||
psxNextCounter = (u32)c;
|
psxNextCounter = (u32)c;
|
||||||
|
psxSetNextBranch( psxNextsCounter, psxNextCounter ); //Need to update on counter resets/target changes
|
||||||
if((g_psxNextBranchCycle - psxNextsCounter) > (u32)psxNextCounter) //Need to update on counter resets/target changes
|
|
||||||
{
|
|
||||||
g_psxNextBranchCycle = psxNextsCounter + psxNextCounter;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -30,19 +30,16 @@ void so_call(coroutine_t coro);
|
||||||
void so_resume(void);
|
void so_resume(void);
|
||||||
void so_exit(void);
|
void so_exit(void);
|
||||||
|
|
||||||
// I can't find where the Linux recRecompile is defined. Is it used anymore?
|
|
||||||
// If so, namespacing might break it. :/ (air)
|
|
||||||
void recRecompile( u32 startpc );
|
void recRecompile( u32 startpc );
|
||||||
|
|
||||||
// aR3000A.S
|
// aR3000A.S
|
||||||
void iopRecRecompile(u32 startpc);
|
void iopRecRecompile(u32 startpc);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Linux specific
|
|
||||||
#ifdef __LINUX__
|
#ifdef __LINUX__
|
||||||
|
|
||||||
PCSX2_ALIGNED16( u8 _xmm_backup[16*2] );
|
PCSX2_ALIGNED16_EXTERN( u8 _xmm_backup[16*2] );
|
||||||
PCSX2_ALIGNED16( u8 _mmx_backup[8*4] );
|
PCSX2_ALIGNED16_EXTERN( u8 _mmx_backup[8*4] );
|
||||||
|
|
||||||
extern "C"
|
extern "C"
|
||||||
{
|
{
|
||||||
|
@ -60,8 +57,7 @@ void iopDispatcherReg();
|
||||||
void JITCompile();
|
void JITCompile();
|
||||||
void JITCompileInBlock();
|
void JITCompileInBlock();
|
||||||
void DispatcherReg();
|
void DispatcherReg();
|
||||||
|
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
#endif
|
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
|
@ -90,12 +90,13 @@ BASEBLOCKEX* BaseBlocks::GetByX86(uptr ip)
|
||||||
return &blocks[imin];
|
return &blocks[imin];
|
||||||
}
|
}
|
||||||
|
|
||||||
void BaseBlocks::Link(u32 pc, uptr jumpptr)
|
void BaseBlocks::Link(u32 pc, s32* jumpptr)
|
||||||
{
|
{
|
||||||
BASEBLOCKEX *targetblock = Get(pc);
|
BASEBLOCKEX *targetblock = Get(pc);
|
||||||
if (targetblock && targetblock->startpc == pc)
|
if (targetblock && targetblock->startpc == pc)
|
||||||
*(u32*)jumpptr = targetblock->fnptr - (jumpptr + 4);
|
*jumpptr = (s32)(targetblock->fnptr - (sptr)(jumpptr + 1));
|
||||||
else
|
else
|
||||||
*(u32*)jumpptr = recompiler - (jumpptr + 4);
|
*jumpptr = (s32)(recompiler - (sptr)(jumpptr + 1));
|
||||||
links.insert(std::pair<u32, uptr>(pc, jumpptr));
|
links.insert(std::pair<u32, uptr>(pc, (uptr)jumpptr));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -115,7 +115,7 @@ public:
|
||||||
blocks.erase(blocks.begin() + idx);
|
blocks.erase(blocks.begin() + idx);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Link(u32 pc, uptr jumpptr);
|
void Link(u32 pc, s32* jumpptr);
|
||||||
|
|
||||||
__forceinline void Reset()
|
__forceinline void Reset()
|
||||||
{
|
{
|
||||||
|
|
|
@ -39,6 +39,8 @@
|
||||||
#include "SamplProf.h"
|
#include "SamplProf.h"
|
||||||
#include "NakedAsm.h"
|
#include "NakedAsm.h"
|
||||||
|
|
||||||
|
using namespace x86Emitter;
|
||||||
|
|
||||||
extern u32 g_psxNextBranchCycle;
|
extern u32 g_psxNextBranchCycle;
|
||||||
extern void psxBREAK();
|
extern void psxBREAK();
|
||||||
extern void zeroEx();
|
extern void zeroEx();
|
||||||
|
@ -57,11 +59,51 @@ uptr psxhwLUT[0x10000];
|
||||||
// R3000A statics
|
// R3000A statics
|
||||||
int psxreclog = 0;
|
int psxreclog = 0;
|
||||||
|
|
||||||
|
#ifdef _MSC_VER
|
||||||
|
|
||||||
|
static u32 g_temp;
|
||||||
|
|
||||||
|
// The address for all cleared blocks. It recompiles the current pc and then
|
||||||
|
// dispatches to the recompiled block address.
|
||||||
|
static __declspec(naked) void iopJITCompile()
|
||||||
|
{
|
||||||
|
__asm {
|
||||||
|
mov esi, dword ptr [psxRegs.pc]
|
||||||
|
push esi
|
||||||
|
call iopRecRecompile
|
||||||
|
add esp, 4
|
||||||
|
mov ebx, esi
|
||||||
|
shr esi, 16
|
||||||
|
mov ecx, dword ptr [psxRecLUT+esi*4]
|
||||||
|
jmp dword ptr [ecx+ebx]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static __declspec(naked) void iopJITCompileInBlock()
|
||||||
|
{
|
||||||
|
__asm {
|
||||||
|
jmp iopJITCompile
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// called when jumping to variable psxpc address
|
||||||
|
static __declspec(naked) void iopDispatcherReg()
|
||||||
|
{
|
||||||
|
__asm {
|
||||||
|
mov eax, dword ptr [psxRegs.pc]
|
||||||
|
mov ebx, eax
|
||||||
|
shr eax, 16
|
||||||
|
mov ecx, dword ptr [psxRecLUT+eax*4]
|
||||||
|
jmp dword ptr [ecx+ebx]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif // _MSC_VER
|
||||||
|
|
||||||
|
|
||||||
static u8 *recMem = NULL; // the recompiled blocks will be here
|
static u8 *recMem = NULL; // the recompiled blocks will be here
|
||||||
static BASEBLOCK *recRAM = NULL; // and the ptr to the blocks here
|
static BASEBLOCK *recRAM = NULL; // and the ptr to the blocks here
|
||||||
static BASEBLOCK *recROM = NULL; // and here
|
static BASEBLOCK *recROM = NULL; // and here
|
||||||
static BASEBLOCK *recROM1 = NULL; // also here
|
static BASEBLOCK *recROM1 = NULL; // also here
|
||||||
void iopJITCompile();
|
|
||||||
static BaseBlocks recBlocks((uptr)iopJITCompile);
|
static BaseBlocks recBlocks((uptr)iopJITCompile);
|
||||||
static u8 *recPtr = NULL;
|
static u8 *recPtr = NULL;
|
||||||
u32 psxpc; // recompiler psxpc
|
u32 psxpc; // recompiler psxpc
|
||||||
|
@ -596,46 +638,6 @@ static void recShutdown()
|
||||||
|
|
||||||
u32 g_psxlastpc = 0;
|
u32 g_psxlastpc = 0;
|
||||||
|
|
||||||
#ifdef _MSC_VER
|
|
||||||
|
|
||||||
static u32 g_temp;
|
|
||||||
|
|
||||||
// The address for all cleared blocks. It recompiles the current pc and then
|
|
||||||
// dispatches to the recompiled block address.
|
|
||||||
static __declspec(naked) void iopJITCompile()
|
|
||||||
{
|
|
||||||
__asm {
|
|
||||||
mov esi, dword ptr [psxRegs.pc]
|
|
||||||
push esi
|
|
||||||
call iopRecRecompile
|
|
||||||
add esp, 4
|
|
||||||
mov ebx, esi
|
|
||||||
shr esi, 16
|
|
||||||
mov ecx, dword ptr [psxRecLUT+esi*4]
|
|
||||||
jmp dword ptr [ecx+ebx]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static __declspec(naked) void iopJITCompileInBlock()
|
|
||||||
{
|
|
||||||
__asm {
|
|
||||||
jmp iopJITCompile
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// called when jumping to variable psxpc address
|
|
||||||
static __declspec(naked) void iopDispatcherReg()
|
|
||||||
{
|
|
||||||
__asm {
|
|
||||||
mov eax, dword ptr [psxRegs.pc]
|
|
||||||
mov ebx, eax
|
|
||||||
shr eax, 16
|
|
||||||
mov ecx, dword ptr [psxRecLUT+eax*4]
|
|
||||||
jmp dword ptr [ecx+ebx]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif // _MSC_VER
|
|
||||||
|
|
||||||
static void iopClearRecLUT(BASEBLOCK* base, int count)
|
static void iopClearRecLUT(BASEBLOCK* base, int count)
|
||||||
{
|
{
|
||||||
for (int i = 0; i < count; i++)
|
for (int i = 0; i < count; i++)
|
||||||
|
@ -778,7 +780,6 @@ void psxSetBranchReg(u32 reg)
|
||||||
|
|
||||||
void psxSetBranchImm( u32 imm )
|
void psxSetBranchImm( u32 imm )
|
||||||
{
|
{
|
||||||
u32* ptr;
|
|
||||||
psxbranch = 1;
|
psxbranch = 1;
|
||||||
assert( imm );
|
assert( imm );
|
||||||
|
|
||||||
|
@ -787,16 +788,9 @@ void psxSetBranchImm( u32 imm )
|
||||||
_psxFlushCall(FLUSH_EVERYTHING);
|
_psxFlushCall(FLUSH_EVERYTHING);
|
||||||
iPsxBranchTest(imm, imm <= psxpc);
|
iPsxBranchTest(imm, imm <= psxpc);
|
||||||
|
|
||||||
ptr = JMP32(0);
|
recBlocks.Link(HWADDR(imm), xJcc32());
|
||||||
recBlocks.Link(HWADDR(imm), (uptr)ptr);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
//fixme : this is all a huge hack, we base the counter advancements on the average an opcode should take (wtf?)
|
|
||||||
// If that wasn't bad enough we have default values like 9/8 which will get cast to int later
|
|
||||||
// (yeah, that means all sync code couldn't have worked to begin with)
|
|
||||||
// So for now these are new settings that work.
|
|
||||||
// (rama)
|
|
||||||
|
|
||||||
static __forceinline u32 psxScaleBlockCycles()
|
static __forceinline u32 psxScaleBlockCycles()
|
||||||
{
|
{
|
||||||
return s_psxBlockCycles * (Config.Hacks.IOPCycleDouble ? 2 : 1);
|
return s_psxBlockCycles * (Config.Hacks.IOPCycleDouble ? 2 : 1);
|
||||||
|
@ -1139,8 +1133,7 @@ StartRecomp:
|
||||||
assert( psxpc == s_nEndBlock );
|
assert( psxpc == s_nEndBlock );
|
||||||
_psxFlushCall(FLUSH_EVERYTHING);
|
_psxFlushCall(FLUSH_EVERYTHING);
|
||||||
MOV32ItoM((uptr)&psxRegs.pc, psxpc);
|
MOV32ItoM((uptr)&psxRegs.pc, psxpc);
|
||||||
u32 *ptr = JMP32(0);
|
recBlocks.Link(HWADDR(s_nEndBlock), xJcc32() );
|
||||||
recBlocks.Link(HWADDR(s_nEndBlock), (uptr)ptr);
|
|
||||||
psxbranch = 3;
|
psxbranch = 3;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -484,15 +484,6 @@ void recResetEE( void )
|
||||||
recLUT_SetPage(recLUT, hwLUT, recROM1, 0xa000, i, i - 0x1e00);
|
recLUT_SetPage(recLUT, hwLUT, recROM1, 0xa000, i, i - 0x1e00);
|
||||||
}
|
}
|
||||||
|
|
||||||
// drk||Raziel says this is useful but I'm not sure why. Something to do with forward jumps.
|
|
||||||
// Anyways, it causes random crashing for some reasom, possibly because of memory
|
|
||||||
// corrupition elsewhere in the recs. I can't reproduce the problem here though,
|
|
||||||
// so a fix will have to wait until later. -_- (air)
|
|
||||||
|
|
||||||
//x86SetPtr(recMem+REC_CACHEMEM);
|
|
||||||
//dyna_block_discard_recmem=(u8*)x86Ptr;
|
|
||||||
//JMP32( (uptr)&dyna_block_discard - ( (u32)x86Ptr + 5 ));
|
|
||||||
|
|
||||||
x86SetPtr(recMem);
|
x86SetPtr(recMem);
|
||||||
|
|
||||||
recPtr = recMem;
|
recPtr = recMem;
|
||||||
|
@ -725,7 +716,7 @@ void recBREAK( void ) {
|
||||||
} } } // end namespace R5900::Dynarec::OpcodeImpl
|
} } } // end namespace R5900::Dynarec::OpcodeImpl
|
||||||
|
|
||||||
// Clears the recLUT table so that all blocks are mapped to the JIT recompiler by default.
|
// Clears the recLUT table so that all blocks are mapped to the JIT recompiler by default.
|
||||||
static void ClearRecLUT(BASEBLOCK* base, int count)
|
static __releaseinline void ClearRecLUT(BASEBLOCK* base, int count)
|
||||||
{
|
{
|
||||||
for (int i = 0; i < count; i++)
|
for (int i = 0; i < count; i++)
|
||||||
base[i].SetFnptr((uptr)JITCompile);
|
base[i].SetFnptr((uptr)JITCompile);
|
||||||
|
@ -967,7 +958,7 @@ void iFlushCall(int flushtype)
|
||||||
//}
|
//}
|
||||||
|
|
||||||
|
|
||||||
u32 eeScaleBlockCycles()
|
static u32 scaleBlockCycles_helper()
|
||||||
{
|
{
|
||||||
// Note: s_nBlockCycles is 3 bit fixed point. Divide by 8 when done!
|
// Note: s_nBlockCycles is 3 bit fixed point. Divide by 8 when done!
|
||||||
|
|
||||||
|
@ -999,12 +990,6 @@ u32 eeScaleBlockCycles()
|
||||||
scalarHigh = 7;
|
scalarHigh = 7;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 3: // Sync hack x3
|
|
||||||
scalarLow = 10;
|
|
||||||
scalarMid = 19;
|
|
||||||
scalarHigh = 10;
|
|
||||||
break;
|
|
||||||
|
|
||||||
jNO_DEFAULT
|
jNO_DEFAULT
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1016,19 +1001,14 @@ u32 eeScaleBlockCycles()
|
||||||
return temp >> (3+2);
|
return temp >> (3+2);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void iBranch(u32 newpc, int type)
|
static u32 eeScaleBlockCycles()
|
||||||
{
|
{
|
||||||
u32* ptr;
|
// Ensures block cycles count is never less than 1:
|
||||||
|
u32 retval = scaleBlockCycles_helper();
|
||||||
MOV32ItoM((uptr)&cpuRegs.pc, newpc);
|
return (retval < 1) ? 1 : retval;
|
||||||
if (type == 0)
|
|
||||||
ptr = JMP32(0);
|
|
||||||
else if (type == 1)
|
|
||||||
ptr = JS32(0);
|
|
||||||
|
|
||||||
recBlocks.Link(HWADDR(newpc), (uptr)ptr);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// Generates dynarec code for Event tests followed by a block dispatch (branch).
|
// Generates dynarec code for Event tests followed by a block dispatch (branch).
|
||||||
// Parameters:
|
// Parameters:
|
||||||
// newpc - address to jump to at the end of the block. If newpc == 0xffffffff then
|
// newpc - address to jump to at the end of the block. If newpc == 0xffffffff then
|
||||||
|
@ -1058,21 +1038,24 @@ static void iBranchTest(u32 newpc, bool noDispatch)
|
||||||
xCMP(eax, ptr32[&cpuRegs.cycle]);
|
xCMP(eax, ptr32[&cpuRegs.cycle]);
|
||||||
xCMOVL(eax, ptr32[&cpuRegs.cycle]);
|
xCMOVL(eax, ptr32[&cpuRegs.cycle]);
|
||||||
xMOV(ptr32[&cpuRegs.cycle], eax);
|
xMOV(ptr32[&cpuRegs.cycle], eax);
|
||||||
RET();
|
|
||||||
} else {
|
} else {
|
||||||
MOV32MtoR(EAX, (uptr)&cpuRegs.cycle);
|
xMOV(eax, &cpuRegs.cycle);
|
||||||
ADD32ItoR(EAX, eeScaleBlockCycles());
|
xADD(eax, eeScaleBlockCycles());
|
||||||
MOV32RtoM((uptr)&cpuRegs.cycle, EAX); // update cycles
|
xMOV(&cpuRegs.cycle, eax); // update cycles
|
||||||
SUB32MtoR(EAX, (uptr)&g_nextBranchCycle);
|
xSUB(eax, &g_nextBranchCycle);
|
||||||
if (!noDispatch) {
|
if (!noDispatch)
|
||||||
|
{
|
||||||
if (newpc == 0xffffffff)
|
if (newpc == 0xffffffff)
|
||||||
JS32((uptr)DispatcherReg - ( (uptr)x86Ptr + 6 ));
|
xJS( DispatcherReg );
|
||||||
else
|
else
|
||||||
iBranch(newpc, 1);
|
{
|
||||||
|
xMOV( ptr32[&cpuRegs.pc], newpc );
|
||||||
|
recBlocks.Link( HWADDR(newpc), xJcc32( Jcc_Signed ) );
|
||||||
}
|
}
|
||||||
RET();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
xRET();
|
||||||
|
}
|
||||||
|
|
||||||
static void checkcodefn()
|
static void checkcodefn()
|
||||||
{
|
{
|
||||||
|
@ -1170,7 +1153,7 @@ void recompileNextInstruction(int delayslot)
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
//If thh COP0 DIE bit is disabled, double the cycles. Happens rarely.
|
//If the COP0 DIE bit is disabled, double the cycles. Happens rarely.
|
||||||
s_nBlockCycles += opcode.cycles * (2 - ((cpuRegs.CP0.n.Config >> 18) & 0x1));
|
s_nBlockCycles += opcode.cycles * (2 - ((cpuRegs.CP0.n.Config >> 18) & 0x1));
|
||||||
opcode.recompile();
|
opcode.recompile();
|
||||||
|
|
||||||
|
@ -1235,14 +1218,14 @@ void badespfn() {
|
||||||
void __fastcall dyna_block_discard(u32 start,u32 sz)
|
void __fastcall dyna_block_discard(u32 start,u32 sz)
|
||||||
{
|
{
|
||||||
DevCon::WriteLn("dyna_block_discard .. start: %08X count=%d", params start,sz);
|
DevCon::WriteLn("dyna_block_discard .. start: %08X count=%d", params start,sz);
|
||||||
Cpu->Clear(start, sz);
|
recClear(start, sz);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void __fastcall dyna_page_reset(u32 start,u32 sz)
|
void __fastcall dyna_page_reset(u32 start,u32 sz)
|
||||||
{
|
{
|
||||||
DevCon::WriteLn("dyna_page_reset .. start=%08X size=%d", params start,sz*4);
|
DevCon::WriteLn("dyna_page_reset .. start=%08X size=%d", params start,sz*4);
|
||||||
Cpu->Clear(start & ~0xfffUL, 0x400);
|
recClear(start & ~0xfffUL, 0x400);
|
||||||
manual_counter[start >> 12]++;
|
manual_counter[start >> 12]++;
|
||||||
mmap_MarkCountedRamPage(PSM(start), start & ~0xfffUL);
|
mmap_MarkCountedRamPage(PSM(start), start & ~0xfffUL);
|
||||||
}
|
}
|
||||||
|
@ -1657,10 +1640,14 @@ StartRecomp:
|
||||||
// performance reasons.
|
// performance reasons.
|
||||||
|
|
||||||
int numinsts = (pc - startpc) / 4;
|
int numinsts = (pc - startpc) / 4;
|
||||||
if( numinsts > 12 )
|
if( numinsts > 6 )
|
||||||
iBranchTest(pc);
|
iBranchTest(pc);
|
||||||
else
|
else
|
||||||
iBranch(pc,0); // unconditional static link
|
{
|
||||||
|
xMOV( ptr32[&cpuRegs.pc], pc );
|
||||||
|
xADD( ptr32[&cpuRegs.cycle], eeScaleBlockCycles() );
|
||||||
|
recBlocks.Link( HWADDR(pc), xJcc32() );
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -212,6 +212,8 @@ namespace x86Emitter
|
||||||
// JMP / Jcc Instructions!
|
// JMP / Jcc Instructions!
|
||||||
|
|
||||||
extern void xJcc( JccComparisonType comparison, const void* target );
|
extern void xJcc( JccComparisonType comparison, const void* target );
|
||||||
|
extern s8* xJcc8( JccComparisonType comparison=Jcc_Unconditional, s8 displacement=0 );
|
||||||
|
extern s32* xJcc32( JccComparisonType comparison=Jcc_Unconditional, s32 displacement=0 );
|
||||||
|
|
||||||
// ------------------------------------------------------------------------
|
// ------------------------------------------------------------------------
|
||||||
// Conditional jumps to fixed targets.
|
// Conditional jumps to fixed targets.
|
||||||
|
|
|
@ -73,6 +73,34 @@ xSmartJump::~xSmartJump()
|
||||||
m_baseptr = NULL; // just in case (sometimes helps in debugging too)
|
m_baseptr = NULL; // just in case (sometimes helps in debugging too)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ------------------------------------------------------------------------
|
||||||
|
// Emits a 32 bit jump, and returns a pointer to the 32 bit displacement.
|
||||||
|
// (displacements should be assigned relative to the end of the jump instruction,
|
||||||
|
// or in other words *(retval+1) )
|
||||||
|
__emitinline s32* xJcc32( JccComparisonType comparison, s32 displacement )
|
||||||
|
{
|
||||||
|
if( comparison == Jcc_Unconditional )
|
||||||
|
xWrite8( 0xe9 );
|
||||||
|
else
|
||||||
|
{
|
||||||
|
xWrite8( 0x0f );
|
||||||
|
xWrite8( 0x80 | comparison );
|
||||||
|
}
|
||||||
|
xWrite<s32>( displacement );
|
||||||
|
|
||||||
|
return ((s32*)xGetPtr()) - 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// ------------------------------------------------------------------------
|
||||||
|
// Emits a 32 bit jump, and returns a pointer to the 8 bit displacement.
|
||||||
|
// (displacements should be assigned relative to the end of the jump instruction,
|
||||||
|
// or in other words *(retval+1) )
|
||||||
|
__emitinline s8* xJcc8( JccComparisonType comparison, s8 displacement )
|
||||||
|
{
|
||||||
|
xWrite8( (comparison == Jcc_Unconditional) ? 0xeb : (0x70 | comparison) );
|
||||||
|
xWrite<s8>( displacement );
|
||||||
|
return (s8*)xGetPtr() - 1;
|
||||||
|
}
|
||||||
|
|
||||||
// ------------------------------------------------------------------------
|
// ------------------------------------------------------------------------
|
||||||
// Writes a jump at the current x86Ptr, which targets a pre-established target address.
|
// Writes a jump at the current x86Ptr, which targets a pre-established target address.
|
||||||
|
@ -84,7 +112,7 @@ xSmartJump::~xSmartJump()
|
||||||
__emitinline void Internal::xJccKnownTarget( JccComparisonType comparison, const void* target, bool slideForward )
|
__emitinline void Internal::xJccKnownTarget( JccComparisonType comparison, const void* target, bool slideForward )
|
||||||
{
|
{
|
||||||
// Calculate the potential j8 displacement first, assuming an instruction length of 2:
|
// Calculate the potential j8 displacement first, assuming an instruction length of 2:
|
||||||
sptr displacement8 = (sptr)target - ((sptr)xGetPtr() + 2);
|
sptr displacement8 = (sptr)target - (sptr)(xGetPtr() + 2);
|
||||||
|
|
||||||
const int slideVal = slideForward ? ((comparison == Jcc_Unconditional) ? 3 : 4) : 0;
|
const int slideVal = slideForward ? ((comparison == Jcc_Unconditional) ? 3 : 4) : 0;
|
||||||
displacement8 -= slideVal;
|
displacement8 -= slideVal;
|
||||||
|
@ -94,22 +122,12 @@ __emitinline void Internal::xJccKnownTarget( JccComparisonType comparison, const
|
||||||
if( slideForward ) jASSUME( displacement8 >= 0 );
|
if( slideForward ) jASSUME( displacement8 >= 0 );
|
||||||
|
|
||||||
if( is_s8( displacement8 ) )
|
if( is_s8( displacement8 ) )
|
||||||
{
|
xJcc8( comparison, displacement8 );
|
||||||
xWrite8( (comparison == Jcc_Unconditional) ? 0xeb : (0x70 | comparison) );
|
|
||||||
xWrite<s8>( displacement8 );
|
|
||||||
}
|
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
// Perform a 32 bit jump instead. :(
|
// Perform a 32 bit jump instead. :(
|
||||||
|
s32* bah = xJcc32( comparison );
|
||||||
if( comparison == Jcc_Unconditional )
|
*bah = (s32)target - (s32)xGetPtr();
|
||||||
xWrite8( 0xe9 );
|
|
||||||
else
|
|
||||||
{
|
|
||||||
xWrite8( 0x0f );
|
|
||||||
xWrite8( 0x80 | comparison );
|
|
||||||
}
|
|
||||||
xWrite<s32>( (sptr)target - ((sptr)xGetPtr() + 4) );
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue