microVU/VU Int: Implemented T-Bit/D-Bit handling on microVU.

-Dreamworks Over The Hedge actually requires this to load.  Interpreter kind of had it right, but didn't stop the VU on these flags, causing the graphics to look bad and hang on the start button, works fine now.

git-svn-id: http://pcsx2.googlecode.com/svn/trunk@5583 96395faa-99c1-11dd-bbfe-3dabce05a288
This commit is contained in:
refraction@gmail.com 2013-03-09 11:03:59 +00:00
parent 2e363b376e
commit 8d0f6701b7
6 changed files with 35 additions and 4 deletions

View File

@ -553,7 +553,7 @@ extern void setDmacStat(u32 num);
extern tDMA_TAG *SPRdmaGetAddr(u32 addr, bool write); extern tDMA_TAG *SPRdmaGetAddr(u32 addr, bool write);
extern tDMA_TAG *dmaGetAddr(u32 addr, bool write); extern tDMA_TAG *dmaGetAddr(u32 addr, bool write);
extern void hwIntcIrq(int n); extern void __fastcall hwIntcIrq(int n);
extern void hwDmacIrq(int n); extern void hwDmacIrq(int n);
extern void FireMFIFOEmpty(); extern void FireMFIFOEmpty();

View File

@ -129,7 +129,8 @@ __fi uint dmacInterrupt()
return 0x800; return 0x800;
} }
void hwIntcIrq(int n) //Fastcall required for VU recompilers. Test Over the Hedge when changing.
void __fastcall hwIntcIrq(int n)
{ {
psHu32(INTC_STAT) |= 1<<n; psHu32(INTC_STAT) |= 1<<n;
if(psHu32(INTC_MASK) & (1<<n))cpuTestINTCInts(); if(psHu32(INTC_MASK) & (1<<n))cpuTestINTCInts();

View File

@ -62,12 +62,14 @@ static void _vu0Exec(VURegs* VU)
VU0.VI[REG_VPU_STAT].UL|= 0x2; VU0.VI[REG_VPU_STAT].UL|= 0x2;
hwIntcIrq(INTC_VU0); hwIntcIrq(INTC_VU0);
} }
VU->ebit = 1;
} }
if (ptr[1] & 0x08000000) { /* T flag */ if (ptr[1] & 0x08000000) { /* T flag */
if (VU0.VI[REG_FBRST].UL & 0x8) { if (VU0.VI[REG_FBRST].UL & 0x8) {
VU0.VI[REG_VPU_STAT].UL|= 0x4; VU0.VI[REG_VPU_STAT].UL|= 0x4;
hwIntcIrq(INTC_VU0); hwIntcIrq(INTC_VU0);
} }
VU->ebit = 1;
} }
VU->code = ptr[1]; VU->code = ptr[1];

View File

@ -60,12 +60,14 @@ static void _vu1Exec(VURegs* VU)
VU0.VI[REG_VPU_STAT].UL|= 0x200; VU0.VI[REG_VPU_STAT].UL|= 0x200;
hwIntcIrq(INTC_VU1); hwIntcIrq(INTC_VU1);
} }
VU->ebit = 1;
} }
if (ptr[1] & 0x08000000) { /* T flag */ if (ptr[1] & 0x08000000) { /* T flag */
if (VU0.VI[REG_FBRST].UL & 0x800) { if (VU0.VI[REG_FBRST].UL & 0x800) {
VU0.VI[REG_VPU_STAT].UL|= 0x400; VU0.VI[REG_VPU_STAT].UL|= 0x400;
hwIntcIrq(INTC_VU1); hwIntcIrq(INTC_VU1);
} }
VU->ebit = 1;
} }
//VUM_LOG("VU->cycle = %d (flags st=%x;mac=%x;clip=%x,q=%f)", VU->cycle, VU->statusflag, VU->macflag, VU->clipflag, VU->q.F); //VUM_LOG("VU->cycle = %d (flags st=%x;mac=%x;clip=%x,q=%f)", VU->cycle, VU->statusflag, VU->macflag, VU->clipflag, VU->q.F);

