diff --git a/pcsx2/Gif.cpp b/pcsx2/Gif.cpp index 38ec6abaaf..986fb30d2d 100644 --- a/pcsx2/Gif.cpp +++ b/pcsx2/Gif.cpp @@ -29,38 +29,48 @@ __aligned16 GIF_Fifo gif_fifo; __aligned16 gifStruct gif; -static __fi void GifDMAInt(int cycles) { - if (dmacRegs.ctrl.MFD == MFD_GIF) { +static __fi void GifDMAInt(int cycles) +{ + if (dmacRegs.ctrl.MFD == MFD_GIF) + { if (!(cpuRegs.interrupt & (1 << DMAC_MFIFO_GIF)) || cpuRegs.eCycle[DMAC_MFIFO_GIF] < (u32)cycles) { CPU_INT(DMAC_MFIFO_GIF, cycles); } - } else if (!(cpuRegs.interrupt & (1 << DMAC_GIF)) || cpuRegs.eCycle[DMAC_GIF] < (u32)cycles) + } + else if (!(cpuRegs.interrupt & (1 << DMAC_GIF)) || cpuRegs.eCycle[DMAC_GIF] < (u32)cycles) { CPU_INT(DMAC_GIF, cycles); } } -__fi void clearFIFOstuff(bool full) { - CSRreg.FIFO = full ? CSR_FIFO_FULL : CSR_FIFO_EMPTY; +__fi void clearFIFOstuff(bool full) +{ + CSRreg.FIFO = full ? CSR_FIFO_FULL : CSR_FIFO_EMPTY; } //I suspect this is GS side which should really be handled by GS which also doesn't current have a fifo, but we can guess from our fifo -static __fi void CalculateFIFOCSR() { - if (gifRegs.stat.FQC >= 15) { +static __fi void CalculateFIFOCSR() +{ + if (gifRegs.stat.FQC >= 15) + { CSRreg.FIFO = CSR_FIFO_FULL; } - else if (gifRegs.stat.FQC == 0) { + else if (gifRegs.stat.FQC == 0) + { CSRreg.FIFO = CSR_FIFO_EMPTY; } - else { + else + { CSRreg.FIFO = CSR_FIFO_NORMAL; } } -bool CheckPaths() { +bool CheckPaths() +{ // Can't do Path 3, so try dma again later... - if (!gifUnit.CanDoPath3()) { + if (!gifUnit.CanDoPath3()) + { if (!gifUnit.Path3Masked()) { //DevCon.Warning("Path3 stalled APATH %x PSE %x DIR %x Signal %x", gifRegs.stat.APATH, gifRegs.stat.PSE, gifRegs.stat.DIR, gifUnit.gsSIGNAL.queued); @@ -83,13 +93,13 @@ void GIF_Fifo::init() gif.gscycles = 0; gif.prevcycles = 0; gif.mfifocycles = 0; - } int GIF_Fifo::write_fifo(u32* pMem, int size) { - if (fifoSize == 16) { + if (fifoSize == 16) + { //GIF_LOG("GIF FIFO Full"); return 0; } @@ -130,7 +140,8 @@ int GIF_Fifo::read_fifo() GIF_LOG("GIF FIFO Read %d QW from FIFO Current Size %d", sizeRead, fifoSize); - if (sizeRead < fifoSize) { + if (sizeRead < fifoSize) + { if (sizeRead > 0) { int copyAmount = fifoSize - sizeRead; @@ -160,16 +171,20 @@ int GIF_Fifo::read_fifo() return sizeRead; } -void incGifChAddr(u32 qwc) { - if (gifch.chcr.STR) { +void incGifChAddr(u32 qwc) +{ + if (gifch.chcr.STR) + { gifch.madr += qwc * 16; - gifch.qwc -= qwc; + gifch.qwc -= qwc; hwDmacSrcTadrInc(gifch); } - else DevCon.Error("incGifAddr() Error!"); + else + DevCon.Error("incGifAddr() Error!"); } -__fi void gifCheckPathStatus() { +__fi void gifCheckPathStatus() +{ if (gifRegs.stat.APATH == 3) { @@ -177,7 +192,8 @@ __fi void gifCheckPathStatus() { gifRegs.stat.OPH = 0; if (gifUnit.gifPath[GIF_PATH_3].state == GIF_PATH_IDLE || gifUnit.gifPath[GIF_PATH_3].state == GIF_PATH_WAIT) { - if (gifUnit.checkPaths(1, 1, 0)) gifUnit.Execute(false, true); + if (gifUnit.checkPaths(1, 1, 0)) + gifUnit.Execute(false, true); } } @@ -191,34 +207,37 @@ __fi void gifInterrupt() GIF_LOG("gifInterrupt caught qwc=%d fifo=%d apath=%d oph=%d state=%d!", gifch.qwc, gifRegs.stat.FQC, gifRegs.stat.APATH, gifRegs.stat.OPH, gifUnit.gifPath[GIF_PATH_3].state); gifCheckPathStatus(); - if(gifUnit.gifPath[GIF_PATH_3].state == GIF_PATH_IDLE) + if (gifUnit.gifPath[GIF_PATH_3].state == GIF_PATH_IDLE) { - if(vif1Regs.stat.VGW) + if (vif1Regs.stat.VGW) { // Check if VIF is in a cycle or is currently "idle" waiting for GIF to come back. - if(!(cpuRegs.interrupt & (1< 0) || (!gif.gspath3done)) { - if (!dmacRegs.ctrl.DMAE) { + if ((gifch.qwc > 0) || (!gif.gspath3done)) + { + if (!dmacRegs.ctrl.DMAE) + { Console.Warning("gs dma masked, re-scheduling..."); // Re-raise the int shortly in the future - GifDMAInt( 64 ); + GifDMAInt(64); return; } GIFdma(); @@ -262,17 +285,18 @@ __fi void gifInterrupt() } gif.gscycles = 0; - gifch.chcr.STR = false; + gifch.chcr.STR = false; gifRegs.stat.FQC = gif_fifo.fifoSize; CalculateFIFOCSR(); hwDmacIrq(DMAC_GIF); - if(gif_fifo.fifoSize) + if (gif_fifo.fifoSize) GifDMAInt(8 * BIAS); GIF_LOG("GIF DMA End QWC in fifo %x APATH = %x OPH = %x state = %x", gifRegs.stat.FQC, gifRegs.stat.APATH, gifRegs.stat.OPH, gifUnit.gifPath[GIF_PATH_3].state); } -static u32 WRITERING_DMA(u32 *pMem, u32 qwc) { +static u32 WRITERING_DMA(u32* pMem, u32 qwc) +{ u32 originalQwc = qwc; if (gifRegs.stat.IMT) @@ -282,7 +306,7 @@ static u32 WRITERING_DMA(u32 *pMem, u32 qwc) { // so we can get away with transferring "most" of it when it's a big packet. // Use Wallace and Gromit Project Zoo or The Suffering for testing if (qwc > 64) - qwc = qwc-64; + qwc = qwc - 64; else qwc = std::min(qwc, 8u); } @@ -305,11 +329,13 @@ static u32 WRITERING_DMA(u32 *pMem, u32 qwc) { return size; } -static __fi void GIFchain() { +static __fi void GIFchain() +{ tDMA_TAG* pMem; pMem = dmaGetAddr(gifch.madr, false); - if (pMem == NULL) { + if (pMem == NULL) + { // Must increment madr and clear qwc, else it loops gifch.madr += gifch.qwc * 16; gifch.qwc = 0; @@ -317,16 +343,17 @@ static __fi void GIFchain() { return; } - int transferred= WRITERING_DMA((u32*)pMem, gifch.qwc); + int transferred = WRITERING_DMA((u32*)pMem, gifch.qwc); gif.gscycles += transferred * BIAS; if (!gifUnit.Path3Masked() || (gif_fifo.fifoSize < 16)) GifDMAInt(gif.gscycles); } -static __fi bool checkTieBit(tDMA_TAG* &ptag) +static __fi bool checkTieBit(tDMA_TAG*& ptag) { - if (gifch.chcr.TIE && ptag->IRQ) { + if (gifch.chcr.TIE && ptag->IRQ) + { GIF_LOG("dmaIrq Set"); gif.gspath3done = true; return true; @@ -336,12 +363,13 @@ static __fi bool checkTieBit(tDMA_TAG* &ptag) static __fi tDMA_TAG* ReadTag() { - tDMA_TAG* ptag = dmaGetAddr(gifch.tadr, false); // Set memory pointer to TADR + tDMA_TAG* ptag = dmaGetAddr(gifch.tadr, false); // Set memory pointer to TADR - if (!(gifch.transfer("Gif", ptag))) return NULL; + if (!(gifch.transfer("Gif", ptag))) + return NULL; - gifch.madr = ptag[1]._u32; // MADR = ADDR field + SPR - gif.gscycles += 2; // Add 1 cycles from the QW read for the tag + gifch.madr = ptag[1]._u32; // MADR = ADDR field + SPR + gif.gscycles += 2; // Add 1 cycles from the QW read for the tag gif.gspath3done = hwDmacSrcChainWithStack(gifch, ptag->ID); return ptag; @@ -349,7 +377,7 @@ static __fi tDMA_TAG* ReadTag() static __fi tDMA_TAG* ReadTag2() { - tDMA_TAG* ptag = dmaGetAddr(gifch.tadr, false); // Set memory pointer to TADR + tDMA_TAG* ptag = dmaGetAddr(gifch.tadr, false); // Set memory pointer to TADR gifch.unsafeTransfer(ptag); gifch.madr = ptag[1]._u32; @@ -360,19 +388,23 @@ static __fi tDMA_TAG* ReadTag2() void GIFdma() { - while (gifch.qwc > 0 || !gif.gspath3done) { + while (gifch.qwc > 0 || !gif.gspath3done) + { tDMA_TAG* ptag; gif.gscycles = gif.prevcycles; - if (gifRegs.ctrl.PSE) { // Temporarily stop + if (gifRegs.ctrl.PSE) + { // Temporarily stop Console.WriteLn("Gif dma temp paused? (non MFIFO GIF)"); GifDMAInt(16); return; } - if ((dmacRegs.ctrl.STD == STD_GIF) && (gif.prevcycles != 0)) { + if ((dmacRegs.ctrl.STD == STD_GIF) && (gif.prevcycles != 0)) + { //Console.WriteLn("GS Stall Control Source = %x, Drain = %x\n MADR = %x, STADR = %x", (psHu32(0xe000) >> 4) & 0x3, (psHu32(0xe000) >> 6) & 0x3, gifch.madr, psHu32(DMAC_STADR)); - if ((gifch.madr + (gifch.qwc * 16)) > dmacRegs.stadr.ADDR) { + if ((gifch.madr + (gifch.qwc * 16)) > dmacRegs.stadr.ADDR) + { GifDMAInt(4); gif.gscycles = 0; return; @@ -384,7 +416,8 @@ void GIFdma() if ((gifch.chcr.MOD == CHAIN_MODE) && (!gif.gspath3done) && gifch.qwc == 0) // Chain Mode { ptag = ReadTag(); - if (ptag == NULL) return; + if (ptag == NULL) + return; //DevCon.Warning("GIF Reading Tag MSK = %x", vif1Regs.mskpath3); GIF_LOG("gifdmaChain %8.8x_%8.8x size=%d, id=%d, addr=%lx tadr=%lx", ptag[1]._u32, ptag[0]._u32, gifch.qwc, ptag->ID, gifch.madr, gifch.tadr); gifRegs.stat.FQC = std::min((u32)0x10, gifch.qwc); @@ -418,7 +451,7 @@ void GIFdma() // Transfer Dn_QWC from Dn_MADR to GIF if (gifch.qwc > 0) // Normal Mode { - GIFchain(); // Transfers the data set by the switch + GIFchain(); // Transfers the data set by the switch return; } } @@ -433,13 +466,16 @@ void dmaGIF() gif.gspath3done = false; // For some reason this doesn't clear? So when the system starts the thread, we will clear it :) - if (gifch.chcr.MOD == NORMAL_MODE) { // Else it really is a normal transfer and we want to quit, else it gets confused with chains + if (gifch.chcr.MOD == NORMAL_MODE) + { // Else it really is a normal transfer and we want to quit, else it gets confused with chains gif.gspath3done = true; } - if(gifch.chcr.MOD == CHAIN_MODE && gifch.qwc > 0) { + if (gifch.chcr.MOD == CHAIN_MODE && gifch.qwc > 0) + { //DevCon.Warning(L"GIF QWC on Chain " + gifch.chcr.desc()); - if ((gifch.chcr.tag().ID == TAG_REFE) || (gifch.chcr.tag().ID == TAG_END) || (gifch.chcr.tag().IRQ && gifch.chcr.TIE)) { + if ((gifch.chcr.tag().ID == TAG_REFE) || (gifch.chcr.tag().ID == TAG_END) || (gifch.chcr.tag().IRQ && gifch.chcr.TIE)) + { gif.gspath3done = true; } } @@ -453,17 +489,19 @@ static u32 QWCinGIFMFIFO(u32 DrainADDR) SPR_LOG("GIF MFIFO Requesting %x QWC from the MFIFO Base %x, SPR MADR %x Drain %x", gifch.qwc, dmacRegs.rbor.ADDR, spr0ch.madr, DrainADDR); // Calculate what we have in the fifo. - if (DrainADDR <= spr0ch.madr) { + if (DrainADDR <= spr0ch.madr) + { // Drain is below the write position, calculate the difference between them ret = (spr0ch.madr - DrainADDR) >> 4; } - else { + else + { u32 limit = dmacRegs.rbor.ADDR + dmacRegs.rbsr.RMSK + 16; - // Drain is higher than SPR so it has looped round, + // Drain is higher than SPR so it has looped round, // calculate from base to the SPR tag addr and what is left in the top of the ring ret = ((spr0ch.madr - dmacRegs.rbor.ADDR) + (limit - DrainADDR)) >> 4; } - if (ret == 0) + if (ret == 0) gif.gifstate = GIF_STATE_EMPTY; SPR_LOG("%x Available of the %x requested", ret, gifch.qwc); @@ -478,7 +516,8 @@ static __fi bool mfifoGIFrbTransfer() return true; u8* src = (u8*)PSM(gifch.madr); - if (src == NULL) return false; + if (src == NULL) + return false; u32 MFIFOUntilEnd = ((dmacRegs.rbor.ADDR + dmacRegs.rbsr.RMSK + 16) - gifch.madr) >> 4; bool needWrap = MFIFOUntilEnd < qwc; @@ -497,7 +536,8 @@ static __fi bool mfifoGIFrbTransfer() uint secondTransQWC = qwc - MFIFOUntilEnd; src = (u8*)PSM(dmacRegs.rbor.ADDR); - if (src == NULL) return false; + if (src == NULL) + return false; transferred2 = WRITERING_DMA((u32*)src, secondTransQWC); // Second part @@ -520,7 +560,8 @@ static __fi void mfifoGIFchain() if ((gifch.madr & ~dmacRegs.rbsr.RMSK) == dmacRegs.rbor.ADDR) { - if (QWCinGIFMFIFO(gifch.madr) == 0) { + if (QWCinGIFMFIFO(gifch.madr) == 0) + { SPR_LOG("GIF FIFO EMPTY before transfer"); gif.gifstate = GIF_STATE_EMPTY; gif.mfifocycles += 4; @@ -540,10 +581,11 @@ static __fi void mfifoGIFchain() gifch.madr = dmacRegs.rbor.ADDR + (gifch.madr & dmacRegs.rbsr.RMSK); gifch.tadr = gifch.madr; } - else { + else + { SPR_LOG("Non-MFIFO Location transfer doing %x Total QWC", gifch.qwc); - tDMA_TAG *pMem = dmaGetAddr(gifch.madr, false); - if (pMem == NULL) + tDMA_TAG* pMem = dmaGetAddr(gifch.madr, false); + if (pMem == NULL) { gif.mfifocycles += 4; gifch.qwc = 0; @@ -557,27 +599,29 @@ static __fi void mfifoGIFchain() return; } -static u32 qwctag(u32 mask) { +static u32 qwctag(u32 mask) +{ return (dmacRegs.rbor.ADDR + (mask & dmacRegs.rbsr.RMSK)); } void mfifoGifMaskMem(int id) { - switch (id) { + switch (id) + { // These five transfer data following the tag, need to check its within the buffer (Front Mission 4) case TAG_CNT: case TAG_NEXT: - case TAG_CALL: + case TAG_CALL: case TAG_RET: case TAG_END: - if(gifch.madr < dmacRegs.rbor.ADDR) // Probably not needed but we will check anyway. + if (gifch.madr < dmacRegs.rbor.ADDR) // Probably not needed but we will check anyway. { SPR_LOG("GIF MFIFO MADR below bottom of ring buffer, wrapping GIF MADR = %x Ring Bottom %x", gifch.madr, dmacRegs.rbor.ADDR); gifch.madr = qwctag(gifch.madr); - } else - if(gifch.madr > (dmacRegs.rbor.ADDR + (u32)dmacRegs.rbsr.RMSK)) // Usual scenario is the tag is near the end (Front Mission 4) + } + else if (gifch.madr > (dmacRegs.rbor.ADDR + (u32)dmacRegs.rbsr.RMSK)) // Usual scenario is the tag is near the end (Front Mission 4) { - SPR_LOG("GIF MFIFO MADR outside top of ring buffer, wrapping GIF MADR = %x Ring Top %x", gifch.madr, (dmacRegs.rbor.ADDR + dmacRegs.rbsr.RMSK)+16); + SPR_LOG("GIF MFIFO MADR outside top of ring buffer, wrapping GIF MADR = %x Ring Top %x", gifch.madr, (dmacRegs.rbor.ADDR + dmacRegs.rbsr.RMSK) + 16); gifch.madr = qwctag(gifch.madr); } break; @@ -589,19 +633,22 @@ void mfifoGifMaskMem(int id) void mfifoGIFtransfer() { - tDMA_TAG *ptag; + tDMA_TAG* ptag; gif.mfifocycles = 0; - if (gifRegs.ctrl.PSE) { // Temporarily stop + if (gifRegs.ctrl.PSE) + { // Temporarily stop Console.WriteLn("Gif dma temp paused?"); CPU_INT(DMAC_MFIFO_GIF, 16); return; } - if (gifch.qwc == 0) { + if (gifch.qwc == 0) + { gifch.tadr = qwctag(gifch.tadr); - if (QWCinGIFMFIFO(gifch.tadr) == 0) { + if (QWCinGIFMFIFO(gifch.tadr) == 0) + { SPR_LOG("GIF FIFO EMPTY before tag read"); gif.gifstate = GIF_STATE_EMPTY; GifDMAInt(4); @@ -618,7 +665,7 @@ void mfifoGIFtransfer() gif.mfifocycles += 2; GIF_LOG("dmaChain %8.8x_%8.8x size=%d, id=%d, madr=%lx, tadr=%lx mfifo qwc = %x spr0 madr = %x", - ptag[1]._u32, ptag[0]._u32, gifch.qwc, ptag->ID, gifch.madr, gifch.tadr, gif.gifqwc, spr0ch.madr); + ptag[1]._u32, ptag[0]._u32, gifch.qwc, ptag->ID, gifch.madr, gifch.tadr, gif.gifqwc, spr0ch.madr); gif.gspath3done = hwDmacSrcChainWithStack(gifch, ptag->ID); @@ -630,14 +677,15 @@ void mfifoGIFtransfer() gifch.tadr = qwctag(gifch.tadr); - if ((gifch.chcr.TIE) && (ptag->IRQ)) { + if ((gifch.chcr.TIE) && (ptag->IRQ)) + { SPR_LOG("dmaIrq Set"); gif.gspath3done = true; } - } + } mfifoGIFchain(); - + GifDMAInt(std::max(gif.mfifocycles, (u32)4)); SPR_LOG("mfifoGIFtransfer end %x madr %x, tadr %x", gifch.chcr._u32, gifch.madr, gifch.tadr); @@ -648,7 +696,8 @@ void gifMFIFOInterrupt() //DevCon.Warning("gifMFIFOInterrupt"); gif.mfifocycles = 0; - if (dmacRegs.ctrl.MFD != MFD_GIF) { // GIF not in MFIFO anymore, come out. + if (dmacRegs.ctrl.MFD != MFD_GIF) + { // GIF not in MFIFO anymore, come out. DevCon.WriteLn("GIF Leaving MFIFO - Report if any errors"); gifInterrupt(); return; @@ -667,14 +716,16 @@ void gifMFIFOInterrupt() // Make sure it loops if the GIF packet is empty to prepare for the next packet // or end if it was the end of a packet. // This must trigger after VIF retriggers as VIf might instantly mask Path3 - if (!gifUnit.Path3Masked() || gifch.qwc == 0) { + if (!gifUnit.Path3Masked() || gifch.qwc == 0) + { GifDMAInt(16); } return; } } - if (gifUnit.gsSIGNAL.queued) { + if (gifUnit.gsSIGNAL.queued) + { GifDMAInt(128); return; } @@ -692,7 +743,8 @@ void gifMFIFOInterrupt() // Make sure it loops if the GIF packet is empty to prepare for the next packet // or end if it was the end of a packet. // This must trigger after VIF retriggers as VIf might instantly mask Path3 - if (!gifUnit.Path3Masked() || gifch.qwc == 0) { + if (!gifUnit.Path3Masked() || gifch.qwc == 0) + { GifDMAInt(16); } return; @@ -714,7 +766,8 @@ void gifMFIFOInterrupt() if (vif1Regs.stat.VGW) { //Check if VIF is in a cycle or is currently "idle" waiting for GIF to come back. - if (!(cpuRegs.interrupt & (1 << DMAC_VIF1))) { + if (!(cpuRegs.interrupt & (1 << DMAC_VIF1))) + { CPU_INT(DMAC_VIF1, 1); } } @@ -727,7 +780,8 @@ void gifMFIFOInterrupt() if (!gifch.chcr.STR) return; - if (spr0ch.madr == gifch.tadr || (gif.gifstate & GIF_STATE_EMPTY)) { + if (spr0ch.madr == gifch.tadr || (gif.gifstate & GIF_STATE_EMPTY)) + { gif.gifstate = GIF_STATE_EMPTY; // In case of madr = tadr we need to set it FireMFIFOEmpty(); @@ -735,14 +789,15 @@ void gifMFIFOInterrupt() return; } - if (gifch.qwc > 0 || !gif.gspath3done) { + if (gifch.qwc > 0 || !gif.gspath3done) + { mfifoGIFtransfer(); return; } gif.gscycles = 0; - + gifch.chcr.STR = false; gif.gifstate = GIF_STATE_READY; gifRegs.stat.FQC = gif_fifo.fifoSize; @@ -754,7 +809,8 @@ void gifMFIFOInterrupt() DMA_LOG("GIF MFIFO DMA End"); } -void SaveStateBase::gifDmaFreeze() { +void SaveStateBase::gifDmaFreeze() +{ // Note: mfifocycles is not a persistent var, so no need to save it here. FreezeTag("GIFdma"); Freeze(gif.gifstate); diff --git a/pcsx2/Vif.cpp b/pcsx2/Vif.cpp index 9947993cb7..ff53b35cf2 100644 --- a/pcsx2/Vif.cpp +++ b/pcsx2/Vif.cpp @@ -23,7 +23,7 @@ #include "MTVU.h" #include "Gif_Unit.h" -__aligned16 vifStruct vif0, vif1; +__aligned16 vifStruct vif0, vif1; void vif0Reset() { @@ -46,7 +46,7 @@ void vif1Reset() void SaveStateBase::vif0Freeze() { FreezeTag("VIF0dma"); - + Freeze(g_vif0Cycles); Freeze(vif0); @@ -58,7 +58,7 @@ void SaveStateBase::vif0Freeze() void SaveStateBase::vif1Freeze() { FreezeTag("VIF1dma"); - + Freeze(g_vif1Cycles); Freeze(vif1); @@ -71,7 +71,8 @@ void SaveStateBase::vif1Freeze() // Vif0/Vif1 Write32 //------------------------------------------------------------------ -__fi void vif0FBRST(u32 value) { +__fi void vif0FBRST(u32 value) +{ VIF_LOG("VIF0_FBRST write32 0x%8.8x", value); if (value & 0x1) // Reset Vif. @@ -80,7 +81,7 @@ __fi void vif0FBRST(u32 value) { u128 SaveCol; u128 SaveRow; - // if(vif0ch.chcr.STR) DevCon.Warning("FBRST While Vif0 active"); + // if(vif0ch.chcr.STR) DevCon.Warning("FBRST While Vif0 active"); //Must Preserve Row/Col registers! (Downhill Domination for testing) SaveCol._u64[0] = vif0.MaskCol._u64[0]; SaveCol._u64[1] = vif0.MaskCol._u64[1]; @@ -135,19 +136,21 @@ __fi void vif0FBRST(u32 value) { cancel = true; vif0Regs.stat.clear_flags(VIF0_STAT_VSS | VIF0_STAT_VFS | VIF0_STAT_VIS | - VIF0_STAT_INT | VIF0_STAT_ER0 | VIF0_STAT_ER1); + VIF0_STAT_INT | VIF0_STAT_ER0 | VIF0_STAT_ER1); if (cancel) - { - g_vif0Cycles = 0; - // loop necessary for spiderman - if(vif0ch.chcr.STR) CPU_INT(DMAC_VIF0, 0); // Gets the timing right - Flatout + { + g_vif0Cycles = 0; + // loop necessary for spiderman + if (vif0ch.chcr.STR) + CPU_INT(DMAC_VIF0, 0); // Gets the timing right - Flatout } } } -__fi void vif1FBRST(u32 value) { +__fi void vif1FBRST(u32 value) +{ VIF_LOG("VIF1_FBRST write32 0x%8.8x", value); - + if (FBRST(value).RST) // Reset Vif. { u128 SaveCol; @@ -168,7 +171,7 @@ __fi void vif1FBRST(u32 value) { GUNIT_WARN(Color_Red, "VIF FBRST Reset MSK = %x", vif1Regs.mskpath3); vif1Regs.mskpath3 = false; - gifRegs.stat.M3P = 0; + gifRegs.stat.M3P = 0; vif1Regs.err.reset(); vif1.inprogress = mfifo_empty; vif1.cmd = 0; @@ -211,41 +214,46 @@ __fi void vif1FBRST(u32 value) { } vif1Regs.stat.clear_flags(VIF1_STAT_VSS | VIF1_STAT_VFS | VIF1_STAT_VIS | - VIF1_STAT_INT | VIF1_STAT_ER0 | VIF1_STAT_ER1); + VIF1_STAT_INT | VIF1_STAT_ER0 | VIF1_STAT_ER1); if (cancel) { - g_vif1Cycles = 0; - // loop necessary for spiderman - switch(dmacRegs.ctrl.MFD) - { - case MFD_VIF1: - //Console.WriteLn("MFIFO Stall"); - //MFIFO active and not empty - if(vif1ch.chcr.STR && !vif1Regs.stat.test(VIF1_STAT_FDR)) CPU_INT(DMAC_MFIFO_VIF, 0); - break; + g_vif1Cycles = 0; + // loop necessary for spiderman + switch (dmacRegs.ctrl.MFD) + { + case MFD_VIF1: + //Console.WriteLn("MFIFO Stall"); + //MFIFO active and not empty + if (vif1ch.chcr.STR && !vif1Regs.stat.test(VIF1_STAT_FDR)) + CPU_INT(DMAC_MFIFO_VIF, 0); + break; - case NO_MFD: - case MFD_RESERVED: - case MFD_GIF: // Wonder if this should be with VIF? - // Gets the timing right - Flatout - if(vif1ch.chcr.STR && !vif1Regs.stat.test(VIF1_STAT_FDR)) CPU_INT(DMAC_VIF1, 0); - break; - } + case NO_MFD: + case MFD_RESERVED: + case MFD_GIF: // Wonder if this should be with VIF? + // Gets the timing right - Flatout + if (vif1ch.chcr.STR && !vif1Regs.stat.test(VIF1_STAT_FDR)) + CPU_INT(DMAC_VIF1, 0); + break; + } - //vif1ch.chcr.STR = true; + //vif1ch.chcr.STR = true; } } } -__fi void vif1STAT(u32 value) { +__fi void vif1STAT(u32 value) +{ VIF_LOG("VIF1_STAT write32 0x%8.8x", value); /* Only FDR bit is writable, so mask the rest */ - if ((vif1Regs.stat.FDR) ^ ((tVIF_STAT&)value).FDR) { + if ((vif1Regs.stat.FDR) ^ ((tVIF_STAT&)value).FDR) + { bool isStalled = false; // different so can't be stalled - if (vif1Regs.stat.test(VIF1_STAT_INT | VIF1_STAT_VSS | VIF1_STAT_VIS | VIF1_STAT_VFS)) { + if (vif1Regs.stat.test(VIF1_STAT_INT | VIF1_STAT_VSS | VIF1_STAT_VIS | VIF1_STAT_VFS)) + { DbgCon.WriteLn("changing dir when vif1 fifo stalled done = %x qwc = %x stat = %x", vif1.done, vif1ch.qwc, vif1Regs.stat._u32); isStalled = true; } @@ -255,7 +263,8 @@ __fi void vif1STAT(u32 value) { //Hotwheels had this in the "direction when stalled" area, however Sled Storm seems to keep an eye on the dma //position, as we clear it and set it to the end well before the interrupt, the game assumes it's finished, //then proceeds to reverse the dma before we have even done it ourselves. So lets just make sure VIF is ready :) - if (vif1ch.qwc > 0 || isStalled == false){ + if (vif1ch.qwc > 0 || isStalled == false) + { vif1ch.qwc = 0; hwDmacIrq(DMAC_VIF1); vif1ch.chcr.STR = false; @@ -268,7 +277,7 @@ __fi void vif1STAT(u32 value) { if (vif1Regs.stat.FDR) // Vif transferring to memory. { - // Hack but it checks this is true before transfer? (fatal frame) + // Hack but it checks this is true before transfer? (fatal frame) // Update Refraction: Use of this function has been investigated and understood. // Before this ever happens, a DIRECT/HL command takes place sending the transfer info to the GS // One of the registers told about this is TRXREG which tells us how much data is going to transfer (th x tw) in words @@ -277,74 +286,137 @@ __fi void vif1STAT(u32 value) { vif1Regs.stat.FQC = std::min((u32)16, vif1.GSLastDownloadSize); //Console.Warning("Reversing VIF Transfer for %x QWC", vif1.GSLastDownloadSize); - } else // Memory transferring to Vif. { //Sometimes the value from the GS is bigger than vif wanted, so it just sets it back and cancels it. //Other times it can read it off ;) vif1Regs.stat.FQC = 0; - if (vif1ch.chcr.STR) CPU_INT(DMAC_VIF1, 0); + if (vif1ch.chcr.STR) + CPU_INT(DMAC_VIF1, 0); } } #define caseVif(x) (idx ? VIF1_##x : VIF0_##x) -_vifT __fi u32 vifRead32(u32 mem) { +_vifT __fi u32 vifRead32(u32 mem) +{ vifStruct& vif = MTVU_VifX; bool wait = idx && THREAD_VU1; - switch (mem) { - case caseVif(ROW0): if (wait) vu1Thread.WaitVU(); return vif.MaskRow._u32[0]; - case caseVif(ROW1): if (wait) vu1Thread.WaitVU(); return vif.MaskRow._u32[1]; - case caseVif(ROW2): if (wait) vu1Thread.WaitVU(); return vif.MaskRow._u32[2]; - case caseVif(ROW3): if (wait) vu1Thread.WaitVU(); return vif.MaskRow._u32[3]; + switch (mem) + { + case caseVif(ROW0): + if (wait) + vu1Thread.WaitVU(); + return vif.MaskRow._u32[0]; + case caseVif(ROW1): + if (wait) + vu1Thread.WaitVU(); + return vif.MaskRow._u32[1]; + case caseVif(ROW2): + if (wait) + vu1Thread.WaitVU(); + return vif.MaskRow._u32[2]; + case caseVif(ROW3): + if (wait) + vu1Thread.WaitVU(); + return vif.MaskRow._u32[3]; - case caseVif(COL0): if (wait) vu1Thread.WaitVU(); return vif.MaskCol._u32[0]; - case caseVif(COL1): if (wait) vu1Thread.WaitVU(); return vif.MaskCol._u32[1]; - case caseVif(COL2): if (wait) vu1Thread.WaitVU(); return vif.MaskCol._u32[2]; - case caseVif(COL3): if (wait) vu1Thread.WaitVU(); return vif.MaskCol._u32[3]; + case caseVif(COL0): + if (wait) + vu1Thread.WaitVU(); + return vif.MaskCol._u32[0]; + case caseVif(COL1): + if (wait) + vu1Thread.WaitVU(); + return vif.MaskCol._u32[1]; + case caseVif(COL2): + if (wait) + vu1Thread.WaitVU(); + return vif.MaskCol._u32[2]; + case caseVif(COL3): + if (wait) + vu1Thread.WaitVU(); + return vif.MaskCol._u32[3]; } - + return psHu32(mem); } // returns FALSE if no writeback is needed (or writeback is handled internally) // returns TRUE if the caller should writeback the value to the eeHw register map. -_vifT __fi bool vifWrite32(u32 mem, u32 value) { +_vifT __fi bool vifWrite32(u32 mem, u32 value) +{ vifStruct& vif = GetVifX; - switch (mem) { + switch (mem) + { case caseVif(MARK): VIF_LOG("VIF%d_MARK write32 0x%8.8x", idx, value); vifXRegs.stat.MRK = false; //vifXRegs.mark = value; - break; + break; case caseVif(FBRST): - if (!idx) vif0FBRST(value); - else vif1FBRST(value); - return false; + if (!idx) + vif0FBRST(value); + else + vif1FBRST(value); + return false; case caseVif(STAT): - if (idx) { // Only Vif1 does this stuff? + if (idx) + { // Only Vif1 does this stuff? vif1STAT(value); } - return false; + return false; case caseVif(ERR): case caseVif(MODE): // standard register writes -- handled by caller. - break; + break; - case caseVif(ROW0): vif.MaskRow._u32[0] = value; if (idx && THREAD_VU1) vu1Thread.WriteRow(vif); return false; - case caseVif(ROW1): vif.MaskRow._u32[1] = value; if (idx && THREAD_VU1) vu1Thread.WriteRow(vif); return false; - case caseVif(ROW2): vif.MaskRow._u32[2] = value; if (idx && THREAD_VU1) vu1Thread.WriteRow(vif); return false; - case caseVif(ROW3): vif.MaskRow._u32[3] = value; if (idx && THREAD_VU1) vu1Thread.WriteRow(vif); return false; + case caseVif(ROW0): + vif.MaskRow._u32[0] = value; + if (idx && THREAD_VU1) + vu1Thread.WriteRow(vif); + return false; + case caseVif(ROW1): + vif.MaskRow._u32[1] = value; + if (idx && THREAD_VU1) + vu1Thread.WriteRow(vif); + return false; + case caseVif(ROW2): + vif.MaskRow._u32[2] = value; + if (idx && THREAD_VU1) + vu1Thread.WriteRow(vif); + return false; + case caseVif(ROW3): + vif.MaskRow._u32[3] = value; + if (idx && THREAD_VU1) + vu1Thread.WriteRow(vif); + return false; - case caseVif(COL0): vif.MaskCol._u32[0] = value; if (idx && THREAD_VU1) vu1Thread.WriteCol(vif); return false; - case caseVif(COL1): vif.MaskCol._u32[1] = value; if (idx && THREAD_VU1) vu1Thread.WriteCol(vif); return false; - case caseVif(COL2): vif.MaskCol._u32[2] = value; if (idx && THREAD_VU1) vu1Thread.WriteCol(vif); return false; - case caseVif(COL3): vif.MaskCol._u32[3] = value; if (idx && THREAD_VU1) vu1Thread.WriteCol(vif); return false; + case caseVif(COL0): + vif.MaskCol._u32[0] = value; + if (idx && THREAD_VU1) + vu1Thread.WriteCol(vif); + return false; + case caseVif(COL1): + vif.MaskCol._u32[1] = value; + if (idx && THREAD_VU1) + vu1Thread.WriteCol(vif); + return false; + case caseVif(COL2): + vif.MaskCol._u32[2] = value; + if (idx && THREAD_VU1) + vu1Thread.WriteCol(vif); + return false; + case caseVif(COL3): + vif.MaskCol._u32[3] = value; + if (idx && THREAD_VU1) + vu1Thread.WriteCol(vif); + return false; } // fall-through case: issue standard writeback behavior. diff --git a/pcsx2/Vif_Codes.cpp b/pcsx2/Vif_Codes.cpp index fdde320f4d..d8acda504a 100644 --- a/pcsx2/Vif_Codes.cpp +++ b/pcsx2/Vif_Codes.cpp @@ -22,12 +22,16 @@ #include "VUmicro.h" #include "MTVU.h" -#define vifOp(vifCodeName) _vifT int __fastcall vifCodeName(int pass, const u32 *data) -#define pass1 if (pass == 0) -#define pass2 if (pass == 1) -#define pass3 if (pass == 2) +#define vifOp(vifCodeName) _vifT int __fastcall vifCodeName(int pass, const u32* data) +#define pass1 if (pass == 0) +#define pass2 if (pass == 1) +#define pass3 if (pass == 2) #define pass1or2 if (pass == 0 || pass == 1) -#define vif1Only() { if (!idx) return vifCode_Null(pass, (u32*)data); } +#define vif1Only() \ + { \ + if (!idx) \ + return vifCode_Null(pass, (u32*)data); \ + } vifOp(vifCode_Null); //------------------------------------------------------------------ @@ -41,8 +45,10 @@ __ri void vifExecQueue(int idx) GetVifX.queued_program = false; - if (!idx) vu0ExecMicro(vif0.queued_pc); - else vu1ExecMicro(vif1.queued_pc); + if (!idx) + vu0ExecMicro(vif0.queued_pc); + else + vu1ExecMicro(vif1.queued_pc); // Hack for Wakeboarding Unleashed, game runs a VU program in parallel with a VIF unpack list. // The start of the VU program clears the VU memory, while VIF populates it from behind, so we need to get the clear out of the way. @@ -53,40 +59,48 @@ __ri void vifExecQueue(int idx) }*/ } -static __fi void vifFlush(int idx) { +static __fi void vifFlush(int idx) +{ vifExecQueue(idx); - if (!idx) vif0FLUSH(); - else vif1FLUSH(); + if (!idx) + vif0FLUSH(); + else + vif1FLUSH(); vifExecQueue(idx); } -static __fi void vuExecMicro(int idx, u32 addr) { +static __fi void vuExecMicro(int idx, u32 addr) +{ VIFregisters& vifRegs = vifXRegs; vifFlush(idx); - if(GetVifX.waitforvu) + if (GetVifX.waitforvu) return; - if (vifRegs.itops > (idx ? 0x3ffu : 0xffu)) { + if (vifRegs.itops > (idx ? 0x3ffu : 0xffu)) + { Console.WriteLn("VIF%d ITOP overrun! %x", idx, vifRegs.itops); vifRegs.itops &= (idx ? 0x3ffu : 0xffu); } vifRegs.itop = vifRegs.itops; - if (idx) { + if (idx) + { // in case we're handling a VIF1 execMicro, set the top with the tops value vifRegs.top = vifRegs.tops & 0x3ff; // is DBF flag set in VIF_STAT? - if (vifRegs.stat.DBF) { + if (vifRegs.stat.DBF) + { // it is, so set tops with base, and clear the stat DBF flag vifRegs.tops = vifRegs.base; vifRegs.stat.DBF = false; } - else { + else + { // it is not, so set tops with base + offset, and set stat DBF flag vifRegs.tops = vifRegs.base + vifRegs.ofst; vifRegs.stat.DBF = true; @@ -108,13 +122,13 @@ void ExecuteVU(int idx) { vifStruct& vifX = GetVifX; - if((vifX.cmd & 0x7f) == 0x17) + if ((vifX.cmd & 0x7f) == 0x17) { vuExecMicro(idx, -1); vifX.cmd = 0; vifX.pass = 0; } - else if((vifX.cmd & 0x7f) == 0x14 || (vifX.cmd & 0x7f) == 0x15) + else if ((vifX.cmd & 0x7f) == 0x14 || (vifX.cmd & 0x7f) == 0x15) { vuExecMicro(idx, (u16)(vifXRegs.code)); vifX.cmd = 0; @@ -127,44 +141,58 @@ void ExecuteVU(int idx) // Vif0/Vif1 Code Implementations //------------------------------------------------------------------ -vifOp(vifCode_Base) { +vifOp(vifCode_Base) +{ vif1Only(); - pass1 { vif1Regs.base = vif1Regs.code & 0x3ff; vif1.cmd = 0; vif1.pass = 0; } + pass1 + { + vif1Regs.base = vif1Regs.code & 0x3ff; + vif1.cmd = 0; + vif1.pass = 0; + } pass3 { VifCodeLog("Base"); } return 1; } -template __fi int _vifCode_Direct(int pass, const u8* data, bool isDirectHL) { +template +__fi int _vifCode_Direct(int pass, const u8* data, bool isDirectHL) +{ vif1Only(); - pass1 { - int vifImm = (u16)vif1Regs.code; - vif1.tag.size = vifImm ? (vifImm*4) : (65536*4); + pass1 + { + int vifImm = (u16)vif1Regs.code; + vif1.tag.size = vifImm ? (vifImm * 4) : (65536 * 4); vif1.pass = 1; return 1; } - pass2 { + pass2 + { const char* name = isDirectHL ? "DirectHL" : "Direct"; GIF_TRANSFER_TYPE tranType = isDirectHL ? GIF_TRANS_DIRECTHL : GIF_TRANS_DIRECT; uint size = std::min(vif1.vifpacketsize, vif1.tag.size) * 4; // Get size in bytes - uint ret = gifUnit.TransferGSPacketData(tranType, (u8*)data, size); + uint ret = gifUnit.TransferGSPacketData(tranType, (u8*)data, size); - vif1.tag.size -= ret/4; // Convert to u32's + vif1.tag.size -= ret / 4; // Convert to u32's vif1Regs.stat.VGW = false; - if (ret & 3) DevCon.Warning("Vif %s: Ret wasn't a multiple of 4!", name); // Shouldn't happen - if (size == 0) DevCon.Warning("Vif %s: No Data Transfer?", name); // Can this happen? - if (size != ret) { // Stall if gif didn't process all the data (path2 queued) + if (ret & 3) + DevCon.Warning("Vif %s: Ret wasn't a multiple of 4!", name); // Shouldn't happen + if (size == 0) + DevCon.Warning("Vif %s: No Data Transfer?", name); // Can this happen? + if (size != ret) + { // Stall if gif didn't process all the data (path2 queued) GUNIT_WARN("Vif %s: Stall! [size=%d][ret=%d]", name, size, ret); //gifUnit.PrintInfo(); - vif1.vifstalled.enabled = VifStallEnable(vif1ch); + vif1.vifstalled.enabled = VifStallEnable(vif1ch); vif1.vifstalled.value = VIF_TIMING_BREAK; vif1Regs.stat.VGW = true; return 0; } - if (vif1.tag.size == 0) { + if (vif1.tag.size == 0) + { vif1.cmd = 0; vif1.pass = 0; - vif1.vifstalled.enabled = VifStallEnable(vif1ch); + vif1.vifstalled.enabled = VifStallEnable(vif1ch); vif1.vifstalled.value = VIF_TIMING_BREAK; } return ret / 4; @@ -172,56 +200,63 @@ template __fi int _vifCode_Direct(int pass, const u8* data, bool isDire return 0; } -vifOp(vifCode_Direct) { +vifOp(vifCode_Direct) +{ pass3 { VifCodeLog("Direct"); } return _vifCode_Direct(pass, (u8*)data, 0); } -vifOp(vifCode_DirectHL) { +vifOp(vifCode_DirectHL) +{ pass3 { VifCodeLog("DirectHL"); } return _vifCode_Direct(pass, (u8*)data, 1); } -vifOp(vifCode_Flush) { +vifOp(vifCode_Flush) +{ vif1Only(); //vifStruct& vifX = GetVifX; - pass1or2 { + pass1or2 + { bool p1or2 = (gifRegs.stat.APATH != 0 && gifRegs.stat.APATH != 3); vif1Regs.stat.VGW = false; vifFlush(idx); - if (gifUnit.checkPaths(1,1,0) || p1or2) { + if (gifUnit.checkPaths(1, 1, 0) || p1or2) + { GUNIT_WARN("Vif Flush: Stall!"); //gifUnit.PrintInfo(); vif1Regs.stat.VGW = true; - vif1.vifstalled.enabled = VifStallEnable(vif1ch); + vif1.vifstalled.enabled = VifStallEnable(vif1ch); vif1.vifstalled.value = VIF_TIMING_BREAK; return 0; } - else vif1.cmd = 0; + else + vif1.cmd = 0; vif1.pass = 0; } pass3 { VifCodeLog("Flush"); } return 1; } -vifOp(vifCode_FlushA) { +vifOp(vifCode_FlushA) +{ vif1Only(); //vifStruct& vifX = GetVifX; - pass1or2 { + pass1or2 + { //Gif_Path& p3 = gifUnit.gifPath[GIF_PATH_3]; - u32 gifBusy = gifUnit.checkPaths(1,1,1) || (gifRegs.stat.APATH != 0); + u32 gifBusy = gifUnit.checkPaths(1, 1, 1) || (gifRegs.stat.APATH != 0); //bool doStall = false; vif1Regs.stat.VGW = false; vifFlush(idx); - if (gifBusy) + if (gifBusy) { GUNIT_WARN("Vif FlushA: Stall!"); vif1Regs.stat.VGW = true; - vif1.vifstalled.enabled = VifStallEnable(vif1ch); + vif1.vifstalled.enabled = VifStallEnable(vif1ch); vif1.vifstalled.value = VIF_TIMING_BREAK; return 0; - } vif1.cmd = 0; vif1.pass = 0; @@ -231,32 +266,47 @@ vifOp(vifCode_FlushA) { } // ToDo: FixMe -vifOp(vifCode_FlushE) { +vifOp(vifCode_FlushE) +{ vifStruct& vifX = GetVifX; - pass1 { vifFlush(idx); vifX.cmd = 0; vifX.pass = 0;} + pass1 + { + vifFlush(idx); + vifX.cmd = 0; + vifX.pass = 0; + } pass3 { VifCodeLog("FlushE"); } return 1; } -vifOp(vifCode_ITop) { - pass1 { vifXRegs.itops = vifXRegs.code & 0x3ff; GetVifX.cmd = 0; GetVifX.pass = 0; } +vifOp(vifCode_ITop) +{ + pass1 + { + vifXRegs.itops = vifXRegs.code & 0x3ff; + GetVifX.cmd = 0; + GetVifX.pass = 0; + } pass3 { VifCodeLog("ITop"); } return 1; } -vifOp(vifCode_Mark) { +vifOp(vifCode_Mark) +{ vifStruct& vifX = GetVifX; - pass1 { - vifXRegs.mark = (u16)vifXRegs.code; + pass1 + { + vifXRegs.mark = (u16)vifXRegs.code; vifXRegs.stat.MRK = true; - vifX.cmd = 0; - vifX.pass = 0; + vifX.cmd = 0; + vifX.pass = 0; } pass3 { VifCodeLog("Mark"); } return 1; } -static __fi void _vifCode_MPG(int idx, u32 addr, const u32 *data, int size) { +static __fi void _vifCode_MPG(int idx, u32 addr, const u32* data, int size) +{ VURegs& VUx = idx ? VU1 : VU0; vifStruct& vifX = GetVifX; u16 vuMemSize = idx ? 0x4000 : 0x1000; @@ -264,7 +314,8 @@ static __fi void _vifCode_MPG(int idx, u32 addr, const u32 *data, int size) { vifExecQueue(idx); - if (idx && THREAD_VU1) { + if (idx && THREAD_VU1) + { if ((addr + size * 4) > vuMemSize) { vu1Thread.WriteMicroMem(addr, (u8*)data, vuMemSize - addr); @@ -282,12 +333,14 @@ static __fi void _vifCode_MPG(int idx, u32 addr, const u32 *data, int size) { } // Don't forget the Unsigned designator for these checks - if((addr + size *4) > vuMemSize) + if ((addr + size * 4) > vuMemSize) { //DevCon.Warning("Handling split MPG"); - if (!idx) CpuVU0->Clear(addr, vuMemSize - addr); - else CpuVU1->Clear(addr, vuMemSize - addr); - + if (!idx) + CpuVU0->Clear(addr, vuMemSize - addr); + else + CpuVU1->Clear(addr, vuMemSize - addr); + memcpy(VUx.Micro + addr, data, vuMemSize - addr); size -= (vuMemSize - addr) / 4; data += (vuMemSize - addr) / 4; @@ -297,51 +350,61 @@ static __fi void _vifCode_MPG(int idx, u32 addr, const u32 *data, int size) { } else { - //The compare is pretty much a waste of time, likelyhood is that the program isnt there, thats why its copying it. - //Faster without. - //if (memcmp_mmx(VUx.Micro + addr, data, size*4)) { + //The compare is pretty much a waste of time, likelyhood is that the program isnt there, thats why its copying it. + //Faster without. + //if (memcmp_mmx(VUx.Micro + addr, data, size*4)) { // Clear VU memory before writing! - if (!idx) CpuVU0->Clear(addr, size*4); - else CpuVU1->Clear(addr, size*4); - memcpy(VUx.Micro + addr, data, size*4); //from tests, memcpy is 1fps faster on Grandia 3 than memcpy + if (!idx) + CpuVU0->Clear(addr, size * 4); + else + CpuVU1->Clear(addr, size * 4); + memcpy(VUx.Micro + addr, data, size * 4); //from tests, memcpy is 1fps faster on Grandia 3 than memcpy vifX.tag.addr += size * 4; } } -vifOp(vifCode_MPG) { +vifOp(vifCode_MPG) +{ vifStruct& vifX = GetVifX; - pass1 { - int vifNum = (u8)(vifXRegs.code >> 16); - vifX.tag.addr = (u16)(vifXRegs.code << 3) & (idx ? 0x3fff : 0xfff); - vifX.tag.size = vifNum ? (vifNum*2) : 512; + pass1 + { + int vifNum = (u8)(vifXRegs.code >> 16); + vifX.tag.addr = (u16)(vifXRegs.code << 3) & (idx ? 0x3fff : 0xfff); + vifX.tag.size = vifNum ? (vifNum * 2) : 512; vifFlush(idx); - if(vifX.vifstalled.enabled) return 0; + if (vifX.vifstalled.enabled) + return 0; else { vifX.pass = 1; return 1; } } - pass2 { - if (vifX.vifpacketsize < vifX.tag.size) { // Partial Transfer - if((vifX.tag.addr + vifX.vifpacketsize*4) > (idx ? 0x4000 : 0x1000)) { + pass2 + { + if (vifX.vifpacketsize < vifX.tag.size) + { // Partial Transfer + if ((vifX.tag.addr + vifX.vifpacketsize * 4) > (idx ? 0x4000 : 0x1000)) + { //DevCon.Warning("Vif%d MPG Split Overflow", idx); } - _vifCode_MPG(idx, vifX.tag.addr, data, vifX.vifpacketsize); + _vifCode_MPG(idx, vifX.tag.addr, data, vifX.vifpacketsize); vifX.tag.size -= vifX.vifpacketsize; //We can do this first as its passed as a pointer return vifX.vifpacketsize; } - else { // Full Transfer - if((vifX.tag.addr + vifX.tag.size*4) > (idx ? 0x4000 : 0x1000)) { + else + { // Full Transfer + if ((vifX.tag.addr + vifX.tag.size * 4) > (idx ? 0x4000 : 0x1000)) + { //DevCon.Warning("Vif%d MPG Split Overflow full %x", idx, vifX.tag.addr + vifX.tag.size*4); } - _vifCode_MPG(idx, vifX.tag.addr, data, vifX.tag.size); + _vifCode_MPG(idx, vifX.tag.addr, data, vifX.tag.size); int ret = vifX.tag.size; vifX.tag.size = 0; - vifX.cmd = 0; - vifX.pass = 0; + vifX.cmd = 0; + vifX.pass = 0; return ret; } } @@ -349,43 +412,48 @@ vifOp(vifCode_MPG) { return 0; } -vifOp(vifCode_MSCAL) { +vifOp(vifCode_MSCAL) +{ vifStruct& vifX = GetVifX; - pass1 { - vifFlush(idx); + pass1 + { + vifFlush(idx); - if(!vifX.waitforvu) + if (!vifX.waitforvu) { - vuExecMicro(idx, (u16)(vifXRegs.code)); + vuExecMicro(idx, (u16)(vifXRegs.code)); vifX.cmd = 0; vifX.pass = 0; - if(GetVifX.vifpacketsize > 1) + if (GetVifX.vifpacketsize > 1) { //Warship Gunner 2 has a rather big dislike for the delays - if(((data[1] >> 24) & 0x60) == 0x60) // Immediate following Unpack - { + if (((data[1] >> 24) & 0x60) == 0x60) // Immediate following Unpack + { //Snowblind games only use MSCAL, so other MS kicks force the program directly. vifExecQueue(idx); } } - } + } } pass3 { VifCodeLog("MSCAL"); } return 1; } -vifOp(vifCode_MSCALF) { +vifOp(vifCode_MSCALF) +{ vifStruct& vifX = GetVifX; - pass1or2 { + pass1or2 + { vifXRegs.stat.VGW = false; vifFlush(idx); - if (u32 a = gifUnit.checkPaths(1,1,0)) { - GUNIT_WARN("Vif MSCALF: Stall! [%d,%d]", !!(a&1), !!(a&2)); + if (u32 a = gifUnit.checkPaths(1, 1, 0)) + { + GUNIT_WARN("Vif MSCALF: Stall! [%d,%d]", !!(a & 1), !!(a & 2)); vif1Regs.stat.VGW = true; - vifX.vifstalled.enabled = VifStallEnable(vifXch); + vifX.vifstalled.enabled = VifStallEnable(vifXch); vifX.vifstalled.value = VIF_TIMING_BREAK; } - if(!vifX.waitforvu) + if (!vifX.waitforvu) { vuExecMicro(idx, (u16)(vifXRegs.code)); vifX.cmd = 0; @@ -397,11 +465,13 @@ vifOp(vifCode_MSCALF) { return 1; } -vifOp(vifCode_MSCNT) { +vifOp(vifCode_MSCNT) +{ vifStruct& vifX = GetVifX; - pass1 { - vifFlush(idx); - if(!vifX.waitforvu) + pass1 + { + vifFlush(idx); + if (!vifX.waitforvu) { vuExecMicro(idx, -1); vifX.cmd = 0; @@ -420,13 +490,16 @@ vifOp(vifCode_MSCNT) { } // ToDo: FixMe -vifOp(vifCode_MskPath3) { +vifOp(vifCode_MskPath3) +{ vif1Only(); - pass1 { + pass1 + { vif1Regs.mskpath3 = (vif1Regs.code >> 15) & 0x1; - gifRegs.stat.M3P = (vif1Regs.code >> 15) & 0x1; + gifRegs.stat.M3P = (vif1Regs.code >> 15) & 0x1; GUNIT_LOG("Vif1 - MskPath3 [p3 = %s]", vif1Regs.mskpath3 ? "masked" : "enabled"); - if(!vif1Regs.mskpath3) { + if (!vif1Regs.mskpath3) + { GUNIT_WARN("VIF MSKPATH3 off Path3 triggering!"); gifInterrupt(); } @@ -437,17 +510,19 @@ vifOp(vifCode_MskPath3) { return 1; } -vifOp(vifCode_Nop) { - pass1 { +vifOp(vifCode_Nop) +{ + pass1 + { GetVifX.cmd = 0; GetVifX.pass = 0; vifExecQueue(idx); if (GetVifX.vifpacketsize > 1) { - if(((data[1] >> 24) & 0x7f) == 0x6 && (data[1] & 0x1)) //is mskpath3 next - { - GetVifX.vifstalled.enabled = VifStallEnable(vifXch); + if (((data[1] >> 24) & 0x7f) == 0x6 && (data[1] & 0x1)) //is mskpath3 next + { + GetVifX.vifstalled.enabled = VifStallEnable(vifXch); GetVifX.vifstalled.value = VIF_TIMING_BREAK; } } @@ -457,11 +532,14 @@ vifOp(vifCode_Nop) { } // ToDo: Review Flags -vifOp(vifCode_Null) { +vifOp(vifCode_Null) +{ vifStruct& vifX = GetVifX; - pass1 { + pass1 + { // if ME1, then force the vif to interrupt - if (!(vifXRegs.err.ME1)) { // Ignore vifcode and tag mismatch error + if (!(vifXRegs.err.ME1)) + { // Ignore vifcode and tag mismatch error Console.WriteLn("Vif%d: Unknown VifCmd! [%x]", idx, vifX.cmd); vifXRegs.stat.ER1 = true; vifX.vifstalled.enabled = VifStallEnable(vifXch); @@ -472,34 +550,40 @@ vifOp(vifCode_Null) { vifX.pass = 0; //If the top bit was set to interrupt, we don't want it to take commands from a bad code - if (vifXRegs.code & 0x80000000) vifX.irq = 0; + if (vifXRegs.code & 0x80000000) + vifX.irq = 0; } pass2 { Console.Error("Vif%d bad vifcode! [CMD = %x]", idx, vifX.cmd); } pass3 { VifCodeLog("Null"); } return 1; } -vifOp(vifCode_Offset) { +vifOp(vifCode_Offset) +{ vif1Only(); - pass1 { - vif1Regs.stat.DBF = false; - vif1Regs.ofst = vif1Regs.code & 0x3ff; - vif1Regs.tops = vif1Regs.base; - vif1.cmd = 0; - vif1.pass = 0; + pass1 + { + vif1Regs.stat.DBF = false; + vif1Regs.ofst = vif1Regs.code & 0x3ff; + vif1Regs.tops = vif1Regs.base; + vif1.cmd = 0; + vif1.pass = 0; } pass3 { VifCodeLog("Offset"); } return 1; } -template static __fi int _vifCode_STColRow(const u32* data, u32* pmem2) { +template +static __fi int _vifCode_STColRow(const u32* data, u32* pmem2) +{ vifStruct& vifX = GetVifX; int ret = std::min(4 - vifX.tag.addr, vifX.vifpacketsize); pxAssume(vifX.tag.addr < 4); pxAssume(ret > 0); - switch (ret) { + switch (ret) + { case 4: pmem2[3] = data[3]; [[fallthrough]]; @@ -511,8 +595,8 @@ template static __fi int _vifCode_STColRow(const u32* data, u32* pmem2) [[fallthrough]]; case 1: pmem2[0] = data[0]; - break; - jNO_DEFAULT + break; + jNO_DEFAULT } vifX.tag.addr += ret; @@ -523,20 +607,23 @@ template static __fi int _vifCode_STColRow(const u32* data, u32* pmem2) vifX.cmd = 0; } - + return ret; } -vifOp(vifCode_STCol) { +vifOp(vifCode_STCol) +{ vifStruct& vifX = GetVifX; - pass1 { + pass1 + { vifX.tag.addr = 0; vifX.tag.size = 4; vifX.pass = 1; return 1; } - pass2 { + pass2 + { u32 ret = _vifCode_STColRow(data, &vifX.MaskCol._u32[vifX.tag.addr]); if (idx && vifX.tag.size == 0) vu1Thread.WriteCol(vifX); @@ -546,15 +633,18 @@ vifOp(vifCode_STCol) { return 0; } -vifOp(vifCode_STRow) { +vifOp(vifCode_STRow) +{ vifStruct& vifX = GetVifX; - pass1 { + pass1 + { vifX.tag.addr = 0; vifX.tag.size = 4; vifX.pass = 1; return 1; } - pass2 { + pass2 + { u32 ret = _vifCode_STColRow(data, &vifX.MaskRow._u32[vifX.tag.addr]); if (idx && vifX.tag.size == 0) vu1Thread.WriteRow(vifX); @@ -564,65 +654,89 @@ vifOp(vifCode_STRow) { return 1; } -vifOp(vifCode_STCycl) { +vifOp(vifCode_STCycl) +{ vifStruct& vifX = GetVifX; - pass1 { + pass1 + { vifXRegs.cycle.cl = (u8)(vifXRegs.code); vifXRegs.cycle.wl = (u8)(vifXRegs.code >> 8); - vifX.cmd = 0; - vifX.pass = 0; + vifX.cmd = 0; + vifX.pass = 0; } pass3 { VifCodeLog("STCycl"); } return 1; } -vifOp(vifCode_STMask) { +vifOp(vifCode_STMask) +{ vifStruct& vifX = GetVifX; - pass1 { vifX.tag.size = 1; vifX.pass = 1; return 1; } - pass2 { vifXRegs.mask = data[0]; vifX.tag.size = 0; vifX.cmd = 0; vifX.pass = 0;} + pass1 + { + vifX.tag.size = 1; + vifX.pass = 1; + return 1; + } + pass2 + { + vifXRegs.mask = data[0]; + vifX.tag.size = 0; + vifX.cmd = 0; + vifX.pass = 0; + } pass3 { VifCodeLog("STMask"); } return 1; } -vifOp(vifCode_STMod) { - pass1 { vifXRegs.mode = vifXRegs.code & 0x3; GetVifX.cmd = 0; GetVifX.pass = 0;} +vifOp(vifCode_STMod) +{ + pass1 + { + vifXRegs.mode = vifXRegs.code & 0x3; + GetVifX.cmd = 0; + GetVifX.pass = 0; + } pass3 { VifCodeLog("STMod"); } return 1; } -template< uint idx > +template static uint calc_addr(bool flg) { VIFregisters& vifRegs = vifXRegs; uint retval = vifRegs.code; - if (idx && flg) retval += vifRegs.tops; + if (idx && flg) + retval += vifRegs.tops; return retval & (idx ? 0x3ff : 0xff); } -vifOp(vifCode_Unpack) { - pass1 { +vifOp(vifCode_Unpack) +{ + pass1 + { vifUnpackSetup(data); - + return 1; } - pass2 { + pass2 + { return nVifUnpack((u8*)data); } - pass3 { + pass3 + { vifStruct& vifX = GetVifX; VIFregisters& vifRegs = vifXRegs; uint vl = vifX.cmd & 0x03; uint vn = (vifX.cmd >> 2) & 0x3; bool flg = (vifRegs.code >> 15) & 1; - static const char* const vntbl[] = { "S", "V2", "V3", "V4" }; - static const uint vltbl[] = { 32, 16, 8, 5 }; + static const char* const vntbl[] = {"S", "V2", "V3", "V4"}; + static const uint vltbl[] = {32, 16, 8, 5}; VifCodeLog("Unpack %s_%u (%s) @ 0x%04X%s (cl=%u wl=%u num=0x%02X)", vntbl[vn], vltbl[vl], (vifX.cmd & 0x10) ? "masked" : "unmasked", calc_addr(flg), flg ? "(FLG)" : "", - vifRegs.cycle.cl, vifRegs.cycle.wl, (vifXRegs.code >> 16) & 0xff - ); + vifRegs.cycle.cl, vifRegs.cycle.wl, (vifXRegs.code >> 16) & 0xff); } return 0; } @@ -669,4 +783,4 @@ __aligned16 FnType_VifCmdHandler* const vifCmdHandler[2][128] = vifCode_Unpack<1> , vifCode_Unpack<1> , vifCode_Unpack<1> , vifCode_Unpack<1> , vifCode_Unpack<1> , vifCode_Unpack<1> , vifCode_Unpack<1> , vifCode_Null<1>, /*0x70*/ vifCode_Unpack<1> , vifCode_Unpack<1> , vifCode_Unpack<1> , vifCode_Null<1> , vifCode_Unpack<1> , vifCode_Unpack<1> , vifCode_Unpack<1> , vifCode_Unpack<1> /*0x78*/ } -}; +}; \ No newline at end of file