// SPDX-FileCopyrightText: 2002-2026 PCSX2 Dev Team // SPDX-License-Identifier: GPL-3.0+ #include "Common.h" #include "Gif_Unit.h" #include "Hardware.h" #include "SPU2/spu2.h" #include "USB/USB.h" #include "common/WrappedMemCopy.h" #include "fmt/format.h" #include using namespace R5900; const int rdram_devices = 2; // put 8 for TOOL and 2 for PS2 and PSX int rdram_sdevid = 0; std::deque ee_sio_rx_fifo; std::deque ee_sio_tx_fifo; void hwReset() { std::memset(eeHw, 0, sizeof(eeHw)); { std::deque empty_rx; std::swap(ee_sio_rx_fifo, empty_rx); std::deque empty_tx; std::swap(ee_sio_tx_fifo, empty_tx); } psHu32(SBUS_F260) = 0x1D000060; // i guess this is kinda a version, it's used by some bioses psHu32(DMAC_ENABLEW) = 0x1201; psHu32(DMAC_ENABLER) = 0x1201; rcntInit(); // Sets SPU2 sample rate to PS2 standard (48KHz) whenever emulator is reset. // For PSX mode sample rate setting, see HwWrite.cpp SPU2::Reset(false); sifReset(); gsReset(); gifUnit.Reset(); ipuReset(); vif0Reset(); vif1Reset(); gif_fifo.init(); USBreset(); } __fi uint intcInterrupt() { if ((psHu32(INTC_STAT)) == 0) { //DevCon.Warning("*PCSX2*: intcInterrupt already cleared"); return 0; } if ((psHu32(INTC_STAT) & psHu32(INTC_MASK)) == 0) { //DevCon.Warning("*PCSX2*: No valid interrupt INTC_MASK: %x INTC_STAT: %x", psHu32(INTC_MASK), psHu32(INTC_STAT)); return 0; } HW_LOG("intcInterrupt %x", psHu32(INTC_STAT) & psHu32(INTC_MASK)); if(psHu32(INTC_STAT) & 0x2){ counters[0].hold = rcntRcount(0); counters[1].hold = rcntRcount(1); } //cpuException(0x400, cpuRegs.branch); return 0x400; } __fi uint dmacInterrupt() { if( ((psHu16(DMAC_STAT + 2) & psHu16(DMAC_STAT)) == 0 ) && ( psHu16(DMAC_STAT) & 0x8000) == 0 ) { //DevCon.Warning("No valid DMAC interrupt MASK %x STAT %x", psHu16(DMAC_STAT+2), psHu16(DMAC_STAT)); return 0; } if (!dmacRegs.ctrl.DMAE || psHu8(DMAC_ENABLER+2) == 1) { //DevCon.Warning("DMAC Suspended or Disabled on interrupt"); return 0; } DMA_LOG("dmacInterrupt %x", ((psHu16(DMAC_STAT + 2) & psHu16(DMAC_STAT)) | (psHu16(DMAC_STAT) & 0x8000)) ); //cpuException(0x800, cpuRegs.branch); return 0x800; } void hwIntcIrq(int n) { psHu32(INTC_STAT) |= 1< ((dmacRegs.rbsr.RMSK + 16u) >> 4u)) DevCon.Warning("MFIFO Write bigger than MFIFO! QWC=%x FifoSize=%x", qwc, ((dmacRegs.rbsr.RMSK + 16) >> 4)); // DMAC Address resolution: FIFO can be placed anywhere in the *physical* memory map // for the PS2. Its probably a serious error for a PS2 app to have the buffer cross // valid/invalid page areas of ram, so realistically we only need to test the base address // of the FIFO for address validity. if (u128* dst = (u128*)PSM(dmacRegs.rbor.ADDR)) { const u32 ringsize = (dmacRegs.rbsr.RMSK / 16) + 1; pxAssertMsg( PSM(dmacRegs.rbor.ADDR+ringsize-1) != NULL, "Scratchpad/MFIFO ringbuffer spans into invalid (unmapped) physical memory!" ); uint startpos = (addr & dmacRegs.rbsr.RMSK)/16; MemCopy_WrappedDest( data, dst, startpos, ringsize, qwc ); } else { SPR_LOG( "Scratchpad/MFIFO: invalid base physical address: 0x%08x", dmacRegs.rbor.ADDR ); pxFail( fmt::format( "Scratchpad/MFIFO: Invalid base physical address: 0x{:08x}", u32(dmacRegs.rbor.ADDR)).c_str() ); return false; } return true; } __ri void hwMFIFOResume(u32 transferred) { if (transferred == 0) { return; //Nothing got put in the MFIFO, we don't care } switch (dmacRegs.ctrl.MFD) { case MFD_VIF1: // Most common case. { SPR_LOG("Added %x qw to mfifo, Vif CHCR %x Stalled %x done %x", transferred, vif1ch.chcr._u32, vif1.vifstalled.enabled, vif1.done); if (vif1.inprogress & 0x10) { vif1.inprogress &= ~0x10; //Don't resume if stalled or already looping if (vif1ch.chcr.STR && !(cpuRegs.interrupt & (1 << DMAC_MFIFO_VIF)) && !vif1Regs.stat.INT) { SPR_LOG("Data Added, Resuming"); //Need to simulate the time it takes to copy here, if the VIF resumes before the SPR has finished, it isn't happy. CPU_INT(DMAC_MFIFO_VIF, transferred * BIAS); } //Apparently this is bad, i guess so, the data is going to memory rather than the FIFO //vif1Regs.stat.FQC = 0x10; // FQC=16 } break; } case MFD_GIF: { SPR_LOG("Added %x qw to mfifo, Gif CHCR %x done %x", transferred, gifch.chcr._u32, gif.gspath3done); if ((gif.gifstate & GIF_STATE_EMPTY)) { CPU_INT(DMAC_MFIFO_GIF, transferred * BIAS); gif.gifstate = GIF_STATE_READY; } break; } default: break; } } __ri bool hwDmacSrcChainWithStack(DMACh& dma, int id) { switch (id) { case TAG_REFE: // Refe - Transfer Packet According to ADDR field dma.tadr += 16; //End Transfer return true; case TAG_CNT: // CNT - Transfer QWC following the tag. // Set MADR to QW afer tag, and set TADR to QW following the data. dma.tadr += 16; dma.madr = dma.tadr; //dma.tadr = dma.madr + (dma.qwc << 4); return false; case TAG_NEXT: // Next - Transfer QWC following tag. TADR = ADDR { // Set MADR to QW following the tag, and set TADR to the address formerly in MADR. u32 temp = dma.madr; dma.madr = dma.tadr + 16; dma.tadr = temp; return false; } case TAG_REF: // Ref - Transfer QWC from ADDR field case TAG_REFS: // Refs - Transfer QWC from ADDR field (Stall Control) //Set TADR to next tag dma.tadr += 16; return false; case TAG_CALL: // Call - Transfer QWC following the tag, save succeeding tag { // Store the address in MADR in temp, and set MADR to the data following the tag. u32 temp = dma.madr; dma.madr = dma.tadr + 16; // Stash an address on the address stack pointer. switch(dma.chcr.ASP) { case 0: //Check if ASR0 is empty // Store the succeeding tag in asr0, and mark chcr as having 1 address. dma.asr0 = dma.madr + (dma.qwc << 4); dma.chcr.ASP++; break; case 1: // Store the succeeding tag in asr1, and mark chcr as having 2 addresses. dma.asr1 = dma.madr + (dma.qwc << 4); dma.chcr.ASP++; break; default: Console.Warning("Call Stack Overflow (report if it fixes/breaks anything)"); return true; } // Set TADR to the address from MADR we stored in temp. dma.tadr = temp; return false; } case TAG_RET: // Ret - Transfer QWC following the tag, load next tag //Set MADR to data following the tag. dma.madr = dma.tadr + 16; // Snag an address from the address stack pointer. switch(dma.chcr.ASP) { case 2: // Pull asr1 from the stack, give it to TADR, and decrease the # of addresses. dma.tadr = dma.asr1; dma.asr1 = 0; dma.chcr.ASP--; break; case 1: // Pull asr0 from the stack, give it to TADR, and decrease the # of addresses. dma.tadr = dma.asr0; dma.asr0 = 0; dma.chcr.ASP--; break; case 0: // There aren't any addresses to pull, so end the transfer. //dma.tadr += 16; //Clear tag address - Kills Klonoa 2 return true; default: // If ASR1 and ASR0 are messed up, end the transfer. //Console.Error("TAG_RET: ASR 1 & 0 == 1. This shouldn't happen!"); //dma.tadr += 16; //Clear tag address - Kills Klonoa 2 return true; } return false; case TAG_END: // End - Transfer QWC following the tag //Set MADR to data following the tag, and end the transfer. dma.madr = dma.tadr + 16; //Don't Increment tadr; breaks Soul Calibur II and III return true; } return false; } /********TADR NOTES*********** From what i've gathered from testing tadr increment stuff (with CNT) is that we might not be 100% accurate in what increments it and what doesnt. Previously we presumed REFE and END didn't increment the tag, but SIF and IPU never liked this. From what i've deduced, REFE does in fact increment, but END doesn't, after much testing, i've concluded this is how we can standardize DMA chains, so i've modified the code to work like this. The below function controls the increment of the TADR along with the MADR on VIF, GIF and SPR1 when using the CNT tag, the others don't use it yet, but they can probably be modified to do so now. Reason for this:- Many games (such as clock tower 3 and FFX Videos) watched the TADR to see when a transfer has finished, so we need to simulate this wherever we can! Even the FFX video gets corruption and tries to fire multiple DMA Kicks if this doesnt happen, which was the reasoning for the hacked up SPR timing we had, that is no longer required. -Refraction ******************************/ void hwDmacSrcTadrInc(DMACh& dma) { //Don't touch it if in normal/interleave mode. if (dma.chcr.STR == 0) return; if (dma.chcr.MOD != 1) return; u16 tagid = (dma.chcr.TAG >> 12) & 0x7; if (tagid == TAG_CNT) { dma.tadr = dma.madr; } } bool hwDmacSrcChain(DMACh& dma, int id) { u32 temp; switch (id) { case TAG_REFE: // Refe - Transfer Packet According to ADDR field dma.tadr += 16; // End the transfer. return true; case TAG_CNT: // CNT - Transfer QWC following the tag. // Set MADR to QW after the tag, and TADR to QW following the data. dma.madr = dma.tadr + 16; dma.tadr = dma.madr; return false; case TAG_NEXT: // Next - Transfer QWC following tag. TADR = ADDR // Set MADR to QW following the tag, and set TADR to the address formerly in MADR. temp = dma.madr; dma.madr = dma.tadr + 16; dma.tadr = temp; return false; case TAG_REF: // Ref - Transfer QWC from ADDR field case TAG_REFS: // Refs - Transfer QWC from ADDR field (Stall Control) //Set TADR to next tag dma.tadr += 16; return false; case TAG_END: // End - Transfer QWC following the tag //Set MADR to data following the tag, and end the transfer. dma.madr = dma.tadr + 16; //Don't Increment tadr; breaks Soul Calibur II and III return true; // Undefined Tag handling ends the DMA, maintaining the bad TADR and Tag in upper CHCR // Some games such as DT racer try to use RET tags on IPU, which it doesn't support default: return true; } return false; }