View File

@ -479,6 +479,26 @@ void* mVUcompileSingleInstruction(microVU& mVU, u32 startPC, uptr pState, microF
return thisPtr; return thisPtr;
} }
void mVUDoDBit(microVU& mVU)
{
xTEST(ptr32[&VU0.VI[REG_FBRST].UL], (isVU1 ? 0x400 : 0x4));
xForwardJump32 eJMP(Jcc_Zero);
xOR(ptr32[&VU0.VI[REG_VPU_STAT].UL], (isVU1 ? 0x200 : 0x2));
xMOV( gprT2, (isVU1 ? 7 : 6));
xCALL( hwIntcIrq );
eJMP.SetTarget();
}
void mVUDoTBit(microVU& mVU)
{
xTEST(ptr32[&VU0.VI[REG_FBRST].UL], (isVU1 ? 0x800 : 0x8));
xForwardJump32 eJMP(Jcc_Zero);
xOR(ptr32[&VU0.VI[REG_VPU_STAT].UL], (isVU1 ? 0x400 : 0x4));
xMOV( gprT2, (isVU1 ? 7 : 6));
xCALL( hwIntcIrq );
eJMP.SetTarget();
}
void* mVUcompile(microVU& mVU, u32 startPC, uptr pState) { void* mVUcompile(microVU& mVU, u32 startPC, uptr pState) {
microFlagCycles mFC; microFlagCycles mFC;
@ -498,8 +518,9 @@ void* mVUcompile(microVU& mVU, u32 startPC, uptr pState) {
mVUopU(mVU, 0); mVUopU(mVU, 0);
mVUcheckBadOp(mVU); mVUcheckBadOp(mVU);
if (curI & _Ebit_) { eBitPass1(mVU, branch); } if (curI & _Ebit_) { eBitPass1(mVU, branch); }
if (curI & _DTbit_) { branch = 4; } if (curI & _Dbit_) { mVUup.dBit = 1; branch = 2; }
if (curI & _Mbit_) { mVUup.mBit = 1; } if (curI & _Mbit_) { mVUup.mBit = 1; }
if (curI & _Tbit_) { mVUup.tBit = 1; branch = 2; }
if (curI & _Ibit_) { mVUlow.isNOP = 1; mVUup.iBit = 1; } if (curI & _Ibit_) { mVUlow.isNOP = 1; mVUup.iBit = 1; }
else { incPC(-1); mVUopL(mVU, 0); incPC(1); } else { incPC(-1); mVUopL(mVU, 0); incPC(1); }
mVUsetCycles(mVU); mVUsetCycles(mVU);
@ -554,7 +575,10 @@ void* mVUcompile(microVU& mVU, u32 startPC, uptr pState) {
} }
} }
if ((x == endCount) && (x!=1)) { Console.Error("microVU%d: Possible infinite compiling loop!", mVU.index); } if ((x == endCount) && (x!=1)) { Console.Error("microVU%d: Possible infinite compiling loop!", mVU.index); }
incPC(-1);
if(mVUup.tBit) mVUDoTBit(mVU);
if(mVUup.dBit) mVUDoDBit(mVU);
incPC(1);
// E-bit End // E-bit End
mVUsetupRange(mVU, xPC-8, 0); mVUsetupRange(mVU, xPC-8, 0);
mVUendProgram(mVU, &mFC, 1); mVUendProgram(mVU, &mFC, 1);

View File

@ -112,6 +112,8 @@ struct microUpperOp {
bool eBit; // Has E-bit set bool eBit; // Has E-bit set
bool iBit; // Has I-bit set bool iBit; // Has I-bit set
bool mBit; // Has M-bit set bool mBit; // Has M-bit set
bool tBit; // Has T-bit set
bool dBit; // Has D-bit set
microVFreg VF_write; // VF Vectors written to by this instruction microVFreg VF_write; // VF Vectors written to by this instruction
microVFreg VF_read[2]; // VF Vectors read by this instruction microVFreg VF_read[2]; // VF Vectors read by this instruction
}; };