microVU - more cleanups

git-svn-id: http://pcsx2.googlecode.com/svn/trunk@4515 96395faa-99c1-11dd-bbfe-3dabce05a288
This commit is contained in:
cottonvibes 2011-03-31 05:31:33 +00:00
parent 69896ce034
commit 0fef0e4e6c
5 changed files with 175 additions and 169 deletions

View File

@ -189,7 +189,6 @@ struct microVU {
ScopedPtr<microRegAlloc> regAlloc; // Reg Alloc Class ScopedPtr<microRegAlloc> regAlloc; // Reg Alloc Class
ScopedPtr<AsciiFile> logFile; // Log File Pointer ScopedPtr<AsciiFile> logFile; // Log File Pointer
RecompiledCodeReserve* cache_reserve; RecompiledCodeReserve* cache_reserve;
u8* cache; // Dynarec Cache Start (where we will start writing the recompiled code to) u8* cache; // Dynarec Cache Start (where we will start writing the recompiled code to)
u8* dispCache; // Dispatchers Cache (where startFunct and exitFunct are written to) u8* dispCache; // Dispatchers Cache (where startFunct and exitFunct are written to)
@ -212,37 +211,9 @@ struct microVU {
VURegs& regs() const { return ::vuRegs[index]; } VURegs& regs() const { return ::vuRegs[index]; }
VIFregisters& getVifRegs() const { return regs().GetVifRegs(); } __fi VIFregisters& getVifRegs() const { return regs().GetVifRegs(); }
__fi REG_VI& getVI(uint reg) const { return regs().VI[reg]; } __fi REG_VI& getVI(uint reg) const { return regs().VI[reg]; }
__fi VECTOR& getVF(uint reg) const { return regs().VF[reg]; } __fi VECTOR& getVF(uint reg) const { return regs().VF[reg]; }
__fi s16 Imm5() const { return ((code & 0x400) ? 0xfff0 : 0) | ((code >> 6) & 0xf); }
__fi s32 Imm11() const { return (code & 0x400) ? (0xfffffc00 | (code & 0x3ff)) : (code & 0x3ff); }
__fi u32 Imm12() const { return (((code >> 21) & 0x1) << 11) | (code & 0x7ff); }
__fi u32 Imm15() const { return ((code >> 10) & 0x7800) | (code & 0x7ff); }
__fi u32 Imm24() const { return code & 0xffffff; }
// Fetches the PC and instruction opcode relative to the current PC. Used to rewind and
// fast-forward the IR state while calculating VU pipeline conditions (branches, writebacks, etc)
__fi void advancePC( int x )
{
prog.IRinfo.curPC += x;
prog.IRinfo.curPC &= progMemMask;
code = ((u32*)regs().Micro)[prog.IRinfo.curPC];
}
__ri uint getBranchAddr() const
{
pxAssumeDev((prog.IRinfo.curPC & 1) == 0, "microVU recompiler: Upper instructions cannot have valid branch addresses.");
return (((prog.IRinfo.curPC + 2) + (Imm11() * 2)) & progMemMask) * 4;
}
__ri uint getBranchAddrN() const
{
pxAssumeDev((prog.IRinfo.curPC & 1) == 0, "microVU recompiler: Upper instructions cannot have valid branch addresses.");
return (((prog.IRinfo.curPC + 4) + (Imm11() * 2)) & progMemMask) * 4;
}
}; };
// microVU rec structs // microVU rec structs

View File

