totally rewrite main emulation loop! YOUR SAVESTATES ARE INVALIDATED. your timing is all different. but, your timing is more precise, a few hidden old bugs surfaced and were smashed, and some games work that didn't used to. INSTRUCTIONS_PER_BATCH is a thing of the past. crazymax and luigi, I may have broken your parts, but hopefully only a little bit. your new hookups into the emulation loop are more complicated but potentially faster and definitely more precise.

This commit is contained in:
zeromus 2009-07-15 19:42:32 +00:00
parent b54420e5a7
commit 28435d9334
16 changed files with 1272 additions and 1633 deletions

View File

@ -1,21 +1,31 @@
0.9.4 -> ??? (r2437-r???) 0.9.4 -> ??? (r2437-r???)
??? introduces an entirely rewritten main emulation loop.
This totally changes the timing, and totally breaks old savestates.
Highlights: Highlights:
* win32: lua engine, path configuration, 7z dearchiving support * win32: lua engine, path configuration, 7z dearchiving support
* rewritten main emulation loop
General/Core: General/Core:
bug: fix cflash directory support for non-windows bug: fix cflash directory support for non-windows
bug: fix freeze in cart irq bug: fix freeze in cart irq
bug: correctly emulate dma to/from tcm bug: correctly emulate dma to/from tcm
enh: add guitar grip emulation
enh: add more powerful antigrain-based drawing library and rewrite OSD system
enh: ideas-style debugging prints enh: ideas-style debugging prints
Graphics: Graphics:
bug: fixing of obj blending and bmp obj rendering bug: fixing of obj blending and bmp obj rendering
bug: fix backdrop blending with garbage
bug: fix 256B granularity sprite addressing for sub gpu
bug: swrast: add clear image emulation bug: swrast: add clear image emulation
bug: swrast: add edge marking bug: swrast: add edge marking
Windows: Windows:
bug: improve map view tool bug: improve map view tool to support more modes
enh: added 2x resizing filters (hq2x, 2xsai, supereagle, scanlines)
enh: soundview can now mute channels
Linux: Linux:
enh: alsa microphone support enh: alsa microphone support

View File

@ -28,6 +28,7 @@
#include "debug.h" #include "debug.h"
#include "mem.h" #include "mem.h"
#include "MMU.h" #include "MMU.h"
#include "NDSSystem.h"
// ========================================================= IPC FIFO // ========================================================= IPC FIFO
IPC_FIFO ipc_fifo[2]; // 0 - ARM9 IPC_FIFO ipc_fifo[2]; // 0 - ARM9
@ -174,6 +175,8 @@ void GFX_FIFOsend(u8 cmd, u32 param)
#endif #endif
T1WriteLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0x600, gxstat); T1WriteLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0x600, gxstat);
NDS_RescheduleGXFIFO();
} }
BOOL GFX_FIFOrecv(u8 *cmd, u32 *param) BOOL GFX_FIFOrecv(u8 *cmd, u32 *param)
@ -183,6 +186,7 @@ BOOL GFX_FIFOrecv(u8 *cmd, u32 *param)
if (gxstat & 0xC0000000) if (gxstat & 0xC0000000)
{ {
setIF(0, (1<<21)); setIF(0, (1<<21));
//NDS_makeARM9Int(21);
} }
#endif #endif
if (gxFIFO.tail == 0) // empty if (gxFIFO.tail == 0) // empty
@ -193,18 +197,23 @@ BOOL GFX_FIFOrecv(u8 *cmd, u32 *param)
if ((gxstat & 0x80000000)) // empty if ((gxstat & 0x80000000)) // empty
{ {
setIF(0, (1<<21)); setIF(0, (1<<21));
//NDS_makeARM9Int(21);
} }
return FALSE; return FALSE;
} }
if (gxstat & 0x40000000) // IRQ: less half if (gxstat & 0x40000000) // IRQ: less half
{ {
if (gxstat & 0x02000000) setIF(0, (1<<21)); if (gxstat & 0x02000000)
setIF(0, (1<<21));
//NDS_makeARM9Int(21);
} }
if ((gxstat & 0x80000000)) // IRQ: empty if ((gxstat & 0x80000000)) // IRQ: empty
{ {
if (gxstat & 0x04000000) setIF(0, (1<<21)); if (gxstat & 0x04000000)
setIF(0, (1<<21));
//NDS_makeARM9Int(21);
} }
gxstat &= 0xF000FFFF; gxstat &= 0xF000FFFF;
@ -228,6 +237,8 @@ BOOL GFX_FIFOrecv(u8 *cmd, u32 *param)
T1WriteLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0x600, gxstat); T1WriteLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0x600, gxstat);
NDS_RescheduleGXFIFO();
return TRUE; return TRUE;
} }
@ -242,9 +253,12 @@ void GFX_FIFOcnt(u32 val)
} }
T1WriteLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0x600, gxstat); T1WriteLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0x600, gxstat);
NDS_RescheduleGXFIFO();
/*if (gxstat & 0xC0000000) /*if (gxstat & 0xC0000000)
{ {
setIF(0, (1<<21)); setIF(0, (1<<21));
//NDS_makeARM9Int(21);
}*/ }*/
} }

View File

