pcsx2/PsxDma.c

255 lines
7.4 KiB
C

/* Pcsx2 - Pc Ps2 Emulator
* Copyright (C) 2002-2004 Pcsx2 Team
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/
#include <string.h>
#include "PsxCommon.h"
// Dma0/1 in Mdec.c
// Dma3 in CdRom.c
// Dma8 in PsxSpd.c
// Dma11/12 in PsxSio2.c
//static int spudmaenable[2];
void psxDma4(u32 madr, u32 bcr, u32 chcr) { // SPU
int size;
if(chcr & 0x400) SysPrintf("SPU 2 DMA 4 linked list chain mode! chcr = %x madr = %x bcr = %x\n", chcr, madr, bcr);
if(chcr & 0x40000000) SysPrintf("SPU 2 DMA 4 Unusual bit set on 'to' direction chcr = %x madr = %x bcr = %x\n", chcr, madr, bcr);
if((chcr & 0x1) == 0) SysPrintf("SPU 2 DMA 4 loading from spu2 memory chcr = %x madr = %x bcr = %x\n", chcr, madr, bcr);
switch (chcr) {
case 0x01000201: //cpu to spu transfer
#ifdef PSXDMA_LOG
PSXDMA_LOG("*** DMA 4 - SPU mem2spu *** %lx addr = %lx size = %lx\n", chcr, madr, bcr);
#endif
//SysPrintf("DMA4 write blocks %x, size per block %x\n", (bcr >> 16), (bcr & 0xFFFF));
size = (bcr >> 16) * (bcr & 0xFFFF); // Number of blocks to transfer
SPU2writeDMA4Mem((u16 *)PSXM(madr), size);
HW_DMA4_MADR += size;
break;
case 0x01000200: //spu to cpu transfer
#ifdef PSXDMA_LOG
PSXDMA_LOG("*** DMA 4 - SPU spu2mem *** %lx addr = %lx size = %lx\n", chcr, madr, bcr);
#endif
//SysPrintf("DMA4 read blocks %x, size per block %x\n", (bcr >> 16), (bcr & 0xFFFF));
size = (bcr >> 16) * (bcr & 0xFFFF); // Number of blocks to transfer
SPU2readDMA4Mem((u16 *)PSXM(madr), size);
psxCpu->Clear(HW_DMA4_MADR, size);
HW_DMA4_MADR += size;
break;
default:
SysPrintf("*** DMA 4 - SPU unknown *** %lx addr = %lx size = %lx\n", chcr, madr, bcr);
break;
}
//spudmaenable[0] = size;
}
int psxDma4Interrupt() {
HW_DMA4_CHCR &= ~0x01000000;
psxDmaInterrupt(4);
psxHu32(0x1070)|= 1<<9;
return 1;
}
void psxDma2(u32 madr, u32 bcr, u32 chcr) { // GPU
HW_DMA2_CHCR &= ~0x01000000;
psxDmaInterrupt(2);
}
void psxDma6(u32 madr, u32 bcr, u32 chcr) {
u32 *mem = (u32 *)PSXM(madr);
#ifdef PSXDMA_LOG
PSXDMA_LOG("*** DMA 6 - OT *** %lx addr = %lx size = %lx\n", chcr, madr, bcr);
#endif
if (chcr == 0x11000002) {
while (bcr--) {
*mem-- = (madr - 4) & 0xffffff;
madr -= 4;
}
mem++; *mem = 0xffffff;
} else {
// Unknown option
#ifdef PSXDMA_LOG
PSXDMA_LOG("*** DMA 6 - OT unknown *** %lx addr = %lx size = %lx\n", chcr, madr, bcr);
#endif
}
HW_DMA6_CHCR &= ~0x01000000;
psxDmaInterrupt(6);
}
void psxDma7(u32 madr, u32 bcr, u32 chcr) {
int size;
if(chcr & 0x400) SysPrintf("SPU 2 DMA 7 linked list chain mode! chcr = %x madr = %x bcr = %x\n", chcr, madr, bcr);
if(chcr & 0x40000000) SysPrintf("SPU 2 DMA 7 Unusual bit set on 'to' direction chcr = %x madr = %x bcr = %x\n", chcr, madr, bcr);
if((chcr & 0x1) == 0) SysPrintf("SPU 2 DMA 7 loading from spu2 memory chcr = %x madr = %x bcr = %x\n", chcr, madr, bcr);
switch (chcr) {
case 0x01000201: //cpu to spu2 transfer
#ifdef PSXDMA_LOG
PSXDMA_LOG("*** DMA 7 - SPU2 mem2spu *** %lx addr = %lx size = %lx\n", chcr, madr, bcr);
#endif
//SysPrintf("DMA7 write blocks %x, size per block %x\n", (bcr >> 16), (bcr & 0xFFFF));
size = (bcr >> 16) * (bcr & 0xFFFF); // Number of blocks to transfer
SPU2writeDMA7Mem((u16 *)PSXM(madr), size);
HW_DMA7_MADR += size;
break;
case 0x01000200: //spu2 to cpu transfer
#ifdef PSXDMA_LOG
PSXDMA_LOG("*** DMA 7 - SPU2 spu2mem *** %lx addr = %lx size = %lx\n", chcr, madr, bcr);
#endif
//SysPrintf("DMA4 read blocks %x, size per block %x\n", (bcr >> 16), (bcr & 0xFFFF));
size = (bcr >> 16) * (bcr & 0xFFFF); // Number of blocks to transfer
SPU2readDMA7Mem((u16 *)PSXM(madr), size);
psxCpu->Clear(HW_DMA7_MADR, size);
HW_DMA7_MADR += size;
break;
default:
SysPrintf("*** DMA 7 - SPU unknown *** %lx addr = %lx size = %lx\n", chcr, madr, bcr);
break;
}
}
int psxDma7Interrupt() {
HW_DMA7_CHCR &= ~0x01000000;
psxDmaInterrupt2(0);
return 1;
}
void psxDma9(u32 madr, u32 bcr, u32 chcr) {
DMACh *dma = (DMACh*)&PS2MEM_HW[0xc000];
#ifdef SIF_LOG
SIF_LOG("IOP: dmaSIF0 chcr = %lx, madr = %lx, bcr = %lx, tadr = %lx\n", chcr, madr, bcr, HW_DMA9_TADR);
#endif
psHu32(0x1000F240) |= 0x2000;
if (dma->chcr & 0x100 && HW_DMA9_CHCR & 0x01000000) {
SIF0Dma();
psHu32(0x1000F240) &= ~0x20;
psHu32(0x1000F240) &= ~0x2000;
}
}
void psxDma10(u32 madr, u32 bcr, u32 chcr) {
DMACh *dma = (DMACh*)&PS2MEM_HW[0xc400];
#ifdef SIF_LOG
SIF_LOG("IOP: dmaSIF1 chcr = %lx, madr = %lx, bcr = %lx\n", chcr, madr, bcr);
#endif
psHu32(0x1000F240) |= 0x4000;
if (dma->chcr & 0x100 && HW_DMA10_CHCR & 0x01000000) {
SIF1Dma();
psHu32(0x1000F240) &= ~0x40;
psHu32(0x1000F240) &= ~0x100;
psHu32(0x1000F240) &= ~0x4000;
}
}
void psxDma8(u32 madr, u32 bcr, u32 chcr) {
int size;
switch (chcr & 0x01000201) {
case 0x01000201: //cpu to dev9 transfer
#ifdef PSXDMA_LOG
PSXDMA_LOG("*** DMA 8 - DEV9 mem2dev9 *** %lx addr = %lx size = %lx\n", chcr, madr, bcr);
#endif
size = (bcr >> 16) * (bcr & 0xFFFF); // Number of blocks to transfer
DEV9writeDMA8Mem((u32*)PSXM(madr), size*8);
break;
case 0x01000200: //dev9 to cpu transfer
#ifdef PSXDMA_LOG
PSXDMA_LOG("*** DMA 8 - DEV9 dev9mem *** %lx addr = %lx size = %lx\n", chcr, madr, bcr);
#endif
size = (bcr >> 16) * (bcr & 0xFFFF); // Number of blocks to transfer
DEV9readDMA8Mem((u32*)PSXM(madr), size*8);
break;
#ifdef PSXDMA_LOG
default:
PSXDMA_LOG("*** DMA 8 - DEV9 unknown *** %lx addr = %lx size = %lx\n", chcr, madr, bcr);
break;
#endif
}
HW_DMA8_CHCR &= ~0x01000000;
psxDmaInterrupt2(1);
}
int dev9Interrupt() {
if (dev9Handler == NULL) goto irq;
if (dev9Handler() != 1) return 1;
irq:
psxHu32(0x1070)|= 1<<13;
return 1;
}
void dev9Irq(int cycles) {
PSX_INT(20, (PSXCLK/cycles));
}
int usbInterrupt() {
if (usbHandler == NULL) goto irq;
if (usbHandler() != 1) return 1;
irq:
psxHu32(0x1070)|= 1<<22;
return 1;
}
void usbIrq(int cycles) {
PSX_INT(21, (PSXCLK/cycles));
}
void fwIrq() {
psxHu32(0x1070)|= 1<<24;
}
void spu2DMA4Irq() {
//SysPrintf("IRQ DMA 4\n");
SPU2interruptDMA4();
//if(SPU2read(0x1f90019a) & 0xff) PSX_INT(4, (spudmaenable[0] * PSXSOUNDCLK) / BIAS);
HW_DMA4_BCR = 0;
HW_DMA4_CHCR &= ~0x01000000;
psxDmaInterrupt(4);
//psxHu32(0x1070)|= 1<<9;
}
void spu2DMA7Irq() {
//SysPrintf("IRQ DMA 7\n");
SPU2interruptDMA7();
//if(SPU2read(0x1f90059a) & 0xff) PSX_INT(7, (spudmaenable[1] * PSXSOUNDCLK) / BIAS);
HW_DMA7_BCR = 0;
HW_DMA7_CHCR &= ~0x01000000;
psxDmaInterrupt2(0);
//psxHu32(0x1070)|= 1<<9;
}
void spu2Irq() {
psxHu32(0x1070)|= 1<<9;
}