@ -20,97 +20,95 @@
//------------------------------------------------------------------ //------------------------------------------------------------------
//------------------------------------------------------------------ //------------------------------------------------------------------
// Helper Macros // Helper Functions
//------------------------------------------------------------------ //------------------------------------------------------------------
#define aReg(x) mVUregs.VF[x]
#define bReg(x, y) mVUregsTemp.VFreg[y] = x; mVUregsTemp.VF[y]
#define aMax(x, y) ((x > y) ? x : y)
#define aMin(x, y) ((x < y) ? x : y)
// Read a VF reg // Read a VF reg
#define analyzeReg1(xReg, vfRead) { \ __ri void analyzeReg1(mV, int xReg, microVFreg& vfRead) {
if (xReg) { \ if (xReg) {
if (_X) { mVUstall = aMax(mVUstall, aReg(xReg).x); vfRead.reg = xReg; vfRead.x = 1; } \ if (_X) { mVUstall = max(mVUstall, mVUregs.VF[xReg].x); vfRead.reg = xReg; vfRead.x = 1; }
if (_Y) { mVUstall = aMax(mVUstall, aReg(xReg).y); vfRead.reg = xReg; vfRead.y = 1; } \ if (_Y) { mVUstall = max(mVUstall, mVUregs.VF[xReg].y); vfRead.reg = xReg; vfRead.y = 1; }
if (_Z) { mVUstall = aMax(mVUstall, aReg(xReg).z); vfRead.reg = xReg; vfRead.z = 1; } \ if (_Z) { mVUstall = max(mVUstall, mVUregs.VF[xReg].z); vfRead.reg = xReg; vfRead.z = 1; }
if (_W) { mVUstall = aMax(mVUstall, aReg(xReg).w); vfRead.reg = xReg; vfRead.w = 1; } \ if (_W) { mVUstall = max(mVUstall, mVUregs.VF[xReg].w); vfRead.reg = xReg; vfRead.w = 1; }
} \ }
} }
// Write to a VF reg // Write to a VF reg
#define analyzeReg2(xReg, vfWrite, isLowOp) { \ __ri void analyzeReg2(mV, int xReg, microVFreg& vfWrite, bool isLowOp) {
if (xReg) { \ if (xReg) {
if (_X) { bReg(xReg, isLowOp).x = 4; vfWrite.reg = xReg; vfWrite.x = 4; } \ #define bReg(x, y) mVUregsTemp.VFreg[y] = x; mVUregsTemp.VF[y]
if (_Y) { bReg(xReg, isLowOp).y = 4; vfWrite.reg = xReg; vfWrite.y = 4; } \ if (_X) { bReg(xReg, isLowOp).x = 4; vfWrite.reg = xReg; vfWrite.x = 4; }
if (_Z) { bReg(xReg, isLowOp).z = 4; vfWrite.reg = xReg; vfWrite.z = 4; } \ if (_Y) { bReg(xReg, isLowOp).y = 4; vfWrite.reg = xReg; vfWrite.y = 4; }
if (_W) { bReg(xReg, isLowOp).w = 4; vfWrite.reg = xReg; vfWrite.w = 4; } \ if (_Z) { bReg(xReg, isLowOp).z = 4; vfWrite.reg = xReg; vfWrite.z = 4; }
} \ if (_W) { bReg(xReg, isLowOp).w = 4; vfWrite.reg = xReg; vfWrite.w = 4; }
}
} }
// Read a VF reg (BC opcodes) // Read a VF reg (BC opcodes)
#define analyzeReg3(xReg, vfRead) { \ __ri void analyzeReg3(mV, int xReg, microVFreg& vfRead) {
if (xReg) { \ if (xReg) {
if (_bc_x) { mVUstall = aMax(mVUstall, aReg(xReg).x); vfRead.reg = xReg; vfRead.x = 1; } \ if (_bc_x) { mVUstall = max(mVUstall, mVUregs.VF[xReg].x); vfRead.reg = xReg; vfRead.x = 1; }
else if (_bc_y) { mVUstall = aMax(mVUstall, aReg(xReg).y); vfRead.reg = xReg; vfRead.y = 1; } \ elif (_bc_y) { mVUstall = max(mVUstall, mVUregs.VF[xReg].y); vfRead.reg = xReg; vfRead.y = 1; }
else if (_bc_z) { mVUstall = aMax(mVUstall, aReg(xReg).z); vfRead.reg = xReg; vfRead.z = 1; } \ elif (_bc_z) { mVUstall = max(mVUstall, mVUregs.VF[xReg].z); vfRead.reg = xReg; vfRead.z = 1; }
else { mVUstall = aMax(mVUstall, aReg(xReg).w); vfRead.reg = xReg; vfRead.w = 1; } \ else { mVUstall = max(mVUstall, mVUregs.VF[xReg].w); vfRead.reg = xReg; vfRead.w = 1; }
} \ }
} }
// For Clip Opcode // For Clip Opcode
#define analyzeReg4(xReg, vfRead) { \ __ri void analyzeReg4(mV, int xReg, microVFreg& vfRead) {
if (xReg) { \ if (xReg) {
mVUstall = aMax(mVUstall, aReg(xReg).w); \ mVUstall = max(mVUstall, mVUregs.VF[xReg].w);
vfRead.reg = xReg; vfRead.w = 1; \ vfRead.reg = xReg;
} \ vfRead.w = 1;
}
} }
// Read VF reg (FsF/FtF) // Read VF reg (FsF/FtF)
#define analyzeReg5(xReg, fxf, vfRead) { \ __ri void analyzeReg5(mV, int xReg, int fxf, microVFreg& vfRead) {
if (xReg) { \ if (xReg) {
switch (fxf) { \ switch (fxf) {
case 0: mVUstall = aMax(mVUstall, aReg(xReg).x); vfRead.reg = xReg; vfRead.x = 1; break; \ case 0: mVUstall = max(mVUstall, mVUregs.VF[xReg].x); vfRead.reg = xReg; vfRead.x = 1; break;
case 1: mVUstall = aMax(mVUstall, aReg(xReg).y); vfRead.reg = xReg; vfRead.y = 1; break; \ case 1: mVUstall = max(mVUstall, mVUregs.VF[xReg].y); vfRead.reg = xReg; vfRead.y = 1; break;
case 2: mVUstall = aMax(mVUstall, aReg(xReg).z); vfRead.reg = xReg; vfRead.z = 1; break; \ case 2: mVUstall = max(mVUstall, mVUregs.VF[xReg].z); vfRead.reg = xReg; vfRead.z = 1; break;
case 3: mVUstall = aMax(mVUstall, aReg(xReg).w); vfRead.reg = xReg; vfRead.w = 1; break; \ case 3: mVUstall = max(mVUstall, mVUregs.VF[xReg].w); vfRead.reg = xReg; vfRead.w = 1; break;
} \ }
} \ }
} }
// Flips xyzw stalls to yzwx (MR32 Opcode) // Flips xyzw stalls to yzwx (MR32 Opcode)
#define analyzeReg6(xReg, vfRead) { \ __ri void analyzeReg6(mV, int xReg, microVFreg& vfRead) {
if (xReg) { \ if (xReg) {
if (_X) { mVUstall = aMax(mVUstall, aReg(xReg).y); vfRead.reg = xReg; vfRead.y = 1; } \ if (_X) { mVUstall = max(mVUstall, mVUregs.VF[xReg].y); vfRead.reg = xReg; vfRead.y = 1; }
if (_Y) { mVUstall = aMax(mVUstall, aReg(xReg).z); vfRead.reg = xReg; vfRead.z = 1; } \ if (_Y) { mVUstall = max(mVUstall, mVUregs.VF[xReg].z); vfRead.reg = xReg; vfRead.z = 1; }
if (_Z) { mVUstall = aMax(mVUstall, aReg(xReg).w); vfRead.reg = xReg; vfRead.w = 1; } \ if (_Z) { mVUstall = max(mVUstall, mVUregs.VF[xReg].w); vfRead.reg = xReg; vfRead.w = 1; }
if (_W) { mVUstall = aMax(mVUstall, aReg(xReg).x); vfRead.reg = xReg; vfRead.x = 1; } \ if (_W) { mVUstall = max(mVUstall, mVUregs.VF[xReg].x); vfRead.reg = xReg; vfRead.x = 1; }
} \ }
} }
// Reading a VI reg // Reading a VI reg
#define analyzeVIreg1(xReg, viRead) { \ __ri void analyzeVIreg1(mV, int xReg, microVIreg& viRead) {
if (xReg) { \ if (xReg) {
mVUstall = aMax(mVUstall, mVUregs.VI[xReg]); \ mVUstall = max(mVUstall, mVUregs.VI[xReg]);
viRead.reg = xReg; viRead.used = 1; \ viRead.reg = xReg;
} \ viRead.used = 1;
}
} }
// Writing to a VI reg // Writing to a VI reg
#define analyzeVIreg2(xReg, viWrite, aCycles) { \ __ri void analyzeVIreg2(mV, int xReg, microVIreg& viWrite, int aCycles) {
if (xReg) { \ if (xReg) {
mVUconstReg[xReg].isValid = 0; \ mVUconstReg[xReg].isValid = 0;
mVUregsTemp.VIreg = xReg; \ mVUregsTemp.VIreg = xReg;
mVUregsTemp.VI = aCycles; \ mVUregsTemp.VI = aCycles;
viWrite.reg = xReg; \ viWrite.reg = xReg;
viWrite.used = aCycles; \ viWrite.used = aCycles;
} \ }
} }
#define analyzeQreg(x) { mVUregsTemp.q = x; mVUstall = aMax(mVUstall, mVUregs.q); } #define analyzeQreg(x) { mVUregsTemp.q = x; mVUstall = max(mVUstall, mVUregs.q); }
#define analyzePreg(x) { mVUregsTemp.p = x; mVUstall = aMax(mVUstall, ((mVUregs.p) ? (mVUregs.p - 1) : 0)); } #define analyzePreg(x) { mVUregsTemp.p = x; mVUstall = max(mVUstall, (u8)((mVUregs.p) ? (mVUregs.p - 1) : 0)); }
#define analyzeRreg() { mVUregsTemp.r = 1; } #define analyzeRreg() { mVUregsTemp.r = 1; }
#define analyzeXGkick1() { mVUstall = aMax(mVUstall, mVUregs.xgkick); } #define analyzeXGkick1() { mVUstall = max(mVUstall, mVUregs.xgkick); }
#define analyzeXGkick2(x) { mVUregsTemp.xgkick = x; } #define analyzeXGkick2(x) { mVUregsTemp.xgkick = x; }
#define setConstReg(x, v) { if (x) { mVUconstReg[x].isValid = 1; mVUconstReg[x].regValue = v; } } #define setConstReg(x, v) { if (x) { mVUconstReg[x].isValid = 1; mVUconstReg[x].regValue = v; } }
@ -120,9 +118,9 @@
__fi void mVUanalyzeFMAC1(mV, int Fd, int Fs, int Ft) { __fi void mVUanalyzeFMAC1(mV, int Fd, int Fs, int Ft) {
sFLAG.doFlag = 1; sFLAG.doFlag = 1;
analyzeReg1(Fs, mVUup.VF_read[0]); analyzeReg1(mVU, Fs, mVUup.VF_read[0]);
analyzeReg1(Ft, mVUup.VF_read[1]); analyzeReg1(mVU, Ft, mVUup.VF_read[1]);
analyzeReg2(Fd, mVUup.VF_write, 0); analyzeReg2(mVU, Fd, mVUup.VF_write, 0);
} }
//------------------------------------------------------------------ //------------------------------------------------------------------
@ -130,8 +128,8 @@ __fi void mVUanalyzeFMAC1(mV, int Fd, int Fs, int Ft) {
//------------------------------------------------------------------ //------------------------------------------------------------------
__fi void mVUanalyzeFMAC2(mV, int Fs, int Ft) { __fi void mVUanalyzeFMAC2(mV, int Fs, int Ft) {
analyzeReg1(Fs, mVUup.VF_read[0]); analyzeReg1(mVU, Fs, mVUup.VF_read[0]);
analyzeReg2(Ft, mVUup.VF_write, 0); analyzeReg2(mVU, Ft, mVUup.VF_write, 0);
} }
//------------------------------------------------------------------ //------------------------------------------------------------------
@ -140,9 +138,9 @@ __fi void mVUanalyzeFMAC2(mV, int Fs, int Ft) {
__fi void mVUanalyzeFMAC3(mV, int Fd, int Fs, int Ft) { __fi void mVUanalyzeFMAC3(mV, int Fd, int Fs, int Ft) {
sFLAG.doFlag = 1; sFLAG.doFlag = 1;
analyzeReg1(Fs, mVUup.VF_read[0]); analyzeReg1(mVU, Fs, mVUup.VF_read[0]);
analyzeReg3(Ft, mVUup.VF_read[1]); analyzeReg3(mVU, Ft, mVUup.VF_read[1]);
analyzeReg2(Fd, mVUup.VF_write, 0); analyzeReg2(mVU, Fd, mVUup.VF_write, 0);
} }
//------------------------------------------------------------------ //------------------------------------------------------------------
@ -151,8 +149,8 @@ __fi void mVUanalyzeFMAC3(mV, int Fd, int Fs, int Ft) {
__fi void mVUanalyzeFMAC4(mV, int Fs, int Ft) { __fi void mVUanalyzeFMAC4(mV, int Fs, int Ft) {
cFLAG.doFlag = 1; cFLAG.doFlag = 1;
analyzeReg1(Fs, mVUup.VF_read[0]); analyzeReg1(mVU, Fs, mVUup.VF_read[0]);
analyzeReg4(Ft, mVUup.VF_read[1]); analyzeReg4(mVU, Ft, mVUup.VF_read[1]);
} }
//------------------------------------------------------------------ //------------------------------------------------------------------
@ -160,16 +158,16 @@ __fi void mVUanalyzeFMAC4(mV, int Fs, int Ft) {
//------------------------------------------------------------------ //------------------------------------------------------------------
__fi void mVUanalyzeIALU1(mV, int Id, int Is, int It) { __fi void mVUanalyzeIALU1(mV, int Id, int Is, int It) {
if (!Id) { mVUlow.isNOP = 1; } if (!Id) mVUlow.isNOP = 1;
analyzeVIreg1(Is, mVUlow.VI_read[0]); analyzeVIreg1(mVU, Is, mVUlow.VI_read[0]);
analyzeVIreg1(It, mVUlow.VI_read[1]); analyzeVIreg1(mVU, It, mVUlow.VI_read[1]);
analyzeVIreg2(Id, mVUlow.VI_write, 1); analyzeVIreg2(mVU, Id, mVUlow.VI_write, 1);
} }
__fi void mVUanalyzeIALU2(mV, int Is, int It) { __fi void mVUanalyzeIALU2(mV, int Is, int It) {
if (!It) { mVUlow.isNOP = 1; } if (!It) mVUlow.isNOP = 1;
analyzeVIreg1(Is, mVUlow.VI_read[0]); analyzeVIreg1(mVU, Is, mVUlow.VI_read[0]);
analyzeVIreg2(It, mVUlow.VI_write, 1); analyzeVIreg2(mVU, It, mVUlow.VI_write, 1);
} }
__fi void mVUanalyzeIADDI(mV, int Is, int It, s16 imm) { __fi void mVUanalyzeIADDI(mV, int Is, int It, s16 imm) {
@ -183,8 +181,8 @@ __fi void mVUanalyzeIADDI(mV, int Is, int It, s16 imm) {
__fi void mVUanalyzeMR32(mV, int Fs, int Ft) { __fi void mVUanalyzeMR32(mV, int Fs, int Ft) {
if (!Ft) { mVUlow.isNOP = 1; } if (!Ft) { mVUlow.isNOP = 1; }
analyzeReg6(Fs, mVUlow.VF_read[0]); analyzeReg6(mVU, Fs, mVUlow.VF_read[0]);
analyzeReg2(Ft, mVUlow.VF_write, 1); analyzeReg2(mVU, Ft, mVUlow.VF_write, 1);
} }
//------------------------------------------------------------------ //------------------------------------------------------------------
@ -192,8 +190,8 @@ __fi void mVUanalyzeMR32(mV, int Fs, int Ft) {
//------------------------------------------------------------------ //------------------------------------------------------------------
__fi void mVUanalyzeFDIV(mV, int Fs, int Fsf, int Ft, int Ftf, u8 xCycles) { __fi void mVUanalyzeFDIV(mV, int Fs, int Fsf, int Ft, int Ftf, u8 xCycles) {
analyzeReg5(Fs, Fsf, mVUlow.VF_read[0]); analyzeReg5(mVU, Fs, Fsf, mVUlow.VF_read[0]);
analyzeReg5(Ft, Ftf, mVUlow.VF_read[1]); analyzeReg5(mVU, Ft, Ftf, mVUlow.VF_read[1]);
analyzeQreg(xCycles); analyzeQreg(xCycles);
} }
@ -202,12 +200,12 @@ __fi void mVUanalyzeFDIV(mV, int Fs, int Fsf, int Ft, int Ftf, u8 xCycles) {
//------------------------------------------------------------------ //------------------------------------------------------------------
__fi void mVUanalyzeEFU1(mV, int Fs, int Fsf, u8 xCycles) { __fi void mVUanalyzeEFU1(mV, int Fs, int Fsf, u8 xCycles) {
analyzeReg5(Fs, Fsf, mVUlow.VF_read[0]); analyzeReg5(mVU, Fs, Fsf, mVUlow.VF_read[0]);
analyzePreg(xCycles); analyzePreg(xCycles);
} }
__fi void mVUanalyzeEFU2(mV, int Fs, u8 xCycles) { __fi void mVUanalyzeEFU2(mV, int Fs, u8 xCycles) {
analyzeReg1(Fs, mVUlow.VF_read[0]); analyzeReg1(mVU, Fs, mVUlow.VF_read[0]);
analyzePreg(xCycles); analyzePreg(xCycles);
} }
@ -216,8 +214,8 @@ __fi void mVUanalyzeEFU2(mV, int Fs, u8 xCycles) {
//------------------------------------------------------------------ //------------------------------------------------------------------
__fi void mVUanalyzeMFP(mV, int Ft) { __fi void mVUanalyzeMFP(mV, int Ft) {
if (!Ft) { mVUlow.isNOP = 1; } if (!Ft) mVUlow.isNOP = 1;
analyzeReg2(Ft, mVUlow.VF_write, 1); analyzeReg2(mVU, Ft, mVUlow.VF_write, 1);
} }
//------------------------------------------------------------------ //------------------------------------------------------------------
@ -225,9 +223,9 @@ __fi void mVUanalyzeMFP(mV, int Ft) {
//------------------------------------------------------------------ //------------------------------------------------------------------
__fi void mVUanalyzeMOVE(mV, int Fs, int Ft) { __fi void mVUanalyzeMOVE(mV, int Fs, int Ft) {
if (!Ft || (Ft == Fs)) { mVUlow.isNOP = 1; } if (!Ft||(Ft == Fs)) mVUlow.isNOP = 1;
analyzeReg1(Fs, mVUlow.VF_read[0]); analyzeReg1(mVU, Fs, mVUlow.VF_read[0]);
analyzeReg2(Ft, mVUlow.VF_write, 1); analyzeReg2(mVU, Ft, mVUlow.VF_write, 1);
} }
//------------------------------------------------------------------ //------------------------------------------------------------------
@ -235,10 +233,10 @@ __fi void mVUanalyzeMOVE(mV, int Fs, int Ft) {
//------------------------------------------------------------------ //------------------------------------------------------------------
__fi void mVUanalyzeLQ(mV, int Ft, int Is, bool writeIs) { __fi void mVUanalyzeLQ(mV, int Ft, int Is, bool writeIs) {
analyzeVIreg1(Is, mVUlow.VI_read[0]); analyzeVIreg1(mVU, Is, mVUlow.VI_read[0]);
analyzeReg2 (Ft, mVUlow.VF_write, 1); analyzeReg2 (mVU, Ft, mVUlow.VF_write, 1);
if (!Ft) { if (writeIs && Is) { mVUlow.noWriteVF = 1; } else { mVUlow.isNOP = 1; } } if (!Ft) { if (writeIs && Is) { mVUlow.noWriteVF = 1; } else { mVUlow.isNOP = 1; } }
if (writeIs) { analyzeVIreg2(Is, mVUlow.VI_write, 1); } if (writeIs) { analyzeVIreg2(mVU, Is, mVUlow.VI_write, 1); }
} }
//------------------------------------------------------------------ //------------------------------------------------------------------
@ -246,9 +244,9 @@ __fi void mVUanalyzeLQ(mV, int Ft, int Is, bool writeIs) {
//------------------------------------------------------------------ //------------------------------------------------------------------
__fi void mVUanalyzeSQ(mV, int Fs, int It, bool writeIt) { __fi void mVUanalyzeSQ(mV, int Fs, int It, bool writeIt) {
analyzeReg1 (Fs, mVUlow.VF_read[0]); analyzeReg1 (mVU, Fs, mVUlow.VF_read[0]);
analyzeVIreg1(It, mVUlow.VI_read[0]); analyzeVIreg1(mVU, It, mVUlow.VI_read[0]);
if (writeIt) { analyzeVIreg2(It, mVUlow.VI_write, 1); } if (writeIt) { analyzeVIreg2(mVU, It, mVUlow.VI_write, 1); }
} }
//------------------------------------------------------------------ //------------------------------------------------------------------
@ -256,13 +254,16 @@ __fi void mVUanalyzeSQ(mV, int Fs, int It, bool writeIt) {
//------------------------------------------------------------------ //------------------------------------------------------------------
__fi void mVUanalyzeR1(mV, int Fs, int Fsf) { __fi void mVUanalyzeR1(mV, int Fs, int Fsf) {
analyzeReg5(Fs, Fsf, mVUlow.VF_read[0]); analyzeReg5(mVU, Fs, Fsf, mVUlow.VF_read[0]);
analyzeRreg(); analyzeRreg();
} }
__fi void mVUanalyzeR2(mV, int Ft, bool canBeNOP) { __fi void mVUanalyzeR2(mV, int Ft, bool canBeNOP) {
if (!Ft) { if (canBeNOP) { mVUlow.isNOP = 1; } else { mVUlow.noWriteVF = 1; } } if (!Ft) {
analyzeReg2(Ft, mVUlow.VF_write, 1); if (canBeNOP) mVUlow.isNOP = 1;
else mVUlow.noWriteVF = 1;
}
analyzeReg2(mVU, Ft, mVUlow.VF_write, 1);
analyzeRreg(); analyzeRreg();
} }
@ -285,7 +286,7 @@ __ri void flagSet(mV, bool setMacFlag) {
__ri void mVUanalyzeSflag(mV, int It) { __ri void mVUanalyzeSflag(mV, int It) {
mVUlow.readFlags = 1; mVUlow.readFlags = 1;
analyzeVIreg2(It, mVUlow.VI_write, 1); analyzeVIreg2(mVU, It, mVUlow.VI_write, 1);
if (!It) { mVUlow.isNOP = 1; } if (!It) { mVUlow.isNOP = 1; }
else { else {
mVUsFlagHack = 0; // Don't Optimize Out Status Flags for this block mVUsFlagHack = 0; // Don't Optimize Out Status Flags for this block
@ -309,8 +310,8 @@ __ri void mVUanalyzeFSSET(mV) {
__ri void mVUanalyzeMflag(mV, int Is, int It) { __ri void mVUanalyzeMflag(mV, int Is, int It) {
mVUlow.readFlags = 1; mVUlow.readFlags = 1;
analyzeVIreg1(Is, mVUlow.VI_read[0]); analyzeVIreg1(mVU, Is, mVUlow.VI_read[0]);
analyzeVIreg2(It, mVUlow.VI_write, 1); analyzeVIreg2(mVU, It, mVUlow.VI_write, 1);
if (!It) { mVUlow.isNOP = 1; } if (!It) { mVUlow.isNOP = 1; }
else { else {
mVUinfo.swapOps = 1; mVUinfo.swapOps = 1;
@ -333,7 +334,7 @@ __fi void mVUanalyzeCflag(mV, int It) {
if (!(mVUpBlock->pState.needExactMatch & 4)) // The only time this should happen is on the first program block if (!(mVUpBlock->pState.needExactMatch & 4)) // The only time this should happen is on the first program block
DevCon.WriteLn(Color_Green, "microVU%d: pState's cFlag Info was expected to be set [%04x]", getIndex, xPC); DevCon.WriteLn(Color_Green, "microVU%d: pState's cFlag Info was expected to be set [%04x]", getIndex, xPC);
} }
analyzeVIreg2(It, mVUlow.VI_write, 1); analyzeVIreg2(mVU, It, mVUlow.VI_write, 1);
} }
//------------------------------------------------------------------ //------------------------------------------------------------------
@ -341,7 +342,7 @@ __fi void mVUanalyzeCflag(mV, int It) {
//------------------------------------------------------------------ //------------------------------------------------------------------
__fi void mVUanalyzeXGkick(mV, int Fs, int xCycles) { __fi void mVUanalyzeXGkick(mV, int Fs, int xCycles) {
analyzeVIreg1(Fs, mVUlow.VI_read[0]); analyzeVIreg1(mVU, Fs, mVUlow.VI_read[0]);
analyzeXGkick1(); analyzeXGkick1();
analyzeXGkick2(xCycles); analyzeXGkick2(xCycles);
// Note: Technically XGKICK should stall on the next instruction, // Note: Technically XGKICK should stall on the next instruction,
@ -453,15 +454,15 @@ __ri int mVUbranchCheck(mV) {
} }
__fi void mVUanalyzeCondBranch1(mV, int Is) { __fi void mVUanalyzeCondBranch1(mV, int Is) {
analyzeVIreg1(Is, mVUlow.VI_read[0]); analyzeVIreg1(mVU, Is, mVUlow.VI_read[0]);
if (!mVUstall && !mVUbranchCheck(mVU)) { if (!mVUstall && !mVUbranchCheck(mVU)) {
analyzeBranchVI(mVU, Is, mVUlow.memReadIs); analyzeBranchVI(mVU, Is, mVUlow.memReadIs);
} }
} }
__fi void mVUanalyzeCondBranch2(mV, int Is, int It) { __fi void mVUanalyzeCondBranch2(mV, int Is, int It) {
analyzeVIreg1(Is, mVUlow.VI_read[0]); analyzeVIreg1(mVU, Is, mVUlow.VI_read[0]);
analyzeVIreg1(It, mVUlow.VI_read[1]); analyzeVIreg1(mVU, It, mVUlow.VI_read[1]);
if (!mVUstall && !mVUbranchCheck(mVU)) { if (!mVUstall && !mVUbranchCheck(mVU)) {
analyzeBranchVI(mVU, Is, mVUlow.memReadIs); analyzeBranchVI(mVU, Is, mVUlow.memReadIs);
analyzeBranchVI(mVU, It, mVUlow.memReadIt); analyzeBranchVI(mVU, It, mVUlow.memReadIt);
@ -471,7 +472,7 @@ __fi void mVUanalyzeCondBranch2(mV, int Is, int It) {
__fi void mVUanalyzeNormBranch(mV, int It, bool isBAL) { __fi void mVUanalyzeNormBranch(mV, int It, bool isBAL) {
mVUbranchCheck(mVU); mVUbranchCheck(mVU);
if (isBAL) { if (isBAL) {
analyzeVIreg2(It, mVUlow.VI_write, 1); analyzeVIreg2(mVU, It, mVUlow.VI_write, 1);
setConstReg(It, bSaveAddr); setConstReg(It, bSaveAddr);
} }
} }
@ -484,9 +485,9 @@ __ri void mVUanalyzeJump(mV, int Is, int It, bool isJALR) {
mVUlow.constJump.regValue = mVUconstReg[Is].regValue; mVUlow.constJump.regValue = mVUconstReg[Is].regValue;
//DevCon.Status("microVU%d: Constant JR/JALR Address Optimization", mVU.index); //DevCon.Status("microVU%d: Constant JR/JALR Address Optimization", mVU.index);
} }
analyzeVIreg1(Is, mVUlow.VI_read[0]); analyzeVIreg1(mVU, Is, mVUlow.VI_read[0]);
if (isJALR) { if (isJALR) {
analyzeVIreg2(It, mVUlow.VI_write, 1); analyzeVIreg2(mVU, It, mVUlow.VI_write, 1);
setConstReg(It, bSaveAddr); setConstReg(It, bSaveAddr);
} }
} }

View File

@ -72,11 +72,11 @@ void mVUsetupRange(microVU& mVU, s32 pc, bool isStartPC) {
deque<microRange>::iterator it(ranges->begin()); deque<microRange>::iterator it(ranges->begin());
for (++it; it != ranges->end(); ++it) { for (++it; it != ranges->end(); ++it) {
if((it[0].start >= rStart) && (it[0].start <= rEnd)) { if((it[0].start >= rStart) && (it[0].start <= rEnd)) {
it[0].end = aMax(it[0].end, rEnd); it[0].end = max(it[0].end, rEnd);
mergedRange = 1; mergedRange = 1;
} }
elif((it[0].end >= rStart) && (it[0].end <= rEnd)) { elif((it[0].end >= rStart) && (it[0].end <= rEnd)) {
it[0].start = aMin(it[0].start, rStart); it[0].start = min(it[0].start, rStart);
mergedRange = 1; mergedRange = 1;
} }
} }
@ -217,7 +217,7 @@ __ri void eBitWarning(mV) {
//------------------------------------------------------------------ //------------------------------------------------------------------
__fi void optimizeReg(u8& rState) { rState = (rState==1) ? 0 : rState; } __fi void optimizeReg(u8& rState) { rState = (rState==1) ? 0 : rState; }
__fi void calcCycles(u8& reg, u8 x) { reg = ((reg > x) ? (reg - x) : 0); } __fi void calcCycles(u8& reg, u8 x) { reg = ((reg > x) ? (reg - x) : 0); }
__fi void tCycles(u8& dest, u8& src) { dest = aMax(dest, src); } __fi void tCycles(u8& dest, u8& src) { dest = max(dest, src); }
__fi void incP(mV) { mVU.p ^= 1; } __fi void incP(mV) { mVU.p ^= 1; }
__fi void incQ(mV) { mVU.q ^= 1; } __fi void incQ(mV) { mVU.q ^= 1; }

View File

@ -717,7 +717,11 @@ mVUop(mVU_ISUBIU) {
//------------------------------------------------------------------ //------------------------------------------------------------------
mVUop(mVU_MFIR) { mVUop(mVU_MFIR) {
pass1 { if (!_Ft_) { mVUlow.isNOP = 1; } analyzeVIreg1(_Is_, mVUlow.VI_read[0]); analyzeReg2(_Ft_, mVUlow.VF_write, 1); } pass1 {
if (!_Ft_) { mVUlow.isNOP = 1; }
analyzeVIreg1(mVU, _Is_, mVUlow.VI_read[0]);
analyzeReg2 (mVU, _Ft_, mVUlow.VF_write, 1);
}
pass2 { pass2 {
const xmm& Ft = mVU.regAlloc->allocReg(-1, _Ft_, _X_Y_Z_W); const xmm& Ft = mVU.regAlloc->allocReg(-1, _Ft_, _X_Y_Z_W);
mVUallocVIa(mVU, gprT1, _Is_, true); mVUallocVIa(mVU, gprT1, _Is_, true);
@ -761,7 +765,11 @@ mVUop(mVU_MR32) {
} }
mVUop(mVU_MTIR) { mVUop(mVU_MTIR) {
pass1 { if (!_It_) { mVUlow.isNOP = 1; } analyzeReg5(_Fs_, _Fsf_, mVUlow.VF_read[0]); analyzeVIreg2(_It_, mVUlow.VI_write, 1); } pass1 {
if (!_It_) mVUlow.isNOP = 1;
analyzeReg5 (mVU, _Fs_, _Fsf_, mVUlow.VF_read[0]);
analyzeVIreg2(mVU, _It_, mVUlow.VI_write, 1);
}
pass2 { pass2 {
const xmm& Fs = mVU.regAlloc->allocReg(_Fs_, 0, (1 << (3 - _Fsf_))); const xmm& Fs = mVU.regAlloc->allocReg(_Fs_, 0, (1 << (3 - _Fsf_)));
xMOVD(gprT1, Fs); xMOVD(gprT1, Fs);
@ -776,7 +784,11 @@ mVUop(mVU_MTIR) {
//------------------------------------------------------------------ //------------------------------------------------------------------
mVUop(mVU_ILW) { mVUop(mVU_ILW) {
pass1 { if (!_It_) { mVUlow.isNOP = 1; } analyzeVIreg1(_Is_, mVUlow.VI_read[0]); analyzeVIreg2(_It_, mVUlow.VI_write, 4); } pass1 {
if (!_It_) mVUlow.isNOP = 1;
analyzeVIreg1(mVU, _Is_, mVUlow.VI_read[0]);
analyzeVIreg2(mVU, _It_, mVUlow.VI_write, 4);
}
pass2 { pass2 {
xAddressVoid ptr(mVU.regs().Mem + offsetSS); xAddressVoid ptr(mVU.regs().Mem + offsetSS);
if (_Is_) { if (_Is_) {
@ -794,7 +806,11 @@ mVUop(mVU_ILW) {
} }
mVUop(mVU_ILWR) { mVUop(mVU_ILWR) {
pass1 { if (!_It_) { mVUlow.isNOP = 1; } analyzeVIreg1(_Is_, mVUlow.VI_read[0]); analyzeVIreg2(_It_, mVUlow.VI_write, 4); } pass1 {
if (!_It_) mVUlow.isNOP = 1;
analyzeVIreg1(mVU, _Is_, mVUlow.VI_read[0]);
analyzeVIreg2(mVU, _It_, mVUlow.VI_write, 4);
}
pass2 { pass2 {
xAddressVoid ptr(mVU.regs().Mem + offsetSS); xAddressVoid ptr(mVU.regs().Mem + offsetSS);
if (_Is_) { if (_Is_) {
@ -813,7 +829,10 @@ mVUop(mVU_ILWR) {
//------------------------------------------------------------------ //------------------------------------------------------------------
mVUop(mVU_ISW) { mVUop(mVU_ISW) {
pass1 { analyzeVIreg1(_Is_, mVUlow.VI_read[0]); analyzeVIreg1(_It_, mVUlow.VI_read[1]); } pass1 {
analyzeVIreg1(mVU, _Is_, mVUlow.VI_read[0]);
analyzeVIreg1(mVU, _It_, mVUlow.VI_read[1]);
}
pass2 { pass2 {
xAddressVoid ptr(mVU.regs().Mem); xAddressVoid ptr(mVU.regs().Mem);
if (_Is_) { if (_Is_) {
@ -834,7 +853,9 @@ mVUop(mVU_ISW) {
} }
mVUop(mVU_ISWR) { mVUop(mVU_ISWR) {
pass1 { analyzeVIreg1(_Is_, mVUlow.VI_read[0]); analyzeVIreg1(_It_, mVUlow.VI_read[1]); } pass1 {
analyzeVIreg1(mVU, _Is_, mVUlow.VI_read[0]);
analyzeVIreg1(mVU, _It_, mVUlow.VI_read[1]); }
pass2 { pass2 {
xAddressVoid ptr(mVU.regs().Mem); xAddressVoid ptr(mVU.regs().Mem);
if (_Is_) { if (_Is_) {
@ -1053,12 +1074,12 @@ mVUop(mVU_RXOR) {
//------------------------------------------------------------------ //------------------------------------------------------------------
mVUop(mVU_WAITP) { mVUop(mVU_WAITP) {
pass1 { mVUstall = aMax(mVUstall, ((mVUregs.p) ? (mVUregs.p - 1) : 0)); } pass1 { mVUstall = max(mVUstall, (u8)((mVUregs.p) ? (mVUregs.p - 1) : 0)); }
pass3 { mVUlog("WAITP"); } pass3 { mVUlog("WAITP"); }
} }
mVUop(mVU_WAITQ) { mVUop(mVU_WAITQ) {
pass1 { mVUstall = aMax(mVUstall, mVUregs.q); } pass1 { mVUstall = max(mVUstall, mVUregs.q); }
pass3 { mVUlog("WAITQ"); } pass3 { mVUlog("WAITQ"); }
} }
@ -1067,7 +1088,10 @@ mVUop(mVU_WAITQ) {
//------------------------------------------------------------------ //------------------------------------------------------------------
mVUop(mVU_XTOP) { mVUop(mVU_XTOP) {
pass1 { if (!_It_) { mVUlow.isNOP = 1; } analyzeVIreg2(_It_, mVUlow.VI_write, 1); } pass1 {
if (!_It_) mVUlow.isNOP = 1;
analyzeVIreg2(mVU, _It_, mVUlow.VI_write, 1);
}
pass2 { pass2 {
xMOVZX(gprT1, ptr16[&mVU.getVifRegs().top]); xMOVZX(gprT1, ptr16[&mVU.getVifRegs().top]);
mVUallocVIb(mVU, gprT1, _It_); mVUallocVIb(mVU, gprT1, _It_);
@ -1076,10 +1100,12 @@ mVUop(mVU_XTOP) {
} }
mVUop(mVU_XITOP) { mVUop(mVU_XITOP) {
pass1 { if (!_It_) { mVUlow.isNOP = 1; } analyzeVIreg2(_It_, mVUlow.VI_write, 1); } pass1 {
if (!_It_) mVUlow.isNOP = 1;
analyzeVIreg2(mVU, _It_, mVUlow.VI_write, 1); }
pass2 { pass2 {
xMOVZX(gprT1, ptr16[&mVU.getVifRegs().itop]); xMOVZX(gprT1, ptr16[&mVU.getVifRegs().itop]);
xAND(gprT1, isVU1 ? 0x3ff : 0xff); xAND (gprT1, isVU1 ? 0x3ff : 0xff);
mVUallocVIb(mVU, gprT1, _It_); mVUallocVIb(mVU, gprT1, _It_);
} }
pass3 { mVUlog("XITOP vi%02d", _Ft_); } pass3 { mVUlog("XITOP vi%02d", _Ft_); }

View File

@ -113,11 +113,11 @@ static const uint divD = 0x2080000;
#define _Fsf_ ((mVU.code >> 21) & 0x03) #define _Fsf_ ((mVU.code >> 21) & 0x03)
#define _Ftf_ ((mVU.code >> 23) & 0x03) #define _Ftf_ ((mVU.code >> 23) & 0x03)
#define _Imm5_ (mVU.Imm5()) #define _Imm5_ ((s16) (((mVU.code & 0x400) ? 0xfff0 : 0) | ((mVU.code >> 6) & 0xf)))
#define _Imm11_ (mVU.Imm11()) #define _Imm11_ ((s32) ((mVU.code & 0x400) ? (0xfffffc00 | (mVU.code & 0x3ff)) : (mVU.code & 0x3ff)))
#define _Imm12_ (mVU.Imm12()) #define _Imm12_ ((u32)((((mVU.code >> 21) & 0x1) << 11) | (mVU.code & 0x7ff)))
#define _Imm15_ (mVU.Imm15()) #define _Imm15_ ((u32) (((mVU.code >> 10) & 0x7800) | (mVU.code & 0x7ff)))
#define _Imm24_ (mVU.Imm24()) #define _Imm24_ ((u32) (mVU.code & 0xffffff))
#define isCOP2 (mVU.cop2 != 0) #define isCOP2 (mVU.cop2 != 0)
#define isVU1 (mVU.index != 0) #define isVU1 (mVU.index != 0)
@ -219,12 +219,6 @@ typedef u32 (__fastcall *mVUCall)(void*, void*);
#define xPC ((iPC / 2) * 8) #define xPC ((iPC / 2) * 8)
#define curI ((u32*)mVU.regs().Micro)[iPC] //mVUcurProg.data[iPC] #define curI ((u32*)mVU.regs().Micro)[iPC] //mVUcurProg.data[iPC]
#define setCode() { mVU.code = curI; } #define setCode() { mVU.code = curI; }
#define incPC(x) (mVU.advancePC(x))
#define branchAddr mVU.getBranchAddr()
#define branchAddrN mVU.getBranchAddrN()
#define incPC2(x) { iPC = ((iPC + (x)) & mVU.progMemMask); }
#define bSaveAddr (((xPC + 16) & (mVU.microMemSize-8)) / 8) #define bSaveAddr (((xPC + 16) & (mVU.microMemSize-8)) / 8)
#define shufflePQ (((mVU.p) ? 0xb0 : 0xe0) | ((mVU.q) ? 0x01 : 0x04)) #define shufflePQ (((mVU.p) ? 0xb0 : 0xe0) | ((mVU.q) ? 0x01 : 0x04))
#define cmpOffset(x) ((u8*)&(((u8*)x)[it[0].start])) #define cmpOffset(x) ((u8*)&(((u8*)x)[it[0].start]))
@ -234,6 +228,20 @@ typedef u32 (__fastcall *mVUCall)(void*, void*);
#define clampE CHECK_VU_EXTRA_OVERFLOW #define clampE CHECK_VU_EXTRA_OVERFLOW
#define elif else if #define elif else if
#define branchAddr ( \
pxAssumeDev((iPC & 1) == 0, "microVU: Expected Lower Op for valid branch addr."), \
((((iPC + 2) + (_Imm11_ * 2)) & mVU.progMemMask) * 4) \
)
#define branchAddrN ( \
pxAssumeDev((iPC & 1) == 0, "microVU: Expected Lower Op for valid branch addr."), \
((((iPC + 4) + (_Imm11_ * 2)) & mVU.progMemMask) * 4) \
)
// Fetches the PC and instruction opcode relative to the current PC. Used to rewind and
// fast-forward the IR state while calculating VU pipeline conditions (branches, writebacks, etc)
#define incPC(x) { iPC = ((iPC + (x)) & mVU.progMemMask); mVU.code = curI; }
#define incPC2(x) { iPC = ((iPC + (x)) & mVU.progMemMask); }
// Flag Info (Set if next-block's first 4 ops will read current-block's flags) // Flag Info (Set if next-block's first 4 ops will read current-block's flags)
#define __Status (mVUregs.needExactMatch & 1) #define __Status (mVUregs.needExactMatch & 1)
#define __Mac (mVUregs.needExactMatch & 2) #define __Mac (mVUregs.needExactMatch & 2)