mirror of https://github.com/PCSX2/pcsx2.git
More Sif stuff. Factor out a few functions. Played around with SIF0Dma & SIF1Dma a bit.
git-svn-id: http://pcsx2.googlecode.com/svn/trunk@2496 96395faa-99c1-11dd-bbfe-3dabce05a288
This commit is contained in:
parent
6d7f46751c
commit
bb3eda1524
10
pcsx2/Dmac.h
10
pcsx2/Dmac.h
|
@ -244,6 +244,16 @@ struct DMACh {
|
||||||
chcrTransfer(ptag);
|
chcrTransfer(ptag);
|
||||||
qwcTransfer(ptag);
|
qwcTransfer(ptag);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
wxString cmq_to_str()
|
||||||
|
{
|
||||||
|
return wxsFormat(L"chcr = %lx, madr = %lx, qwc = %lx", chcr._u32, madr, qwc);
|
||||||
|
}
|
||||||
|
|
||||||
|
wxString cmqt_to_str()
|
||||||
|
{
|
||||||
|
return wxsFormat(L"chcr = %lx, madr = %lx, qwc = %lx, tadr = %1x", chcr._u32, madr, qwc, tadr);
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
enum INTCIrqs
|
enum INTCIrqs
|
||||||
|
|
228
pcsx2/Sif.cpp
228
pcsx2/Sif.cpp
|
@ -23,6 +23,7 @@
|
||||||
DMACh *sif0ch, *sif1ch, *sif2ch;
|
DMACh *sif0ch, *sif1ch, *sif2ch;
|
||||||
static _sif sif0, sif1;
|
static _sif sif0, sif1;
|
||||||
|
|
||||||
|
|
||||||
bool eesifbusy[2] = { false, false };
|
bool eesifbusy[2] = { false, false };
|
||||||
extern bool iopsifbusy[2];
|
extern bool iopsifbusy[2];
|
||||||
|
|
||||||
|
@ -34,6 +35,74 @@ void sifInit()
|
||||||
memzero(iopsifbusy);
|
memzero(iopsifbusy);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Various read/write functions. Could probably be reduced.
|
||||||
|
static __forceinline bool SifEERead(int &cycles)
|
||||||
|
{
|
||||||
|
tDMA_TAG *ptag;
|
||||||
|
int readSize = min((s32)sif0dma->qwc, (sif0.fifo.size >> 2));
|
||||||
|
|
||||||
|
//SIF_LOG(" EE SIF doing transfer %04Xqw to %08X", readSize, sif0dma->madr);
|
||||||
|
SIF_LOG("----------- %lX of %lX", readSize << 2, sif0dma->qwc << 2);
|
||||||
|
|
||||||
|
ptag = safeDmaGetAddr(sif0dma, sif0dma->madr, DMAC_SIF0);
|
||||||
|
if (ptag == NULL) return false;
|
||||||
|
|
||||||
|
sif0.fifo.read((u32*)ptag, readSize << 2);
|
||||||
|
|
||||||
|
// Clearing handled by vtlb memory protection and manual blocks.
|
||||||
|
//Cpu->Clear(sif0dma->madr, readSize*4);
|
||||||
|
|
||||||
|
sif0dma->madr += readSize << 4;
|
||||||
|
cycles += readSize; // fixme : BIAS is factored in above
|
||||||
|
sif0dma->qwc -= readSize;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
static __forceinline bool SifEEWrite(int &cycles)
|
||||||
|
{
|
||||||
|
// There's some data ready to transfer into the fifo..
|
||||||
|
tDMA_TAG *pTag;
|
||||||
|
|
||||||
|
const int writeSize = min((s32)sif1dma->qwc, (FIFO_SIF_W - sif1.fifo.size) / 4);
|
||||||
|
|
||||||
|
pTag = safeDmaGetAddr(sif1dma, sif1dma->madr, DMAC_SIF1);
|
||||||
|
if (pTag == NULL) return false;
|
||||||
|
|
||||||
|
sif1.fifo.write((u32*)pTag, writeSize << 2);
|
||||||
|
|
||||||
|
sif1dma->madr += writeSize << 4;
|
||||||
|
cycles += writeSize; // fixme : BIAS is factored in above
|
||||||
|
sif1dma->qwc -= writeSize;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
static __forceinline void SifIOPWrite(int &psxCycles)
|
||||||
|
{
|
||||||
|
// There's some data ready to transfer into the fifo..
|
||||||
|
int writeSize = min(sif0.counter, FIFO_SIF_W - sif0.fifo.size);
|
||||||
|
|
||||||
|
SIF_LOG("+++++++++++ %lX of %lX", writeSize, sif0.counter);
|
||||||
|
|
||||||
|
sif0.fifo.write((u32*)iopPhysMem(HW_DMA9_MADR), writeSize);
|
||||||
|
HW_DMA9_MADR += writeSize << 2;
|
||||||
|
psxCycles += (writeSize / 4) * BIAS; // fixme : should be / 16
|
||||||
|
sif0.counter -= writeSize;
|
||||||
|
}
|
||||||
|
|
||||||
|
static __forceinline void SifIOPRead(int &psxCycles)
|
||||||
|
{
|
||||||
|
// If we're reading something, continue to do so.
|
||||||
|
const int readSize = min (sif1.counter, sif1.fifo.size);
|
||||||
|
|
||||||
|
SIF_LOG(" IOP SIF doing transfer %04X to %08X", readSize, HW_DMA10_MADR);
|
||||||
|
|
||||||
|
sif1.fifo.read((u32*)iopPhysMem(HW_DMA10_MADR), readSize);
|
||||||
|
psxCpu->Clear(HW_DMA10_MADR, readSize);
|
||||||
|
HW_DMA10_MADR += readSize << 2;
|
||||||
|
psxCycles += readSize / 4; // fixme: should be / 16
|
||||||
|
sif1.counter -= readSize;
|
||||||
|
}
|
||||||
|
|
||||||
//General format of all the SIF#(EE/IOP)Dma functions is this:
|
//General format of all the SIF#(EE/IOP)Dma functions is this:
|
||||||
//First, SIF#Dma does a while loop, calling SIF#EEDma if EE is active,
|
//First, SIF#Dma does a while loop, calling SIF#EEDma if EE is active,
|
||||||
//and then calling SIF#IOPDma if IOP is active.
|
//and then calling SIF#IOPDma if IOP is active.
|
||||||
|
@ -55,7 +124,7 @@ void sifInit()
|
||||||
//
|
//
|
||||||
// This is, of course, simplified, and more study of these functions is needed.
|
// This is, of course, simplified, and more study of these functions is needed.
|
||||||
|
|
||||||
__forceinline void SIF0EEDma(int &cycles, bool &done)
|
static __forceinline void SIF0EEDma(int &cycles, bool &done)
|
||||||
{
|
{
|
||||||
#ifdef PCSX2_DEVBUILD
|
#ifdef PCSX2_DEVBUILD
|
||||||
if (dmacRegs->ctrl.STS == STS_SIF0)
|
if (dmacRegs->ctrl.STS == STS_SIF0)
|
||||||
|
@ -122,27 +191,11 @@ __forceinline void SIF0EEDma(int &cycles, bool &done)
|
||||||
|
|
||||||
if (sif0dma->qwc > 0) // If we're reading something continue to do so
|
if (sif0dma->qwc > 0) // If we're reading something continue to do so
|
||||||
{
|
{
|
||||||
tDMA_TAG *ptag;
|
SifEERead(cycles);
|
||||||
int readSize = min((s32)sif0dma->qwc, (sif0.fifo.size >> 2));
|
|
||||||
|
|
||||||
//SIF_LOG(" EE SIF doing transfer %04Xqw to %08X", readSize, sif0dma->madr);
|
|
||||||
SIF_LOG("----------- %lX of %lX", readSize << 2, sif0dma->qwc << 2);
|
|
||||||
|
|
||||||
ptag = safeDmaGetAddr(sif0dma, sif0dma->madr, DMAC_SIF0);
|
|
||||||
if (ptag == NULL) return;
|
|
||||||
|
|
||||||
sif0.fifo.read((u32*)ptag, readSize << 2);
|
|
||||||
|
|
||||||
// Clearing handled by vtlb memory protection and manual blocks.
|
|
||||||
//Cpu->Clear(sif0dma->madr, readSize*4);
|
|
||||||
|
|
||||||
cycles += readSize; // fixme : BIAS is factored in below
|
|
||||||
sif0dma->qwc -= readSize;
|
|
||||||
sif0dma->madr += readSize << 4;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
__forceinline void SIF1EEDma(int &cycles, bool &done)
|
static __forceinline void SIF1EEDma(int &cycles, bool &done)
|
||||||
{
|
{
|
||||||
#ifdef PCSX2_DEVBUILD
|
#ifdef PCSX2_DEVBUILD
|
||||||
if (dmacRegs->ctrl.STD == STD_SIF1)
|
if (dmacRegs->ctrl.STD == STD_SIF1)
|
||||||
|
@ -235,24 +288,13 @@ __forceinline void SIF1EEDma(int &cycles, bool &done)
|
||||||
}
|
}
|
||||||
if (sif1dma->qwc > 0)
|
if (sif1dma->qwc > 0)
|
||||||
{
|
{
|
||||||
// There's some data ready to transfer into the fifo..
|
SifEEWrite(cycles);
|
||||||
tDMA_TAG *pTag;
|
|
||||||
|
|
||||||
const int qwTransfer = min((s32)sif1dma->qwc, (FIFO_SIF_W - sif1.fifo.size) / 4);
|
|
||||||
|
|
||||||
pTag = safeDmaGetAddr(sif1dma, sif1dma->madr, DMAC_SIF1);
|
|
||||||
if (pTag == NULL) return;
|
|
||||||
|
|
||||||
sif1.fifo.write((u32*)pTag, qwTransfer << 2);
|
|
||||||
|
|
||||||
sif1dma->madr += qwTransfer << 4;
|
|
||||||
cycles += qwTransfer; // fixme : BIAS is factored in above
|
|
||||||
sif1dma->qwc -= qwTransfer;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Note: Test any changes in this function against Grandia III.
|
// Note: Test any changes in this function against Grandia III.
|
||||||
__forceinline void SIF0IOPDma(int &psxCycles, bool &done)
|
static __forceinline void SIF0IOPDma(int &psxCycles, bool &done)
|
||||||
{
|
{
|
||||||
if (sif0.counter <= 0) // If there's no more to transfer
|
if (sif0.counter <= 0) // If there's no more to transfer
|
||||||
{
|
{
|
||||||
|
@ -305,20 +347,11 @@ __forceinline void SIF0IOPDma(int &psxCycles, bool &done)
|
||||||
//if (sif0.counter > 0)
|
//if (sif0.counter > 0)
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
// There's some data ready to transfer into the fifo..
|
SifIOPWrite(psxCycles);
|
||||||
int wTransfer = min(sif0.counter, FIFO_SIF_W - sif0.fifo.size); // HW_DMA9_BCR >> 16;
|
|
||||||
|
|
||||||
SIF_LOG("+++++++++++ %lX of %lX", wTransfer, sif0.counter /*(HW_DMA9_BCR >> 16)*/);
|
|
||||||
|
|
||||||
sif0.fifo.write((u32*)iopPhysMem(HW_DMA9_MADR), wTransfer);
|
|
||||||
HW_DMA9_MADR += wTransfer << 2;
|
|
||||||
psxCycles += (wTransfer / 4) * BIAS; // fixme : should be / 16
|
|
||||||
sif0.counter -= wTransfer;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static __forceinline void SIF1IOPDma(int &psxCycles, bool &done)
|
||||||
__forceinline void SIF1IOPDma(int &psxCycles, bool &done)
|
|
||||||
{
|
{
|
||||||
if (sif1.counter <= 0)
|
if (sif1.counter <= 0)
|
||||||
{
|
{
|
||||||
|
@ -361,19 +394,17 @@ __forceinline void SIF1IOPDma(int &psxCycles, bool &done)
|
||||||
|
|
||||||
if (sif1.counter > 0)
|
if (sif1.counter > 0)
|
||||||
{
|
{
|
||||||
// If we're reading something, continue to do so.
|
SifIOPRead(psxCycles);
|
||||||
const int readSize = min (sif1.counter, sif1.fifo.size);
|
|
||||||
|
|
||||||
SIF_LOG(" IOP SIF doing transfer %04X to %08X", readSize, HW_DMA10_MADR);
|
|
||||||
|
|
||||||
sif1.fifo.read((u32*)iopPhysMem(HW_DMA10_MADR), readSize);
|
|
||||||
psxCpu->Clear(HW_DMA10_MADR, readSize);
|
|
||||||
psxCycles += readSize / 4; // fixme: should be / 16
|
|
||||||
sif1.counter -= readSize;
|
|
||||||
HW_DMA10_MADR += readSize << 2;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Tests if iop & ee are busy before the while statement,
|
||||||
|
// instead of during it, on the premise that anything that
|
||||||
|
// changes the busy state in here also ends the while statement.
|
||||||
|
//
|
||||||
|
// Mostly works, but breaks the Mana Khemia opening movie.
|
||||||
|
// Not sure if it's worth keeping yet.
|
||||||
|
//#define NO_BUSY_TEST
|
||||||
__forceinline void SIF0Dma()
|
__forceinline void SIF0Dma()
|
||||||
{
|
{
|
||||||
bool done = false;
|
bool done = false;
|
||||||
|
@ -381,42 +412,74 @@ __forceinline void SIF0Dma()
|
||||||
|
|
||||||
SIF_LOG("SIF0 DMA start...");
|
SIF_LOG("SIF0 DMA start...");
|
||||||
|
|
||||||
|
#ifdef NO_BUSY_TEST
|
||||||
|
if ((iopsifbusy[0]) && (eesifbusy[0]))
|
||||||
|
{
|
||||||
|
do
|
||||||
|
{
|
||||||
|
SIF0IOPDma(psxCycles, done);
|
||||||
|
SIF0EEDma(cycles, done);
|
||||||
|
} while (!done);
|
||||||
|
}
|
||||||
|
else if (iopsifbusy[0])
|
||||||
|
{
|
||||||
|
do
|
||||||
|
{
|
||||||
|
SIF0IOPDma(psxCycles, done);
|
||||||
|
} while (!done);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
do
|
||||||
|
{
|
||||||
|
SIF0EEDma(cycles, done);
|
||||||
|
} while (!done);
|
||||||
|
}
|
||||||
|
#else
|
||||||
do
|
do
|
||||||
{
|
{
|
||||||
if (iopsifbusy[0])
|
if (iopsifbusy[0]) SIF0IOPDma(psxCycles, done);
|
||||||
{
|
if (eesifbusy[0]) SIF0EEDma(cycles, done);
|
||||||
// If EE SIF0 is enabled.
|
} while (!done);
|
||||||
SIF0IOPDma(psxCycles, done);
|
#endif
|
||||||
}
|
|
||||||
|
|
||||||
if (eesifbusy[0])
|
|
||||||
{
|
|
||||||
// If EE SIF enabled and there's something to transfer.
|
|
||||||
SIF0EEDma(cycles, done);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
while (!done);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
__forceinline void SIF1Dma()
|
__forceinline void SIF1Dma()
|
||||||
{
|
{
|
||||||
bool done = false;
|
bool done = false;
|
||||||
int cycles = 0, psxCycles = 0;
|
int cycles = 0, psxCycles = 0;
|
||||||
|
|
||||||
|
#ifdef NO_BUSY_TEST
|
||||||
|
if ((iopsifbusy[1]) && (eesifbusy[1]))
|
||||||
|
{
|
||||||
|
do
|
||||||
|
{
|
||||||
|
SIF1EEDma(cycles, done);
|
||||||
|
SIF1IOPDma(psxCycles, done);
|
||||||
|
} while (!done);
|
||||||
|
}
|
||||||
|
else if (iopsifbusy[1])
|
||||||
|
{
|
||||||
|
do
|
||||||
|
{
|
||||||
|
SIF1IOPDma(psxCycles, done);
|
||||||
|
} while (!done);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
do
|
||||||
|
{
|
||||||
|
SIF1EEDma(cycles, done);
|
||||||
|
} while (!done);
|
||||||
|
}
|
||||||
|
#else
|
||||||
do
|
do
|
||||||
{
|
{
|
||||||
if (eesifbusy[1])
|
if (eesifbusy[1]) SIF1EEDma(cycles, done);
|
||||||
{
|
if (iopsifbusy[1]) SIF1IOPDma(psxCycles, done);
|
||||||
// If EE SIF1 is enabled.
|
|
||||||
SIF1EEDma(cycles, done);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (iopsifbusy[1])
|
} while (!done);
|
||||||
{
|
#endif
|
||||||
// If IOP SIF enabled and there's something to transfer.
|
|
||||||
SIF1IOPDma(psxCycles, done);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
while (!done);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
__forceinline void sif0Interrupt()
|
__forceinline void sif0Interrupt()
|
||||||
|
@ -445,8 +508,7 @@ __forceinline void EEsif1Interrupt()
|
||||||
|
|
||||||
__forceinline void dmaSIF0()
|
__forceinline void dmaSIF0()
|
||||||
{
|
{
|
||||||
SIF_LOG("EE: dmaSIF0 chcr = %lx, madr = %lx, qwc = %lx, tadr = %lx",
|
SIF_LOG(wxString(L"dmaSIF0" + sif0dma->cmqt_to_str()).To8BitData());
|
||||||
sif0dma->chcr._u32, sif0dma->madr, sif0dma->qwc, sif0dma->tadr);
|
|
||||||
|
|
||||||
if (sif0.fifo.readPos != sif0.fifo.writePos)
|
if (sif0.fifo.readPos != sif0.fifo.writePos)
|
||||||
{
|
{
|
||||||
|
@ -469,8 +531,7 @@ __forceinline void dmaSIF0()
|
||||||
|
|
||||||
__forceinline void dmaSIF1()
|
__forceinline void dmaSIF1()
|
||||||
{
|
{
|
||||||
SIF_LOG("EE: dmaSIF1 chcr = %lx, madr = %lx, qwc = %lx, tadr = %lx",
|
SIF_LOG(wxString(L"dmaSIF1" + sif1dma->cmqt_to_str()).To8BitData());
|
||||||
sif1dma->chcr._u32, sif1dma->madr, sif1dma->qwc, sif1dma->tadr);
|
|
||||||
|
|
||||||
if (sif1.fifo.readPos != sif1.fifo.writePos)
|
if (sif1.fifo.readPos != sif1.fifo.writePos)
|
||||||
{
|
{
|
||||||
|
@ -494,8 +555,7 @@ __forceinline void dmaSIF1()
|
||||||
|
|
||||||
__forceinline void dmaSIF2()
|
__forceinline void dmaSIF2()
|
||||||
{
|
{
|
||||||
SIF_LOG("dmaSIF2 chcr = %lx, madr = %lx, qwc = %lx",
|
SIF_LOG(wxString(L"dmaSIF2" + sif2dma->cmq_to_str()).To8BitData());
|
||||||
sif2dma->chcr._u32, sif2dma->madr, sif2dma->qwc);
|
|
||||||
|
|
||||||
sif2dma->chcr.STR = false;
|
sif2dma->chcr.STR = false;
|
||||||
hwDmacIrq(DMAC_SIF2);
|
hwDmacIrq(DMAC_SIF2);
|
||||||
|
|
Loading…
Reference in New Issue