@ -410,23 +410,8 @@ u8 *MMU_RenderMapToLCD(u32 vram_addr)
} }
template<u8 DMA_CHANNEL>
void DMAtoVRAMmapping()
{
//THIS IS ALSO DANGEROUS!!!!!!
//but i dont think it needs to be done
/*u32 dst = DMADst[ARMCPU_ARM9][DMA_CHANNEL];
bool unmapped;
dst = MMU_LCDmap(dst,unmapped);
DMADst[ARMCPU_ARM9][DMA_CHANNEL] = dst;*/
}
#define LOG_VRAM_ERROR() LOG("No data for block %i MST %i\n", block, VRAMBankCnt & 0x07); #define LOG_VRAM_ERROR() LOG("No data for block %i MST %i\n", block, VRAMBankCnt & 0x07);
struct VramConfiguration { struct VramConfiguration {
enum Purpose { enum Purpose {
@ -983,7 +968,7 @@ void MMU_clearMem()
memset(MMU.reg_IF, 0, sizeof(u32) * 2); memset(MMU.reg_IF, 0, sizeof(u32) * 2);
memset(MMU.DMAStartTime, 0, sizeof(u32) * 2 * 4); memset(MMU.DMAStartTime, 0, sizeof(u32) * 2 * 4);
memset(MMU.DMACycle, 0, sizeof(s32) * 2 * 4); memset(MMU.DMACycle, 0, sizeof(MMU.DMACycle));
memset(MMU.DMACrt, 0, sizeof(u32) * 2 * 4); memset(MMU.DMACrt, 0, sizeof(u32) * 2 * 4);
memset(MMU.DMAing, 0, sizeof(BOOL) * 2 * 4); memset(MMU.DMAing, 0, sizeof(BOOL) * 2 * 4);
@ -1011,9 +996,7 @@ void MMU_clearMem()
partie = 1; partie = 1;
addonsReset(); addonsReset();
Mic_Reset(); Mic_Reset();
#ifdef USE_GEOMETRY_FIFO_EMULATION
MMU.gfx3dCycles = 0; MMU.gfx3dCycles = 0;
#endif
} }
void MMU_setRom(u8 * rom, u32 mask) void MMU_setRom(u8 * rom, u32 mask)
@ -1062,10 +1045,11 @@ static void execsqrt() {
T1WriteLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0x2B4, 0); T1WriteLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0x2B4, 0);
T1WriteLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0x2B0, cnt | 0x8000); T1WriteLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0x2B0, cnt | 0x8000);
MMU.sqrtCycles = (nds.cycles + 26); MMU.sqrtCycles = nds_timer + 26;
MMU.sqrtResult = ret; MMU.sqrtResult = ret;
MMU.sqrtCnt = (cnt & 0x7FFF); MMU.sqrtCnt = (cnt & 0x7FFF);
MMU.sqrtRunning = TRUE; MMU.sqrtRunning = TRUE;
NDS_Reschedule();
} }
static void execdiv() { static void execdiv() {
@ -1078,19 +1062,19 @@ static void execdiv() {
case 0: case 0:
num = (s64) (s32) T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0x290); num = (s64) (s32) T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0x290);
den = (s64) (s32) T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0x298); den = (s64) (s32) T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0x298);
MMU.divCycles = (nds.cycles + 36); MMU.divCycles = nds_timer + 36;
break; break;
case 1: case 1:
case 3: //gbatek says this is same as mode 1 case 3: //gbatek says this is same as mode 1
num = (s64) T1ReadQuad(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0x290); num = (s64) T1ReadQuad(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0x290);
den = (s64) (s32) T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0x298); den = (s64) (s32) T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0x298);
MMU.divCycles = (nds.cycles + 68); MMU.divCycles = nds_timer + 68;
break; break;
case 2: case 2:
default: default:
num = (s64) T1ReadQuad(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0x290); num = (s64) T1ReadQuad(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0x290);
den = (s64) T1ReadQuad(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0x298); den = (s64) T1ReadQuad(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0x298);
MMU.divCycles = (nds.cycles + 68); MMU.divCycles = nds_timer + 68;
break; break;
} }
@ -1122,6 +1106,7 @@ static void execdiv() {
MMU.divMod = mod; MMU.divMod = mod;
MMU.divCnt = (cnt & 0x7FFF); MMU.divCnt = (cnt & 0x7FFF);
MMU.divRunning = TRUE; MMU.divRunning = TRUE;
NDS_Reschedule();
} }
template<int PROCNUM> template<int PROCNUM>
@ -1150,19 +1135,21 @@ void FASTCALL MMU_doDMA(u32 num)
taille = (MMU.DMACrt[PROCNUM][num]&0x1FFFFF); taille = (MMU.DMACrt[PROCNUM][num]&0x1FFFFF);
if(taille == 0) taille = 0x200000; //according to gbatek.. if(taille == 0) taille = 0x200000; //according to gbatek..
//THIS IS A BIG HACK
// If we are in "Main memory display" mode just copy an entire // If we are in "Main memory display" mode just copy an entire
// screen (256x192 pixels). // screen (256x192 pixels).
// Reference: http://nocash.emubase.de/gbatek.htm#dsvideocaptureandmainmemorydisplaymode // Reference: http://nocash.emubase.de/gbatek.htm#dsvideocaptureandmainmemorydisplaymode
// (under DISP_MMEM_FIFO) // (under DISP_MMEM_FIFO)
if ((MMU.DMAStartTime[PROCNUM][num]==4) && // Must be in main memory display mode if ((MMU.DMAStartTime[PROCNUM][num]==EDMAMode_MemDisplay) && // Must be in main memory display mode
(taille==4) && // Word must be 4 (taille==4) && // Word must be 4
(((MMU.DMACrt[PROCNUM][num]>>26)&1) == 1)) // Transfer mode must be 32bit wide (((MMU.DMACrt[PROCNUM][num]>>26)&1) == 1)) // Transfer mode must be 32bit wide
taille = 24576; //256*192/2; taille = 24576; //256*192/2;
if(MMU.DMAStartTime[PROCNUM][num] == 5) if(MMU.DMAStartTime[PROCNUM][num] == EDMAMode_Card)
taille *= 0x80; taille *= 0x80;
MMU.DMACycle[PROCNUM][num] = taille + nds.cycles; MMU.DMACycle[PROCNUM][num] = taille + nds_timer; //TODO - surely this is a gross simplification
MMU.DMAing[PROCNUM][num] = TRUE; MMU.DMAing[PROCNUM][num] = TRUE;
MMU.CheckDMAs |= (1<<(num+(PROCNUM<<2))); MMU.CheckDMAs |= (1<<(num+(PROCNUM<<2)));
@ -1173,6 +1160,8 @@ void FASTCALL MMU_doDMA(u32 num)
if(!(MMU.DMACrt[PROCNUM][num]&(1<<25))) if(!(MMU.DMACrt[PROCNUM][num]&(1<<25)))
MMU.DMAStartTime[PROCNUM][num] = 0; MMU.DMAStartTime[PROCNUM][num] = 0;
NDS_RescheduleDMA();
// transfer // transfer
{ {
u32 i=0; u32 i=0;
@ -1315,6 +1304,97 @@ static INLINE void MMU_IPCSync(u8 proc, u32 val)
setIF(proc^1, ( 1 << 16 )); setIF(proc^1, ( 1 << 16 ));
} }
static INLINE u16 read_timer(int proc, int timerIndex)
{
//chained timers are always up to date
if(MMU.timerMODE[proc][timerIndex] == 0xFFFF)
return MMU.timer[proc][timerIndex];
//for unchained timers, we do not keep the timer up to date. its value will need to be calculated here
s32 diff = (s32)(nds.timerCycle[proc][timerIndex] - nds_timer);
assert(diff>=0);
if(diff<0)
printf("NEW EMULOOP BAD NEWS PLEASE REPORT: TIME READ DIFF < 0 (%d) (%d) (%d)\n",diff,timerIndex,MMU.timerMODE[proc][timerIndex]);
s32 units = diff / (1<<MMU.timerMODE[proc][timerIndex]);
s32 ret = 65535 - units;
assert(ret>=0);
if(ret<0) printf("NEW EMULOOP BAD NEWS PLEASE REPORT: TIME READ RETURN < 0\n");
return ret;
}
static INLINE void write_timer(int proc, int timerIndex, u16 val)
{
int mask = ((val&0x80)>>7) << timerIndex;
//MMU.CheckTimers = (MMU.CheckTimers & (~mask)) | mask;
if(val&0x80)
MMU.timer[proc][timerIndex] = MMU.timerReload[proc][timerIndex];
MMU.timerON[proc][timerIndex] = val & 0x80;
switch(val&7)
{
case 0 :
MMU.timerMODE[proc][timerIndex] = 0+1;
break;
case 1 :
MMU.timerMODE[proc][timerIndex] = 6+1;
break;
case 2 :
MMU.timerMODE[proc][timerIndex] = 8+1;
break;
case 3 :
MMU.timerMODE[proc][timerIndex] = 10+1;
break;
default :
MMU.timerMODE[proc][timerIndex] = 0xFFFF;
break;
}
int remain = 65536 - MMU.timerReload[proc][timerIndex];
nds.timerCycle[proc][timerIndex] = nds_timer + (remain<<MMU.timerMODE[proc][timerIndex]);
T1WriteWord(MMU.MMU_MEM[proc][0x40], 0x102+timerIndex*4, val);
NDS_RescheduleTimers();
}
template<int proc> static INLINE void write_dma_hictrl(const int dmanum, const u16 val)
{
u32 baseAddr = 0xB0 + dmanum*12;
//write this control value
T1WriteWord(MMU.MMU_MEM[proc][0x40], baseAddr+10, val);
//read back the src and dst addr
DMASrc[proc][dmanum] = T1ReadLong(MMU.MMU_MEM[proc][0x40], baseAddr);
DMADst[proc][dmanum] = T1ReadLong(MMU.MMU_MEM[proc][0x40], baseAddr+4);
//analyze the control value
u32 v = T1ReadLong(MMU.MMU_MEM[proc][0x40], baseAddr+8);
if(proc==ARMCPU_ARM9) MMU.DMAStartTime[proc][dmanum] = (v>>27) & 0x7;
else {
static const EDMAMode lookup[] = {EDMAMode_Immediate,EDMAMode_VBlank,EDMAMode_Card,EDMAMode7_Wifi};
MMU.DMAStartTime[proc][dmanum] = lookup[(v>>28) & 0x3];
if(MMU.DMAStartTime[proc][dmanum] == EDMAMode7_Wifi && (dmanum==1 || dmanum==3))
MMU.DMAStartTime[proc][dmanum] = EDMAMode7_GBASlot;
}
MMU.DMACrt[proc][dmanum] = v;
if(MMU.DMAStartTime[proc][dmanum] == EDMAMode_Immediate
//TODO HACK: I think this is a gxfifo hack:
|| MMU.DMAStartTime[proc][dmanum] == EDMAMode_GXFifo)
{
MMU_doDMA<proc>(dmanum);
}
//printf("dma ctrl %d %d\n",proc,dmanum);
//LOG("ARMCPU_ARM9 %d, dma %d src %08X dst %08X %s\r\n", ARMCPU_ARM9, 0, DMASrc[ARMCPU_ARM9][0], DMADst[ARMCPU_ARM9][0], (val&(1<<25))?"ON":"OFF");
NDS_RescheduleDMA();
}
static INLINE void write_auxspicnt(const int proc, const int size, const int adr, const int val) static INLINE void write_auxspicnt(const int proc, const int size, const int adr, const int val)
{ {
//why val==0 to reset? is it a particular bit? its not bit 6... //why val==0 to reset? is it a particular bit? its not bit 6...
@ -1869,6 +1949,7 @@ void FASTCALL _MMU_ARM9_write16(u32 adr, u16 val)
case REG_IME: case REG_IME:
{ {
NDS_Reschedule();
u32 old_val = MMU.reg_IME[ARMCPU_ARM9]; u32 old_val = MMU.reg_IME[ARMCPU_ARM9];
u32 new_val = val & 0x01; u32 new_val = val & 0x01;
MMU.reg_IME[ARMCPU_ARM9] = new_val; MMU.reg_IME[ARMCPU_ARM9] = new_val;
@ -1887,6 +1968,7 @@ void FASTCALL _MMU_ARM9_write16(u32 adr, u16 val)
return; return;
} }
case REG_IE : case REG_IE :
NDS_Reschedule();
MMU.reg_IE[ARMCPU_ARM9] = (MMU.reg_IE[ARMCPU_ARM9]&0xFFFF0000) | val; MMU.reg_IE[ARMCPU_ARM9] = (MMU.reg_IE[ARMCPU_ARM9]&0xFFFF0000) | val;
#ifndef NEW_IRQ #ifndef NEW_IRQ
if ( MMU.reg_IME[ARMCPU_ARM9]) if ( MMU.reg_IME[ARMCPU_ARM9])
@ -1901,6 +1983,7 @@ void FASTCALL _MMU_ARM9_write16(u32 adr, u16 val)
#endif #endif
return; return;
case REG_IE + 2 : case REG_IE + 2 :
NDS_Reschedule();
MMU.reg_IE[ARMCPU_ARM9] = (MMU.reg_IE[ARMCPU_ARM9]&0xFFFF) | (((u32)val)<<16); MMU.reg_IE[ARMCPU_ARM9] = (MMU.reg_IE[ARMCPU_ARM9]&0xFFFF) | (((u32)val)<<16);
#ifndef NEW_IRQ #ifndef NEW_IRQ
if ( MMU.reg_IME[ARMCPU_ARM9]) if ( MMU.reg_IME[ARMCPU_ARM9])
@ -1916,9 +1999,11 @@ void FASTCALL _MMU_ARM9_write16(u32 adr, u16 val)
return; return;
case REG_IF : case REG_IF :
NDS_Reschedule();
MMU.reg_IF[ARMCPU_ARM9] &= (~((u32)val)); MMU.reg_IF[ARMCPU_ARM9] &= (~((u32)val));
return; return;
case REG_IF + 2 : case REG_IF + 2 :
NDS_Reschedule();
MMU.reg_IF[ARMCPU_ARM9] &= (~(((u32)val)<<16)); MMU.reg_IF[ARMCPU_ARM9] &= (~(((u32)val)<<16));
return; return;
@ -1941,37 +2026,7 @@ void FASTCALL _MMU_ARM9_write16(u32 adr, u16 val)
case REG_TM3CNTH : case REG_TM3CNTH :
{ {
int timerIndex = ((adr-2)>>2)&0x3; int timerIndex = ((adr-2)>>2)&0x3;
int mask = ((val&0x80)>>7) << timerIndex; write_timer(ARMCPU_ARM9, timerIndex, val);
MMU.CheckTimers = (MMU.CheckTimers & (~mask)) | mask;
if(val&0x80)
MMU.timer[ARMCPU_ARM9][timerIndex] = MMU.timerReload[ARMCPU_ARM9][((adr-2)>>2)&0x3];
MMU.timerON[ARMCPU_ARM9][((adr-2)>>2)&0x3] = val & 0x80;
switch(val&7)
{
case 0 :
MMU.timerMODE[ARMCPU_ARM9][timerIndex] = 0+1;//ARMCPU_ARM9;
break;
case 1 :
MMU.timerMODE[ARMCPU_ARM9][timerIndex] = 6+1;//ARMCPU_ARM9;
break;
case 2 :
MMU.timerMODE[ARMCPU_ARM9][timerIndex] = 8+1;//ARMCPU_ARM9;
break;
case 3 :
MMU.timerMODE[ARMCPU_ARM9][timerIndex] = 10+1;//ARMCPU_ARM9;
break;
default :
MMU.timerMODE[ARMCPU_ARM9][timerIndex] = 0xFFFF;
break;
}
if(!(val & 0x80))
MMU.timerRUN[ARMCPU_ARM9][timerIndex] = FALSE;
T1WriteWord(MMU.MMU_MEM[ARMCPU_ARM9][0x40], adr & 0xFFF, val);
return; return;
} }
@ -2021,95 +2076,17 @@ void FASTCALL _MMU_ARM9_write16(u32 adr, u16 val)
} }
case REG_DMA0CNTH : case REG_DMA0CNTH :
{ write_dma_hictrl<ARMCPU_ARM9>(0,val);
u32 v;
//if(val&0x8000) emu_halt();
//LOG("16 bit dma0 %04X\r\n", val);
T1WriteWord(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0xBA, val);
DMASrc[ARMCPU_ARM9][0] = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0xB0);
DMADst[ARMCPU_ARM9][0] = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0xB4);
DMAtoVRAMmapping<0>();
v = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0xB8);
MMU.DMAStartTime[ARMCPU_ARM9][0] = (v>>27) & 0x7;
MMU.DMACrt[ARMCPU_ARM9][0] = v;
if(MMU.DMAStartTime[ARMCPU_ARM9][0] == 0)
MMU_doDMA<ARMCPU_ARM9>(0);
#ifdef LOG_DMA2
//else
{
LOG("ARMCPU_ARM9 %d, dma %d src %08X dst %08X %s\r\n", ARMCPU_ARM9, 0, DMASrc[ARMCPU_ARM9][0], DMADst[ARMCPU_ARM9][0], (val&(1<<25))?"ON":"OFF");
}
#endif
}
return; return;
case REG_DMA1CNTH : case REG_DMA1CNTH :
{ write_dma_hictrl<ARMCPU_ARM9>(1,val);
u32 v;
//if(val&0x8000) emu_halt();
//LOG("16 bit dma1 %04X\r\n", val);
T1WriteWord(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0xC6, val);
DMASrc[ARMCPU_ARM9][1] = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0xBC);
DMADst[ARMCPU_ARM9][1] = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0xC0);
DMAtoVRAMmapping<1>();
v = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0xC4);
MMU.DMAStartTime[ARMCPU_ARM9][1] = (v>>27) & 0x7;
MMU.DMACrt[ARMCPU_ARM9][1] = v;
if(MMU.DMAStartTime[ARMCPU_ARM9][1] == 0)
MMU_doDMA<ARMCPU_ARM9>(1);
#ifdef LOG_DMA2
//else
{
LOG("ARMCPU_ARM9 %d, dma %d src %08X dst %08X %s\r\n", ARMCPU_ARM9, 1, DMASrc[ARMCPU_ARM9][1], DMADst[ARMCPU_ARM9][1], (val&(1<<25))?"ON":"OFF");
}
#endif
}
return; return;
case REG_DMA2CNTH : case REG_DMA2CNTH :
{ write_dma_hictrl<ARMCPU_ARM9>(2,val);
u32 v;
//if(val&0x8000) emu_halt();
//LOG("16 bit dma2 %04X\r\n", val);
T1WriteWord(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0xD2, val);
DMASrc[ARMCPU_ARM9][2] = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0xC8);
DMADst[ARMCPU_ARM9][2] = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0xCC);
DMAtoVRAMmapping<2>();
v = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0xD0);
MMU.DMAStartTime[ARMCPU_ARM9][2] = (v>>27) & 0x7;
MMU.DMACrt[ARMCPU_ARM9][2] = v;
if(MMU.DMAStartTime[ARMCPU_ARM9][2] == 0)
MMU_doDMA<ARMCPU_ARM9>(2);
#ifdef LOG_DMA2
//else
{
LOG("ARMCPU_ARM9 %d, dma %d src %08X dst %08X %s\r\n", ARMCPU_ARM9, 2, DMASrc[ARMCPU_ARM9][2], DMADst[ARMCPU_ARM9][2], (val&(1<<25))?"ON":"OFF");
}
#endif
}
return; return;
case REG_DMA3CNTH : case REG_DMA3CNTH :
{ write_dma_hictrl<ARMCPU_ARM9>(3,val);
u32 v;
//if(val&0x8000) emu_halt();
//LOG("16 bit dma3 %04X\r\n", val);
T1WriteWord(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0xDE, val);
DMASrc[ARMCPU_ARM9][3] = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0xD4);
DMADst[ARMCPU_ARM9][3] = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0xD8);
DMAtoVRAMmapping<3>();
v = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0xDC);
MMU.DMAStartTime[ARMCPU_ARM9][3] = (v>>27) & 0x7;
MMU.DMACrt[ARMCPU_ARM9][3] = v;
if(MMU.DMAStartTime[ARMCPU_ARM9][3] == 0)
MMU_doDMA<ARMCPU_ARM9>(3);
#ifdef LOG_DMA2
//else
{
LOG("ARMCPU_ARM9 %d, dma %d src %08X dst %08X %s\r\n", ARMCPU_ARM9, 3, DMASrc[ARMCPU_ARM9][3], DMADst[ARMCPU_ARM9][3], (val&(1<<25))?"ON":"OFF");
}
#endif
}
return; return;
//case REG_AUXSPICNT : emu_halt();
case REG_DISPA_DISPMMEMFIFO: case REG_DISPA_DISPMMEMFIFO:
{ {
DISP_FIFOsend(val); DISP_FIFOsend(val);
@ -2382,6 +2359,7 @@ void FASTCALL _MMU_ARM9_write32(u32 adr, u32 val)
case REG_IME : case REG_IME :
{ {
NDS_Reschedule();
u32 old_val = MMU.reg_IME[ARMCPU_ARM9]; u32 old_val = MMU.reg_IME[ARMCPU_ARM9];
u32 new_val = val & 0x01; u32 new_val = val & 0x01;
MMU.reg_IME[ARMCPU_ARM9] = new_val; MMU.reg_IME[ARMCPU_ARM9] = new_val;
@ -2401,6 +2379,7 @@ void FASTCALL _MMU_ARM9_write32(u32 adr, u32 val)
return; return;
case REG_IE : case REG_IE :
NDS_Reschedule();
MMU.reg_IE[ARMCPU_ARM9] = val; MMU.reg_IE[ARMCPU_ARM9] = val;
#ifndef NEW_IRQ #ifndef NEW_IRQ
if ( MMU.reg_IME[ARMCPU_ARM9]) if ( MMU.reg_IME[ARMCPU_ARM9])
@ -2416,6 +2395,7 @@ void FASTCALL _MMU_ARM9_write32(u32 adr, u32 val)
return; return;
case REG_IF : case REG_IF :
NDS_Reschedule();
MMU.reg_IF[ARMCPU_ARM9] &= (~val); MMU.reg_IF[ARMCPU_ARM9] &= (~val);
return; return;
@ -2425,36 +2405,9 @@ void FASTCALL _MMU_ARM9_write32(u32 adr, u32 val)
case REG_TM3CNTL: case REG_TM3CNTL:
{ {
int timerIndex = (adr>>2)&0x3; int timerIndex = (adr>>2)&0x3;
int mask = ((val & 0x800000)>>(16+7)) << timerIndex;
MMU.CheckTimers = (MMU.CheckTimers & (~mask)) | mask;
MMU.timerReload[ARMCPU_ARM9][timerIndex] = (u16)val; MMU.timerReload[ARMCPU_ARM9][timerIndex] = (u16)val;
if(val&0x800000) T1WriteWord(MMU.MMU_MEM[ARMCPU_ARM9][0x40], adr & 0xFFF, val);
MMU.timer[ARMCPU_ARM9][timerIndex] = MMU.timerReload[ARMCPU_ARM9][(adr>>2)&0x3]; write_timer(ARMCPU_ARM9, timerIndex, val>>16);
MMU.timerON[ARMCPU_ARM9][timerIndex] = val & 0x800000;
switch((val>>16)&7)
{
case 0 :
MMU.timerMODE[ARMCPU_ARM9][timerIndex] = 0+1;//ARMCPU_ARM9;
break;
case 1 :
MMU.timerMODE[ARMCPU_ARM9][timerIndex] = 6+1;//ARMCPU_ARM9;
break;
case 2 :
MMU.timerMODE[ARMCPU_ARM9][timerIndex] = 8+1;//ARMCPU_ARM9;
break;
case 3 :
MMU.timerMODE[ARMCPU_ARM9][timerIndex] = 10+1;//ARMCPU_ARM9;
break;
default :
MMU.timerMODE[ARMCPU_ARM9][timerIndex] = 0xFFFF;
break;
}
if(!(val & 0x800000))
MMU.timerRUN[ARMCPU_ARM9][timerIndex] = FALSE;
T1WriteLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], adr & 0xFFF, val);
return; return;
} }
@ -2492,77 +2445,20 @@ void FASTCALL _MMU_ARM9_write32(u32 adr, u32 val)
return; return;
case REG_DMA0CNTL : case REG_DMA0CNTL :
//LOG("32 bit dma0 %04X\r\n", val); T1WriteWord(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0xB8, val); //write the low word
DMASrc[ARMCPU_ARM9][0] = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0xB0); write_dma_hictrl<ARMCPU_ARM9>(0,val>>16);
DMADst[ARMCPU_ARM9][0] = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0xB4);
DMAtoVRAMmapping<0>();
MMU.DMAStartTime[ARMCPU_ARM9][0] = (val>>27) & 0x7;
MMU.DMACrt[ARMCPU_ARM9][0] = val;
T1WriteLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0xB8, val);
if( MMU.DMAStartTime[ARMCPU_ARM9][0] == 0 ||
MMU.DMAStartTime[ARMCPU_ARM9][0] == 7) // Start Immediately
MMU_doDMA<ARMCPU_ARM9>(0);
#ifdef LOG_DMA2
else
{
LOG("ARMCPU_ARM9 %d, dma %d src %08X dst %08X start taille %d %d\r\n", ARMCPU_ARM9, 0, DMASrc[ARMCPU_ARM9][0], DMADst[ARMCPU_ARM9][0], 0, ((MMU.DMACrt[ARMCPU_ARM9][0]>>27)&7));
}
#endif
//emu_halt();
return; return;
case REG_DMA1CNTL: case REG_DMA1CNTL:
//LOG("32 bit dma1 %04X\r\n", val); T1WriteWord(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0xC4, val); //write the low word
DMASrc[ARMCPU_ARM9][1] = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0xBC); write_dma_hictrl<ARMCPU_ARM9>(1,val>>16);
DMADst[ARMCPU_ARM9][1] = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0xC0);
DMAtoVRAMmapping<1>();
MMU.DMAStartTime[ARMCPU_ARM9][1] = (val>>27) & 0x7;
MMU.DMACrt[ARMCPU_ARM9][1] = val;
T1WriteLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0xC4, val);
if(MMU.DMAStartTime[ARMCPU_ARM9][1] == 0 ||
MMU.DMAStartTime[ARMCPU_ARM9][1] == 7) // Start Immediately
MMU_doDMA<ARMCPU_ARM9>(1);
#ifdef LOG_DMA2
else
{
LOG("ARMCPU_ARM9 %d, dma %d src %08X dst %08X start taille %d %d\r\n", ARMCPU_ARM9, 1, DMASrc[ARMCPU_ARM9][1], DMADst[ARMCPU_ARM9][1], 0, ((MMU.DMACrt[ARMCPU_ARM9][1]>>27)&7));
}
#endif
return; return;
case REG_DMA2CNTL : case REG_DMA2CNTL :
//LOG("32 bit dma2 %04X\r\n", val); T1WriteWord(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0xD0, val); //write the low word
DMASrc[ARMCPU_ARM9][2] = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0xC8); write_dma_hictrl<ARMCPU_ARM9>(2,val>>16);
DMADst[ARMCPU_ARM9][2] = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0xCC);
DMAtoVRAMmapping<2>();
MMU.DMAStartTime[ARMCPU_ARM9][2] = (val>>27) & 0x7;
MMU.DMACrt[ARMCPU_ARM9][2] = val;
T1WriteLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0xD0, val);
if(MMU.DMAStartTime[ARMCPU_ARM9][2] == 0 ||
MMU.DMAStartTime[ARMCPU_ARM9][2] == 7) // Start Immediately
MMU_doDMA<ARMCPU_ARM9>(2);
#ifdef LOG_DMA2
else
{
LOG("ARMCPU_ARM9 %d, dma %d src %08X dst %08X start taille %d %d\r\n", ARMCPU_ARM9, 2, DMASrc[ARMCPU_ARM9][2], DMADst[ARMCPU_ARM9][2], 0, ((MMU.DMACrt[ARMCPU_ARM9][2]>>27)&7));
}
#endif
return; return;
case REG_DMA3CNTL : case REG_DMA3CNTL :
//LOG("32 bit dma3 %04X\r\n", val); T1WriteWord(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0xDC, val); //write the low word
DMASrc[ARMCPU_ARM9][3] = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0xD4); write_dma_hictrl<ARMCPU_ARM9>(3,val>>16);
DMADst[ARMCPU_ARM9][3] = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0xD8);
DMAtoVRAMmapping<3>();
MMU.DMAStartTime[ARMCPU_ARM9][3] = (val>>27) & 0x7;
MMU.DMACrt[ARMCPU_ARM9][3] = val;
T1WriteLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0xDC, val);
if( MMU.DMAStartTime[ARMCPU_ARM9][3] == 0 ||
MMU.DMAStartTime[ARMCPU_ARM9][3] == 7) // Start Immediately
MMU_doDMA<ARMCPU_ARM9>(3);
#ifdef LOG_DMA2
else
{
LOG("ARMCPU_ARM9 %d, dma %d src %08X dst %08X start taille %d %d\r\n", ARMCPU_ARM9, 3, DMASrc[ARMCPU_ARM9][3], DMADst[ARMCPU_ARM9][3], 0, ((MMU.DMACrt[ARMCPU_ARM9][3]>>27)&7));
}
#endif
return; return;
case REG_GCROMCTRL : case REG_GCROMCTRL :
{ {
@ -2640,23 +2536,11 @@ void FASTCALL _MMU_ARM9_write32(u32 adr, u32 val)
val |= 0x00800000; val |= 0x00800000;
T1WriteLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0x1A4, val); T1WriteLong(MMU.MMU_MEM[ARMCPU_ARM9][0x40], 0x1A4, val);
/* launch DMA if start flag was set to "DS Cart" */ //launch DMA if start flag was set to "DS Cart"
if(MMU.DMAStartTime[ARMCPU_ARM9][0] == 5) if(MMU.DMAStartTime[ARMCPU_ARM9][0] == EDMAMode_Card) MMU_doDMA<ARMCPU_ARM9>(0);
{ if(MMU.DMAStartTime[ARMCPU_ARM9][1] == EDMAMode_Card) MMU_doDMA<ARMCPU_ARM9>(1);
MMU_doDMA<ARMCPU_ARM9>(0); if(MMU.DMAStartTime[ARMCPU_ARM9][2] == EDMAMode_Card) MMU_doDMA<ARMCPU_ARM9>(2);
} if(MMU.DMAStartTime[ARMCPU_ARM9][3] == EDMAMode_Card) MMU_doDMA<ARMCPU_ARM9>(3);
if(MMU.DMAStartTime[ARMCPU_ARM9][1] == 5)
{
MMU_doDMA<ARMCPU_ARM9>(1);
}
if(MMU.DMAStartTime[ARMCPU_ARM9][2] == 5)
{
MMU_doDMA<ARMCPU_ARM9>(2);
}
if(MMU.DMAStartTime[ARMCPU_ARM9][3] == 5)
{
MMU_doDMA<ARMCPU_ARM9>(3);
}
} }
return; return;
case REG_DISPA_DISPCAPCNT : case REG_DISPA_DISPCAPCNT :
@ -2768,7 +2652,7 @@ u16 FASTCALL _MMU_ARM9_read16(u32 adr)
case REG_TM1CNTL : case REG_TM1CNTL :
case REG_TM2CNTL : case REG_TM2CNTL :
case REG_TM3CNTL : case REG_TM3CNTL :
return MMU.timer[ARMCPU_ARM9][(adr&0xF)>>2]; return read_timer(ARMCPU_ARM9,(adr&0xF)>>2);
case REG_AUXSPICNT: case REG_AUXSPICNT:
return MMU.AUX_SPI_CNT; return MMU.AUX_SPI_CNT;
@ -3246,6 +3130,7 @@ void FASTCALL _MMU_ARM7_write16(u32 adr, u16 val)
case REG_IME : case REG_IME :
{ {
NDS_Reschedule();
u32 old_val = MMU.reg_IME[ARMCPU_ARM7]; u32 old_val = MMU.reg_IME[ARMCPU_ARM7];
u32 new_val = val & 1; u32 new_val = val & 1;
MMU.reg_IME[ARMCPU_ARM7] = new_val; MMU.reg_IME[ARMCPU_ARM7] = new_val;
@ -3264,6 +3149,7 @@ void FASTCALL _MMU_ARM7_write16(u32 adr, u16 val)
return; return;
} }
case REG_IE : case REG_IE :
NDS_Reschedule();
MMU.reg_IE[ARMCPU_ARM7] = (MMU.reg_IE[ARMCPU_ARM7]&0xFFFF0000) | val; MMU.reg_IE[ARMCPU_ARM7] = (MMU.reg_IE[ARMCPU_ARM7]&0xFFFF0000) | val;
#ifndef NEW_IRQ #ifndef NEW_IRQ
if ( MMU.reg_IME[ARMCPU_ARM7]) if ( MMU.reg_IME[ARMCPU_ARM7])
@ -3278,6 +3164,7 @@ void FASTCALL _MMU_ARM7_write16(u32 adr, u16 val)
#endif #endif
return; return;
case REG_IE + 2 : case REG_IE + 2 :
NDS_Reschedule();
//emu_halt(); //emu_halt();
MMU.reg_IE[ARMCPU_ARM7] = (MMU.reg_IE[ARMCPU_ARM7]&0xFFFF) | (((u32)val)<<16); MMU.reg_IE[ARMCPU_ARM7] = (MMU.reg_IE[ARMCPU_ARM7]&0xFFFF) | (((u32)val)<<16);
#ifndef NEW_IRQ #ifndef NEW_IRQ
@ -3294,10 +3181,12 @@ void FASTCALL _MMU_ARM7_write16(u32 adr, u16 val)
return; return;
case REG_IF : case REG_IF :
NDS_Reschedule();
//emu_halt(); //emu_halt();
MMU.reg_IF[ARMCPU_ARM7] &= (~((u32)val)); MMU.reg_IF[ARMCPU_ARM7] &= (~((u32)val));
return; return;
case REG_IF + 2 : case REG_IF + 2 :
NDS_Reschedule();
//emu_halt(); //emu_halt();
MMU.reg_IF[ARMCPU_ARM7] &= (~(((u32)val)<<16)); MMU.reg_IF[ARMCPU_ARM7] &= (~(((u32)val)<<16));
return; return;
@ -3321,127 +3210,22 @@ void FASTCALL _MMU_ARM7_write16(u32 adr, u16 val)
case REG_TM3CNTH : case REG_TM3CNTH :
{ {
int timerIndex = ((adr-2)>>2)&0x3; int timerIndex = ((adr-2)>>2)&0x3;
int mask = ((val&0x80)>>7) << (timerIndex+(ARMCPU_ARM7<<2)); write_timer(ARMCPU_ARM7, timerIndex, val);
MMU.CheckTimers = (MMU.CheckTimers & (~mask)) | mask;
if(val&0x80)
MMU.timer[ARMCPU_ARM7][timerIndex] = MMU.timerReload[ARMCPU_ARM7][((adr-2)>>2)&0x3];
MMU.timerON[ARMCPU_ARM7][((adr-2)>>2)&0x3] = val & 0x80;
switch(val&7)
{
case 0 :
MMU.timerMODE[ARMCPU_ARM7][timerIndex] = 0+1;//ARMCPU_ARM7;
break;
case 1 :
MMU.timerMODE[ARMCPU_ARM7][timerIndex] = 6+1;//ARMCPU_ARM7;
break;
case 2 :
MMU.timerMODE[ARMCPU_ARM7][timerIndex] = 8+1;//ARMCPU_ARM7;
break;
case 3 :
MMU.timerMODE[ARMCPU_ARM7][timerIndex] = 10+1;//ARMCPU_ARM7;
break;
default :
MMU.timerMODE[ARMCPU_ARM7][timerIndex] = 0xFFFF;
break;
}
if(!(val & 0x80))
MMU.timerRUN[ARMCPU_ARM7][timerIndex] = FALSE;
T1WriteWord(MMU.MMU_MEM[ARMCPU_ARM7][0x40], adr & 0xFFF, val);
return; return;
} }
case REG_DMA0CNTH : case REG_DMA0CNTH :
{ write_dma_hictrl<ARMCPU_ARM7>(0,val);
u32 v;
//if(val&0x8000) emu_halt();
//LOG("16 bit dma0 %04X\r\n", val);
T1WriteWord(MMU.MMU_MEM[ARMCPU_ARM7][0x40], 0xBA, val);
DMASrc[ARMCPU_ARM7][0] = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM7][0x40], 0xB0);
DMADst[ARMCPU_ARM7][0] = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM7][0x40], 0xB4);
v = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM7][0x40], 0xB8);
MMU.DMAStartTime[ARMCPU_ARM7][0] = (v>>28) & 0x3;
MMU.DMACrt[ARMCPU_ARM7][0] = v;
if(MMU.DMAStartTime[ARMCPU_ARM7][0] == 0)
MMU_doDMA<ARMCPU_ARM7>(0);
#ifdef LOG_DMA2
//else
{
LOG("ARMCPU_ARM7 %d, dma %d src %08X dst %08X %s\r\n", ARMCPU_ARM7, 0, DMASrc[ARMCPU_ARM7][0], DMADst[ARMCPU_ARM7][0], (val&(1<<25))?"ON":"OFF");
}
#endif
}
return; return;
case REG_DMA1CNTH : case REG_DMA1CNTH :
{ write_dma_hictrl<ARMCPU_ARM7>(1,val);
u32 v;
//if(val&0x8000) emu_halt();
//LOG("16 bit dma1 %04X\r\n", val);
T1WriteWord(MMU.MMU_MEM[ARMCPU_ARM7][0x40], 0xC6, val);
DMASrc[ARMCPU_ARM7][1] = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM7][0x40], 0xBC);
DMADst[ARMCPU_ARM7][1] = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM7][0x40], 0xC0);
v = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM7][0x40], 0xC4);
MMU.DMAStartTime[ARMCPU_ARM7][1] = (v>>28) & 0x3;
MMU.DMACrt[ARMCPU_ARM7][1] = v;
if(MMU.DMAStartTime[ARMCPU_ARM7][1] == 0)
MMU_doDMA<ARMCPU_ARM7>(1);
#ifdef LOG_DMA2
//else
{
LOG("ARMCPU_ARM7 %d, dma %d src %08X dst %08X %s\r\n", ARMCPU_ARM7, 1, DMASrc[ARMCPU_ARM7][1], DMADst[ARMCPU_ARM7][1], (val&(1<<25))?"ON":"OFF");
}
#endif
}
return; return;
case REG_DMA2CNTH : case REG_DMA2CNTH :
{ write_dma_hictrl<ARMCPU_ARM7>(2,val);
u32 v;
//if(val&0x8000) emu_halt();
//LOG("16 bit dma2 %04X\r\n", val);
T1WriteWord(MMU.MMU_MEM[ARMCPU_ARM7][0x40], 0xD2, val);
DMASrc[ARMCPU_ARM7][2] = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM7][0x40], 0xC8);
DMADst[ARMCPU_ARM7][2] = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM7][0x40], 0xCC);
v = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM7][0x40], 0xD0);
MMU.DMAStartTime[ARMCPU_ARM7][2] = (v>>28) & 0x3;
MMU.DMACrt[ARMCPU_ARM7][2] = v;
if(MMU.DMAStartTime[ARMCPU_ARM7][2] == 0)
MMU_doDMA<ARMCPU_ARM7>(2);
#ifdef LOG_DMA2
//else
{
LOG("ARMCPU_ARM7 %d, dma %d src %08X dst %08X %s\r\n", ARMCPU_ARM7, 2, DMASrc[ARMCPU_ARM7][2], DMADst[ARMCPU_ARM7][2], (val&(1<<25))?"ON":"OFF");
}
#endif
}
return; return;
case REG_DMA3CNTH : case REG_DMA3CNTH :
{ write_dma_hictrl<ARMCPU_ARM7>(3,val);
u32 v;
//if(val&0x8000) emu_halt();
//LOG("16 bit dma3 %04X\r\n", val);
T1WriteWord(MMU.MMU_MEM[ARMCPU_ARM7][0x40], 0xDE, val);
DMASrc[ARMCPU_ARM7][3] = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM7][0x40], 0xD4);
DMADst[ARMCPU_ARM7][3] = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM7][0x40], 0xD8);
v = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM7][0x40], 0xDC);
MMU.DMAStartTime[ARMCPU_ARM7][3] = (v>>28) & 0x3;
MMU.DMACrt[ARMCPU_ARM7][3] = v;
if(MMU.DMAStartTime[ARMCPU_ARM7][3] == 0)
MMU_doDMA<ARMCPU_ARM7>(3);
#ifdef LOG_DMA2
//else
{
LOG("ARMCPU_ARM7 %d, dma %d src %08X dst %08X %s\r\n", ARMCPU_ARM7, 3, DMASrc[ARMCPU_ARM7][3], DMADst[ARMCPU_ARM7][3], (val&(1<<25))?"ON":"OFF");
}
#endif
}
return; return;
//case REG_AUXSPICNT : emu_halt();
} }
T1WriteWord(MMU.MMU_MEM[ARMCPU_ARM7][0x40], adr&MMU.MMU_MASK[ARMCPU_ARM7][adr>>20], val); T1WriteWord(MMU.MMU_MEM[ARMCPU_ARM7][0x40], adr&MMU.MMU_MASK[ARMCPU_ARM7][adr>>20], val);
@ -3497,6 +3281,7 @@ void FASTCALL _MMU_ARM7_write32(u32 adr, u32 val)
case REG_IME : case REG_IME :
{ {
NDS_Reschedule();
u32 old_val = MMU.reg_IME[ARMCPU_ARM7]; u32 old_val = MMU.reg_IME[ARMCPU_ARM7];
u32 new_val = val & 1; u32 new_val = val & 1;
MMU.reg_IME[ARMCPU_ARM7] = new_val; MMU.reg_IME[ARMCPU_ARM7] = new_val;
@ -3516,6 +3301,7 @@ void FASTCALL _MMU_ARM7_write32(u32 adr, u32 val)
} }
case REG_IE : case REG_IE :
NDS_Reschedule();
MMU.reg_IE[ARMCPU_ARM7] = val; MMU.reg_IE[ARMCPU_ARM7] = val;
#ifndef NEW_IRQ #ifndef NEW_IRQ
if ( MMU.reg_IME[ARMCPU_ARM7]) if ( MMU.reg_IME[ARMCPU_ARM7])
@ -3531,6 +3317,7 @@ void FASTCALL _MMU_ARM7_write32(u32 adr, u32 val)
return; return;
case REG_IF : case REG_IF :
NDS_Reschedule();
MMU.reg_IF[ARMCPU_ARM7] &= (~val); MMU.reg_IF[ARMCPU_ARM7] &= (~val);
return; return;
@ -3540,38 +3327,13 @@ void FASTCALL _MMU_ARM7_write32(u32 adr, u32 val)
case REG_TM3CNTL: case REG_TM3CNTL:
{ {
int timerIndex = (adr>>2)&0x3; int timerIndex = (adr>>2)&0x3;
int mask = ((val & 0x800000)>>(16+7)) << (timerIndex+(ARMCPU_ARM7<<2));
MMU.CheckTimers = (MMU.CheckTimers & (~mask)) | mask;
MMU.timerReload[ARMCPU_ARM7][timerIndex] = (u16)val; MMU.timerReload[ARMCPU_ARM7][timerIndex] = (u16)val;
if(val&0x800000) T1WriteWord(MMU.MMU_MEM[ARMCPU_ARM7][0x40], adr & 0xFFF, val);
MMU.timer[ARMCPU_ARM7][timerIndex] = MMU.timerReload[ARMCPU_ARM7][(adr>>2)&0x3]; write_timer(ARMCPU_ARM7, timerIndex, val>>16);
MMU.timerON[ARMCPU_ARM7][timerIndex] = val & 0x800000;
switch((val>>16)&7)
{
case 0 :
MMU.timerMODE[ARMCPU_ARM7][timerIndex] = 0+1;//ARMCPU_ARM7;
break;
case 1 :
MMU.timerMODE[ARMCPU_ARM7][timerIndex] = 6+1;//ARMCPU_ARM7;
break;
case 2 :
MMU.timerMODE[ARMCPU_ARM7][timerIndex] = 8+1;//ARMCPU_ARM7;
break;
case 3 :
MMU.timerMODE[ARMCPU_ARM7][timerIndex] = 10+1;//ARMCPU_ARM7;
break;
default :
MMU.timerMODE[ARMCPU_ARM7][timerIndex] = 0xFFFF;
break;
}
if(!(val & 0x800000))
MMU.timerRUN[ARMCPU_ARM7][timerIndex] = FALSE;
T1WriteLong(MMU.MMU_MEM[ARMCPU_ARM7][0x40], adr & 0xFFF, val);
return; return;
} }
case REG_IPCSYNC : case REG_IPCSYNC :
MMU_IPCSync(ARMCPU_ARM7, val); MMU_IPCSync(ARMCPU_ARM7, val);
return; return;
@ -3579,75 +3341,24 @@ void FASTCALL _MMU_ARM7_write32(u32 adr, u32 val)
case REG_IPCFIFOSEND : case REG_IPCFIFOSEND :
IPC_FIFOsend(ARMCPU_ARM7, val); IPC_FIFOsend(ARMCPU_ARM7, val);
return; return;
case REG_DMA0CNTL : case REG_DMA0CNTL :
//LOG("32 bit dma0 %04X\r\n", val); T1WriteWord(MMU.MMU_MEM[ARMCPU_ARM7][0x40], 0xB8, val); //write the low word
DMASrc[ARMCPU_ARM7][0] = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM7][0x40], 0xB0); write_dma_hictrl<ARMCPU_ARM7>(0,val>>16);
DMADst[ARMCPU_ARM7][0] = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM7][0x40], 0xB4);
MMU.DMAStartTime[ARMCPU_ARM7][0] = (val>>28) & 0x3;
MMU.DMACrt[ARMCPU_ARM7][0] = val;
T1WriteLong(MMU.MMU_MEM[ARMCPU_ARM7][0x40], 0xB8, val);
if( MMU.DMAStartTime[ARMCPU_ARM7][0] == 0 ||
MMU.DMAStartTime[ARMCPU_ARM7][0] == 7) // Start Immediately
MMU_doDMA<ARMCPU_ARM7>(0);
#ifdef LOG_DMA2
else
{
LOG("ARMCPU_ARM7 %d, dma %d src %08X dst %08X start taille %d %d\r\n", ARMCPU_ARM7, 0, DMASrc[ARMCPU_ARM7][0], DMADst[ARMCPU_ARM7][0], 0, ((MMU.DMACrt[ARMCPU_ARM7][0]>>27)&7));
}
#endif
//emu_halt();
return; return;
case REG_DMA1CNTL: case REG_DMA1CNTL:
//LOG("32 bit dma1 %04X\r\n", val); T1WriteWord(MMU.MMU_MEM[ARMCPU_ARM7][0x40], 0xC4, val); //write the low word
DMASrc[ARMCPU_ARM7][1] = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM7][0x40], 0xBC); write_dma_hictrl<ARMCPU_ARM7>(1,val>>16);
DMADst[ARMCPU_ARM7][1] = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM7][0x40], 0xC0);
MMU.DMAStartTime[ARMCPU_ARM7][1] = (val>>28) & 0x3;
MMU.DMACrt[ARMCPU_ARM7][1] = val;
T1WriteLong(MMU.MMU_MEM[ARMCPU_ARM7][0x40], 0xC4, val);
if(MMU.DMAStartTime[ARMCPU_ARM7][1] == 0 ||
MMU.DMAStartTime[ARMCPU_ARM7][1] == 7) // Start Immediately
MMU_doDMA<ARMCPU_ARM7>(1);
#ifdef LOG_DMA2
else
{
LOG("ARMCPU_ARM7 %d, dma %d src %08X dst %08X start taille %d %d\r\n", ARMCPU_ARM7, 1, DMASrc[ARMCPU_ARM7][1], DMADst[ARMCPU_ARM7][1], 0, ((MMU.DMACrt[ARMCPU_ARM7][1]>>27)&7));
}
#endif
return; return;
case REG_DMA2CNTL : case REG_DMA2CNTL :
//LOG("32 bit dma2 %04X\r\n", val); T1WriteWord(MMU.MMU_MEM[ARMCPU_ARM7][0x40], 0xD0, val); //write the low word
DMASrc[ARMCPU_ARM7][2] = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM7][0x40], 0xC8); write_dma_hictrl<ARMCPU_ARM7>(2,val>>16);
DMADst[ARMCPU_ARM7][2] = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM7][0x40], 0xCC);
MMU.DMAStartTime[ARMCPU_ARM7][2] = (val>>28) & 0x3;
MMU.DMACrt[ARMCPU_ARM7][2] = val;
T1WriteLong(MMU.MMU_MEM[ARMCPU_ARM7][0x40], 0xD0, val);
if(MMU.DMAStartTime[ARMCPU_ARM7][2] == 0 ||
MMU.DMAStartTime[ARMCPU_ARM7][2] == 7) // Start Immediately
MMU_doDMA<ARMCPU_ARM7>(2);
#ifdef LOG_DMA2
else
{
LOG("ARMCPU_ARM7 %d, dma %d src %08X dst %08X start taille %d %d\r\n", ARMCPU_ARM7, 2, DMASrc[ARMCPU_ARM7][2], DMADst[ARMCPU_ARM7][2], 0, ((MMU.DMACrt[ARMCPU_ARM7][2]>>27)&7));
}
#endif
return; return;
case REG_DMA3CNTL : case REG_DMA3CNTL :
//LOG("32 bit dma3 %04X\r\n", val); T1WriteWord(MMU.MMU_MEM[ARMCPU_ARM7][0x40], 0xDC, val); //write the low word
DMASrc[ARMCPU_ARM7][3] = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM7][0x40], 0xD4); write_dma_hictrl<ARMCPU_ARM7>(3,val>>16);
DMADst[ARMCPU_ARM7][3] = T1ReadLong(MMU.MMU_MEM[ARMCPU_ARM7][0x40], 0xD8);
MMU.DMAStartTime[ARMCPU_ARM7][3] = (val>>28) & 0x3;
MMU.DMACrt[ARMCPU_ARM7][3] = val;
T1WriteLong(MMU.MMU_MEM[ARMCPU_ARM7][0x40], 0xDC, val);
if( MMU.DMAStartTime[ARMCPU_ARM7][3] == 0 ||
MMU.DMAStartTime[ARMCPU_ARM7][3] == 7) // Start Immediately
MMU_doDMA<ARMCPU_ARM7>(3);
#ifdef LOG_DMA2
else
{
LOG("ARMCPU_ARM7 %d, dma %d src %08X dst %08X start taille %d %d\r\n", ARMCPU_ARM7, 3, DMASrc[ARMCPU_ARM7][3], DMADst[ARMCPU_ARM7][3], 0, ((MMU.DMACrt[ARMCPU_ARM7][3]>>27)&7));
}
#endif
return; return;
case REG_GCROMCTRL : case REG_GCROMCTRL :
{ {
if(!(val & 0x80000000)) if(!(val & 0x80000000))
@ -3707,16 +3418,11 @@ void FASTCALL _MMU_ARM7_write32(u32 adr, u32 val)
val |= 0x00800000; val |= 0x00800000;
T1WriteLong(MMU.MMU_MEM[ARMCPU_ARM7][0x40], 0x1A4, val); T1WriteLong(MMU.MMU_MEM[ARMCPU_ARM7][0x40], 0x1A4, val);
/* launch DMA if start flag was set to "DS Cart" */ //launch DMA if start flag was set to "DS Cart"
if(MMU.DMAStartTime[ARMCPU_ARM7][0] == EDMAMode_Card) MMU_doDMA<ARMCPU_ARM7>(0);
if(MMU.DMAStartTime[ARMCPU_ARM7][2] == 2) if(MMU.DMAStartTime[ARMCPU_ARM7][1] == EDMAMode_Card) MMU_doDMA<ARMCPU_ARM7>(1);
{ if(MMU.DMAStartTime[ARMCPU_ARM7][2] == EDMAMode_Card) MMU_doDMA<ARMCPU_ARM7>(2);
MMU_doDMA<ARMCPU_ARM7>(2); if(MMU.DMAStartTime[ARMCPU_ARM7][3] == EDMAMode_Card) MMU_doDMA<ARMCPU_ARM7>(3);
}
if(MMU.DMAStartTime[ARMCPU_ARM7][3] == 2)
{
MMU_doDMA<ARMCPU_ARM7>(3);
}
return; return;
@ -3803,7 +3509,7 @@ u16 FASTCALL _MMU_ARM7_read16(u32 adr)
case REG_TM1CNTL : case REG_TM1CNTL :
case REG_TM2CNTL : case REG_TM2CNTL :
case REG_TM3CNTL : case REG_TM3CNTL :
return MMU.timer[ARMCPU_ARM7][(adr&0xF)>>2]; return read_timer(ARMCPU_ARM7,(adr&0xF)>>2);
case REG_AUXSPICNT: case REG_AUXSPICNT:
return MMU.AUX_SPI_CNT; return MMU.AUX_SPI_CNT;

View File

@ -97,7 +97,7 @@ struct MMU_struct {
u32 reg_IF[2]; u32 reg_IF[2];
u32 DMAStartTime[2][4]; u32 DMAStartTime[2][4];
s32 DMACycle[2][4]; u64 DMACycle[2][4];
u32 DMACrt[2][4]; u32 DMACrt[2][4];
BOOL DMAing[2][4]; BOOL DMAing[2][4];
@ -105,21 +105,19 @@ struct MMU_struct {
s64 divResult; s64 divResult;
s64 divMod; s64 divMod;
u32 divCnt; u32 divCnt;
s32 divCycles; u64 divCycles;
BOOL sqrtRunning; BOOL sqrtRunning;
u32 sqrtResult; u32 sqrtResult;
u32 sqrtCnt; u32 sqrtCnt;
s32 sqrtCycles; u64 sqrtCycles;
u16 SPI_CNT; u16 SPI_CNT;
u16 SPI_CMD; u16 SPI_CMD;
u16 AUX_SPI_CNT; u16 AUX_SPI_CNT;
u16 AUX_SPI_CMD; u16 AUX_SPI_CMD;
#ifdef USE_GEOMETRY_FIFO_EMULATION u64 gfx3dCycles;
s32 gfx3dCycles;
#endif
u8 powerMan_CntReg; u8 powerMan_CntReg;
BOOL powerMan_CntRegWritten; BOOL powerMan_CntRegWritten;
@ -265,6 +263,20 @@ inline void SetupMMU(bool debugConsole) {
//T1ReadWord(MMU.MMU_MEM[ARMCPU_ARM7][(adr >> 20) & 0xFF], //T1ReadWord(MMU.MMU_MEM[ARMCPU_ARM7][(adr >> 20) & 0xFF],
// adr & MMU.MMU_MASK[ARMCPU_ARM7][(adr >> 20) & 0xFF]); // adr & MMU.MMU_MASK[ARMCPU_ARM7][(adr >> 20) & 0xFF]);
enum EDMAMode
{
EDMAMode_Immediate = 0,
EDMAMode_VBlank = 1,
EDMAMode_HBlank = 2,
EDMAMode_HStart = 3,
EDMAMode_MemDisplay = 4,
EDMAMode_Card = 5,
EDMAMode_GBASlot = 6,
EDMAMode_GXFifo = 7,
EDMAMode7_Wifi = 8,
EDMAMode7_GBASlot = 9,
};
FORCEINLINE u8 _MMU_read08(const int PROCNUM, const MMU_ACCESS_TYPE AT, const u32 addr) FORCEINLINE u8 _MMU_read08(const int PROCNUM, const MMU_ACCESS_TYPE AT, const u32 addr)
{ {
//special handling for DMA: read 0 from TCM //special handling for DMA: read 0 from TCM
@ -427,6 +439,10 @@ FORCEINLINE void _MMU_write16(const int PROCNUM, const MMU_ACCESS_TYPE AT, const
FORCEINLINE void _MMU_write32(const int PROCNUM, const MMU_ACCESS_TYPE AT, const u32 addr, u32 val) FORCEINLINE void _MMU_write32(const int PROCNUM, const MMU_ACCESS_TYPE AT, const u32 addr, u32 val)
{ {
if(addr == 0x022D7900)
{
int zzz=9;
}
//special handling for DMA: discard writes to TCM //special handling for DMA: discard writes to TCM
if(PROCNUM==ARMCPU_ARM9 && AT == MMU_AT_DMA) if(PROCNUM==ARMCPU_ARM9 && AT == MMU_AT_DMA)
{ {

File diff suppressed because it is too large Load Diff

View File

@ -173,19 +173,19 @@ struct NDS_header
extern void debug(); extern void debug();
void emu_halt(); void emu_halt();
extern u64 nds_timer;
void NDS_Reschedule();
void NDS_RescheduleGXFIFO();
void NDS_RescheduleDMA();
void NDS_RescheduleTimers();
typedef struct typedef struct
{ {
s32 ARM9Cycle;
s32 ARM7Cycle;
s32 wifiCycle; s32 wifiCycle;
s32 cycles; s32 cycles;
s32 timerCycle[2][4]; u64 timerCycle[2][4];
BOOL timerOver[2][4];
s32 nextHBlank;
u32 VCount; u32 VCount;
u32 old; u32 old;
s32 diff;
BOOL lignerendu;
u16 touchX; u16 touchX;
u16 touchY; u16 touchY;

View File

@ -234,6 +234,8 @@ static INLINE void setIF(int PROCNUM, u32 flag)
if(ARMPROC.waitIRQ) if(ARMPROC.waitIRQ)
ARMPROC.newIrqFlags |= flag; ARMPROC.newIrqFlags |= flag;
extern void NDS_Reschedule();
NDS_Reschedule();
} }
static INLINE void NDS_makeARM9Int(u32 num) static INLINE void NDS_makeARM9Int(u32 num)

View File

@ -61,8 +61,10 @@ in this function: */
static void gfx3d_doFlush(); static void gfx3d_doFlush();
#ifdef USE_GEOMETRY_FIFO_EMULATION #ifdef USE_GEOMETRY_FIFO_EMULATION
#define GFX_DELAY(x) MMU.gfx3dCycles = nds.cycles + (2*x) inline void GFX_DELAY(int x) {
#define GFX_DELAY_M2(x) MMU.gfx3dCycles += (2*x) MMU.gfx3dCycles = nds_timer + (2*x); NDS_RescheduleGXFIFO(); }
inline void GFX_DELAY_M2(int x) {
MMU.gfx3dCycles += (2*x); NDS_RescheduleGXFIFO(); }
#else #else
#define GFX_DELAY(x) #define GFX_DELAY(x)
#define GFX_DELAY_M2(x) #define GFX_DELAY_M2(x)

View File

@ -85,6 +85,8 @@ int write32le(u32 b, std::ostream* os)
return 4; return 4;
} }
void writebool(bool b, std::ostream* os) { write32le(b?1:0,os); }
int write64le(u64 b, std::ostream* os) int write64le(u64 b, std::ostream* os)
{ {
u8 s[8]; u8 s[8];
@ -155,6 +157,14 @@ int read32le(u32 *Bufo, std::istream *is)
return 1; return 1;
} }
int readbool(bool *b, std::istream* is)
{
u32 temp;
int ret = read32le(&temp,is);
*b = (bool)temp;
return ret;
}
int readbuffer(std::vector<u8> &vec, std::istream* is) int readbuffer(std::vector<u8> &vec, std::istream* is)
{ {
u32 size; u32 size;

View File

@ -21,6 +21,8 @@ int read32le(u32 *Bufo, FILE *fp);
int read16le(u16 *Bufo, std::istream *is); int read16le(u16 *Bufo, std::istream *is);
inline int read16le(s16 *Bufo, std::istream* is) { return read16le((u16*)Bufo,is); } inline int read16le(s16 *Bufo, std::istream* is) { return read16le((u16*)Bufo,is); }
int read8le(u8 *Bufo, std::istream *is); int read8le(u8 *Bufo, std::istream *is);
int readbool(bool *b, std::istream* is);
void writebool(bool b, std::ostream* os);
int readbuffer(std::vector<u8> &vec, std::istream* is); int readbuffer(std::vector<u8> &vec, std::istream* is);
int writebuffer(std::vector<u8>& vec, std::ostream* os); int writebuffer(std::vector<u8>& vec, std::ostream* os);

View File

@ -52,7 +52,7 @@ int lastSaveState = 0; //Keeps track of last savestate used for quick save/load
savestates_t savestates[NB_STATES]; savestates_t savestates[NB_STATES];
#define SAVESTATE_VERSION 11 #define SAVESTATE_VERSION 12
static const char* magic = "DeSmuME SState\0"; static const char* magic = "DeSmuME SState\0";
//a savestate chunk loader can set this if it wants to permit a silent failure (for compatibility) //a savestate chunk loader can set this if it wants to permit a silent failure (for compatibility)
@ -161,17 +161,10 @@ SFORMAT SF_MEM[]={
}; };
SFORMAT SF_NDS[]={ SFORMAT SF_NDS[]={
{ "_9CY", 4, 1, &nds.ARM9Cycle},
{ "_7CY", 4, 1, &nds.ARM7Cycle},
{ "_CYC", 4, 1, &nds.cycles},
{ "_WCY", 4, 1, &nds.wifiCycle}, { "_WCY", 4, 1, &nds.wifiCycle},
{ "_TCY", 4, 8, nds.timerCycle}, { "_TCY", 8, 8, nds.timerCycle},
{ "_TOV", 4, 8, nds.timerOver},
{ "_NHB", 4, 1, &nds.nextHBlank},
{ "_VCT", 4, 1, &nds.VCount}, { "_VCT", 4, 1, &nds.VCount},
{ "_OLD", 4, 1, &nds.old}, { "_OLD", 4, 1, &nds.old},
{ "_DIF", 4, 1, &nds.diff},
{ "_LIG", 4, 1, &nds.lignerendu},
{ "_TPX", 2, 1, &nds.touchX}, { "_TPX", 2, 1, &nds.touchX},
{ "_TPY", 2, 1, &nds.touchY}, { "_TPY", 2, 1, &nds.touchY},
{ "_TPB", 4, 1, &nds.isTouch}, { "_TPB", 4, 1, &nds.isTouch},
@ -201,13 +194,15 @@ SFORMAT SF_MMU[]={
{ "MIE_", 4, 2, MMU.reg_IE}, { "MIE_", 4, 2, MMU.reg_IE},
{ "MIF_", 4, 2, MMU.reg_IF}, { "MIF_", 4, 2, MMU.reg_IF},
{ "MGXC", 8, 1, &MMU.gfx3dCycles},
{ "M_SX", 1, 2, &MMU.SPI_CNT}, { "M_SX", 1, 2, &MMU.SPI_CNT},
{ "M_SC", 1, 2, &MMU.SPI_CMD}, { "M_SC", 1, 2, &MMU.SPI_CMD},
{ "MASX", 1, 2, &MMU.AUX_SPI_CNT}, { "MASX", 1, 2, &MMU.AUX_SPI_CNT},
{ "MASC", 1, 2, &MMU.AUX_SPI_CMD}, { "MASC", 1, 2, &MMU.AUX_SPI_CMD},
{ "MDST", 4, 8, MMU.DMAStartTime}, { "MDST", 4, 8, MMU.DMAStartTime},
{ "MDCY", 4, 8, MMU.DMACycle}, { "MDCY", 8, 8, MMU.DMACycle},
{ "MDCR", 4, 8, MMU.DMACrt}, { "MDCR", 4, 8, MMU.DMACrt},
{ "MDMA", 4, 8, MMU.DMAing}, { "MDMA", 4, 8, MMU.DMAing},
{ "MDSR", 4, 8, DMASrc}, { "MDSR", 4, 8, DMASrc},
@ -217,12 +212,12 @@ SFORMAT SF_MMU[]={
{ "MDV2", 8, 1, &MMU.divResult}, { "MDV2", 8, 1, &MMU.divResult},
{ "MDV3", 8, 1, &MMU.divMod}, { "MDV3", 8, 1, &MMU.divMod},
{ "MDV4", 4, 1, &MMU.divCnt}, { "MDV4", 4, 1, &MMU.divCnt},
{ "MDV5", 4, 1, &MMU.divCycles}, { "MDV5", 8, 1, &MMU.divCycles},
{ "MSQ1", 4, 1, &MMU.sqrtRunning}, { "MSQ1", 4, 1, &MMU.sqrtRunning},
{ "MSQ2", 4, 1, &MMU.sqrtResult}, { "MSQ2", 4, 1, &MMU.sqrtResult},
{ "MSQ3", 4, 1, &MMU.sqrtCnt}, { "MSQ3", 4, 1, &MMU.sqrtCnt},
{ "MSQ4", 4, 1, &MMU.sqrtCycles}, { "MSQ4", 8, 1, &MMU.sqrtCycles},
//begin memory chips //begin memory chips
//we are skipping the firmware, because we really don't want to save the firmware to the savestate //we are skipping the firmware, because we really don't want to save the firmware to the savestate
@ -262,6 +257,9 @@ SFORMAT SF_MOVIE[]={
{ 0 } { 0 }
}; };
void nds_savestate(std::ostream* os);
bool nds_loadstate(std::istream* is, int size);
static void mmu_savestate(std::ostream* os) static void mmu_savestate(std::ostream* os)
{ {
//version //version
@ -824,6 +822,7 @@ static void writechunks(std::ostream* os) {
savestate_WriteChunk(os,3,cp15_savestate); savestate_WriteChunk(os,3,cp15_savestate);
savestate_WriteChunk(os,4,SF_MEM); savestate_WriteChunk(os,4,SF_MEM);
savestate_WriteChunk(os,5,SF_NDS); savestate_WriteChunk(os,5,SF_NDS);
savestate_WriteChunk(os,51,nds_savestate);
savestate_WriteChunk(os,60,SF_MMU); savestate_WriteChunk(os,60,SF_MMU);
savestate_WriteChunk(os,61,mmu_savestate); savestate_WriteChunk(os,61,mmu_savestate);
savestate_WriteChunk(os,7,gpu_savestate); savestate_WriteChunk(os,7,gpu_savestate);
@ -852,6 +851,7 @@ static bool ReadStateChunks(std::istream* is, s32 totalsize)
case 3: if(!cp15_loadstate(is,size)) ret=false; break; case 3: if(!cp15_loadstate(is,size)) ret=false; break;
case 4: if(!ReadStateChunk(is,SF_MEM,size)) ret=false; break; case 4: if(!ReadStateChunk(is,SF_MEM,size)) ret=false; break;
case 5: if(!ReadStateChunk(is,SF_NDS,size)) ret=false; break; case 5: if(!ReadStateChunk(is,SF_NDS,size)) ret=false; break;
case 51: if(!nds_loadstate(is,size)) ret=false; break;
case 60: if(!ReadStateChunk(is,SF_MMU,size)) ret=false; break; case 60: if(!ReadStateChunk(is,SF_MMU,size)) ret=false; break;
case 61: if(!mmu_loadstate(is,size)) ret=false; break; case 61: if(!mmu_loadstate(is,size)) ret=false; break;
case 7: if(!gpu_loadstate(is,size)) ret=false; break; case 7: if(!gpu_loadstate(is,size)) ret=false; break;

View File

@ -105,7 +105,7 @@
typedef unsigned char u8; typedef unsigned char u8;
typedef unsigned short u16; typedef unsigned short u16;
typedef unsigned int u32; typedef unsigned int u32;
typedef unsigned long u64; typedef unsigned long long u64;
typedef signed char s8; typedef signed char s8;
typedef signed short s16; typedef signed short s16;

View File

@ -594,6 +594,22 @@
> >
</File> </File>
</Filter> </Filter>
<Filter
Name="filter"
>
<File
RelativePath=".\filter\2xsai.cpp"
>
</File>
<File
RelativePath=".\filter\hq2x.cpp"
>
</File>
<File
RelativePath=".\filter\scanline.cpp"
>
</File>
</Filter>
</Filter> </Filter>
<Filter <Filter
Name="utils" Name="utils"
@ -750,10 +766,6 @@
> >
</File> </File>
</Filter> </Filter>
<File
RelativePath=".\filter\2xsai.cpp"
>
</File>
<File <File
RelativePath="..\addons.cpp" RelativePath="..\addons.cpp"
> >
@ -958,10 +970,6 @@
RelativePath="..\GPU_osd.h" RelativePath="..\GPU_osd.h"
> >
</File> </File>
<File
RelativePath=".\filter\hq2x.cpp"
>
</File>
<File <File
RelativePath="..\lua-engine.cpp" RelativePath="..\lua-engine.cpp"
> >
@ -1090,10 +1098,6 @@
RelativePath="..\saves.h" RelativePath="..\saves.h"
> >
</File> </File>
<File
RelativePath=".\filter\scanline.cpp"
>
</File>
<File <File
RelativePath="..\shaders.h" RelativePath="..\shaders.h"
> >

View File

@ -924,8 +924,8 @@ DWORD WINAPI run()
for(int i=0;i<16;i++) for(int i=0;i<16;i++)
load = load/8 + nds.runCycleCollector[(i+nds.idleFrameCounter)&15]*7/8; load = load/8 + nds.runCycleCollector[(i+nds.idleFrameCounter)&15]*7/8;
load = std::min(100,std::max(0,(int)(load*100/1120380))); load = std::min(100,std::max(0,(int)(load*100/1120380)));
sprintf(txt,"(%02d%%) %s", load, DESMUME_NAME_AND_VERSION); //sprintf(txt,"(%02d%%) %s", load, DESMUME_NAME_AND_VERSION);
SetWindowText(hwnd, txt); SetWindowText(hwnd, DESMUME_NAME_AND_VERSION);
} }
if(!skipnextframe) if(!skipnextframe)