540 lines
11 KiB
C++
540 lines
11 KiB
C++
#include <stdio.h>
|
|
#include <string.h>
|
|
#include "NDS.h"
|
|
#include "ARM.h"
|
|
#include "CP15.h"
|
|
|
|
|
|
namespace NDS
|
|
{
|
|
|
|
ARM* ARM9;
|
|
ARM* ARM7;
|
|
|
|
s32 ARM9Cycles, ARM7Cycles;
|
|
|
|
u8 ARM9BIOS[0x1000];
|
|
u8 ARM7BIOS[0x4000];
|
|
|
|
u8 MainRAM[0x400000];
|
|
|
|
u8 SharedWRAM[0x8000];
|
|
u8 WRAMCnt;
|
|
u8* SWRAM_ARM9;
|
|
u8* SWRAM_ARM7;
|
|
u32 SWRAM_ARM9Mask;
|
|
u32 SWRAM_ARM7Mask;
|
|
|
|
u8 ARM7WRAM[0x10000];
|
|
|
|
u8 ARM9ITCM[0x8000];
|
|
u32 ARM9ITCMSize;
|
|
u8 ARM9DTCM[0x4000];
|
|
u32 ARM9DTCMBase, ARM9DTCMSize;
|
|
|
|
// IO shit
|
|
u16 IPCSync9, IPCSync7;
|
|
|
|
bool Running;
|
|
|
|
|
|
void Init()
|
|
{
|
|
ARM9 = new ARM(0);
|
|
ARM7 = new ARM(1);
|
|
|
|
Reset();
|
|
}
|
|
|
|
void Reset()
|
|
{
|
|
FILE* f;
|
|
|
|
f = fopen("bios9.bin", "rb");
|
|
if (!f)
|
|
printf("ARM9 BIOS not found\n");
|
|
else
|
|
{
|
|
fseek(f, 0, SEEK_SET);
|
|
fread(ARM9BIOS, 0x1000, 1, f);
|
|
|
|
printf("ARM9 BIOS loaded: %08X\n", ARM9Read32(0xFFFF0000));
|
|
fclose(f);
|
|
}
|
|
|
|
f = fopen("bios7.bin", "rb");
|
|
if (!f)
|
|
printf("ARM7 BIOS not found\n");
|
|
else
|
|
{
|
|
fseek(f, 0, SEEK_SET);
|
|
fread(ARM7BIOS, 0x4000, 1, f);
|
|
|
|
printf("ARM7 BIOS loaded: %08X\n", ARM7Read32(0x00000000));
|
|
fclose(f);
|
|
}
|
|
|
|
memset(MainRAM, 0, 0x400000);
|
|
memset(SharedWRAM, 0, 0x8000);
|
|
memset(ARM7WRAM, 0, 0x10000);
|
|
memset(ARM9ITCM, 0, 0x8000);
|
|
memset(ARM9DTCM, 0, 0x4000);
|
|
|
|
WRAMCnt = 0;
|
|
MapSharedWRAM();
|
|
|
|
ARM9ITCMSize = 0;
|
|
ARM9DTCMBase = 0xFFFFFFFF;
|
|
ARM9DTCMSize = 0;
|
|
|
|
IPCSync9 = 0;
|
|
IPCSync7 = 0;
|
|
|
|
ARM9->Reset();
|
|
ARM7->Reset();
|
|
CP15::Reset();
|
|
|
|
ARM9Cycles = 0;
|
|
ARM7Cycles = 0;
|
|
|
|
Running = true; // hax
|
|
}
|
|
|
|
|
|
void RunFrame()
|
|
{
|
|
s32 framecycles = 560190<<1;
|
|
|
|
// very gross and temp. loop
|
|
|
|
while (Running && framecycles>0)
|
|
{
|
|
ARM9Cycles = ARM9->Execute(32 + ARM9Cycles);
|
|
ARM7Cycles = ARM7->Execute(16 + ARM7Cycles);
|
|
|
|
framecycles -= 32;
|
|
}
|
|
}
|
|
|
|
|
|
void Halt()
|
|
{
|
|
Running = false;
|
|
}
|
|
|
|
|
|
void MapSharedWRAM()
|
|
{
|
|
switch (WRAMCnt & 0x3)
|
|
{
|
|
case 0:
|
|
SWRAM_ARM9 = &SharedWRAM[0];
|
|
SWRAM_ARM9Mask = 0x7FFF;
|
|
SWRAM_ARM7 = NULL;
|
|
SWRAM_ARM7Mask = 0;
|
|
break;
|
|
|
|
case 1:
|
|
SWRAM_ARM9 = &SharedWRAM[0x4000];
|
|
SWRAM_ARM9Mask = 0x3FFF;
|
|
SWRAM_ARM7 = &SharedWRAM[0];
|
|
SWRAM_ARM7Mask = 0x3FFF;
|
|
break;
|
|
|
|
case 2:
|
|
SWRAM_ARM9 = &SharedWRAM[0];
|
|
SWRAM_ARM9Mask = 0x3FFF;
|
|
SWRAM_ARM7 = &SharedWRAM[0x4000];
|
|
SWRAM_ARM7Mask = 0x3FFF;
|
|
break;
|
|
|
|
case 3:
|
|
SWRAM_ARM9 = NULL;
|
|
SWRAM_ARM9Mask = 0;
|
|
SWRAM_ARM7 = &SharedWRAM[0];
|
|
SWRAM_ARM7Mask = 0x7FFF;
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
u8 ARM9Read8(u32 addr)
|
|
{
|
|
if ((addr & 0xFFFFF000) == 0xFFFF0000)
|
|
{
|
|
return *(u8*)&ARM9BIOS[addr & 0xFFF];
|
|
}
|
|
if (addr < ARM9ITCMSize)
|
|
{
|
|
return *(u8*)&ARM9ITCM[addr & 0x7FFF];
|
|
}
|
|
if (addr >= ARM9DTCMBase && addr < (ARM9DTCMBase + ARM9DTCMSize))
|
|
{
|
|
return *(u8*)&ARM9DTCM[(addr - ARM9DTCMBase) & 0x3FFF];
|
|
}
|
|
|
|
switch (addr & 0xFF000000)
|
|
{
|
|
case 0x02000000:
|
|
return *(u8*)&MainRAM[addr & 0x3FFFFF];
|
|
|
|
case 0x03000000:
|
|
if (SWRAM_ARM9) return *(u8*)&SWRAM_ARM9[addr & SWRAM_ARM9Mask];
|
|
else return 0;
|
|
|
|
case 0x04000000:
|
|
switch (addr)
|
|
{
|
|
case 0x04000247:
|
|
return WRAMCnt;
|
|
}
|
|
}
|
|
|
|
printf("unknown arm9 read8 %08X\n", addr);
|
|
return 0;
|
|
}
|
|
|
|
u16 ARM9Read16(u32 addr)
|
|
{
|
|
if ((addr & 0xFFFFF000) == 0xFFFF0000)
|
|
{
|
|
return *(u16*)&ARM9BIOS[addr & 0xFFF];
|
|
}
|
|
if (addr < ARM9ITCMSize)
|
|
{
|
|
return *(u16*)&ARM9ITCM[addr & 0x7FFF];
|
|
}
|
|
if (addr >= ARM9DTCMBase && addr < (ARM9DTCMBase + ARM9DTCMSize))
|
|
{
|
|
return *(u16*)&ARM9DTCM[(addr - ARM9DTCMBase) & 0x3FFF];
|
|
}
|
|
|
|
switch (addr & 0xFF000000)
|
|
{
|
|
case 0x02000000:
|
|
return *(u16*)&MainRAM[addr & 0x3FFFFF];
|
|
|
|
case 0x03000000:
|
|
if (SWRAM_ARM9) return *(u16*)&SWRAM_ARM9[addr & SWRAM_ARM9Mask];
|
|
else return 0;
|
|
|
|
case 0x04000000:
|
|
switch (addr)
|
|
{
|
|
case 0x04000180: return IPCSync9;
|
|
}
|
|
}
|
|
|
|
printf("unknown arm9 read16 %08X\n", addr);
|
|
return 0;
|
|
}
|
|
|
|
u32 ARM9Read32(u32 addr)
|
|
{
|
|
if ((addr & 0xFFFFF000) == 0xFFFF0000)
|
|
{
|
|
return *(u32*)&ARM9BIOS[addr & 0xFFF];
|
|
}
|
|
if (addr < ARM9ITCMSize)
|
|
{
|
|
return *(u32*)&ARM9ITCM[addr & 0x7FFF];
|
|
}
|
|
if (addr >= ARM9DTCMBase && addr < (ARM9DTCMBase + ARM9DTCMSize))
|
|
{
|
|
return *(u32*)&ARM9DTCM[(addr - ARM9DTCMBase) & 0x3FFF];
|
|
}
|
|
|
|
switch (addr & 0xFF000000)
|
|
{
|
|
case 0x02000000:
|
|
return *(u32*)&MainRAM[addr & 0x3FFFFF];
|
|
|
|
case 0x03000000:
|
|
if (SWRAM_ARM9) return *(u32*)&SWRAM_ARM9[addr & SWRAM_ARM9Mask];
|
|
else return 0;
|
|
}
|
|
|
|
printf("unknown arm9 read32 %08X | %08X\n", addr, ARM9->R[15]);
|
|
return 0;
|
|
}
|
|
|
|
void ARM9Write8(u32 addr, u8 val)
|
|
{
|
|
if (addr < ARM9ITCMSize)
|
|
{
|
|
*(u8*)&ARM9ITCM[addr & 0x7FFF] = val;
|
|
return;
|
|
}
|
|
if (addr >= ARM9DTCMBase && addr < (ARM9DTCMBase + ARM9DTCMSize))
|
|
{
|
|
*(u8*)&ARM9DTCM[(addr - ARM9DTCMBase) & 0x3FFF] = val;
|
|
return;
|
|
}
|
|
|
|
switch (addr & 0xFF000000)
|
|
{
|
|
case 0x02000000:
|
|
*(u8*)&MainRAM[addr & 0x3FFFFF] = val;
|
|
return;
|
|
|
|
case 0x03000000:
|
|
if (SWRAM_ARM9) *(u8*)&SWRAM_ARM9[addr & SWRAM_ARM9Mask] = val;
|
|
return;
|
|
|
|
case 0x04000000:
|
|
switch (addr)
|
|
{
|
|
case 0x04000247:
|
|
WRAMCnt = val;
|
|
MapSharedWRAM();
|
|
return;
|
|
}
|
|
}
|
|
|
|
printf("unknown arm9 write8 %08X %02X\n", addr, val);
|
|
}
|
|
|
|
void ARM9Write16(u32 addr, u16 val)
|
|
{
|
|
if (addr < ARM9ITCMSize)
|
|
{
|
|
*(u16*)&ARM9ITCM[addr & 0x7FFF] = val;
|
|
return;
|
|
}
|
|
if (addr >= ARM9DTCMBase && addr < (ARM9DTCMBase + ARM9DTCMSize))
|
|
{
|
|
*(u16*)&ARM9DTCM[(addr - ARM9DTCMBase) & 0x3FFF] = val;
|
|
return;
|
|
}
|
|
|
|
switch (addr & 0xFF000000)
|
|
{
|
|
case 0x02000000:
|
|
*(u16*)&MainRAM[addr & 0x3FFFFF] = val;
|
|
return;
|
|
|
|
case 0x03000000:
|
|
if (SWRAM_ARM9) *(u16*)&SWRAM_ARM9[addr & SWRAM_ARM9Mask] = val;
|
|
return;
|
|
|
|
case 0x04000000:
|
|
switch (addr)
|
|
{
|
|
case 0x04000180:
|
|
IPCSync7 &= 0xFFF0;
|
|
IPCSync7 |= ((val & 0x0F00) >> 8);
|
|
IPCSync9 &= 0xB0FF;
|
|
IPCSync9 |= (val & 0x4F00);
|
|
if ((val & 0x2000) && (IPCSync7 & 0x4000))
|
|
{
|
|
printf("ARM9 IPCSYNC IRQ TODO\n");
|
|
}
|
|
return;
|
|
}
|
|
}
|
|
|
|
printf("unknown arm9 write16 %08X %04X\n", addr, val);
|
|
}
|
|
|
|
void ARM9Write32(u32 addr, u32 val)
|
|
{
|
|
if (addr < ARM9ITCMSize)
|
|
{
|
|
*(u32*)&ARM9ITCM[addr & 0x7FFF] = val;
|
|
return;
|
|
}
|
|
if (addr >= ARM9DTCMBase && addr < (ARM9DTCMBase + ARM9DTCMSize))
|
|
{
|
|
*(u32*)&ARM9DTCM[(addr - ARM9DTCMBase) & 0x3FFF] = val;
|
|
return;
|
|
}
|
|
|
|
switch (addr & 0xFF000000)
|
|
{
|
|
case 0x02000000:
|
|
*(u32*)&MainRAM[addr & 0x3FFFFF] = val;
|
|
return;
|
|
|
|
case 0x03000000:
|
|
if (SWRAM_ARM9) *(u32*)&SWRAM_ARM9[addr & SWRAM_ARM9Mask] = val;
|
|
return;
|
|
}
|
|
|
|
printf("unknown arm9 write32 %08X %08X | %08X\n", addr, val, ARM9->R[15]);
|
|
}
|
|
|
|
|
|
|
|
u8 ARM7Read8(u32 addr)
|
|
{
|
|
if (addr < 0x00004000)
|
|
{
|
|
return *(u8*)&ARM7BIOS[addr];
|
|
}
|
|
|
|
switch (addr & 0xFF800000)
|
|
{
|
|
case 0x02000000:
|
|
return *(u8*)&MainRAM[addr & 0x3FFFFF];
|
|
|
|
case 0x03000000:
|
|
if (SWRAM_ARM7) return *(u8*)&SWRAM_ARM7[addr & SWRAM_ARM7Mask];
|
|
else return *(u8*)&ARM7WRAM[addr & 0xFFFF];
|
|
|
|
case 0x03800000:
|
|
return *(u8*)&ARM7WRAM[addr & 0xFFFF];
|
|
|
|
case 0x04000000:
|
|
switch (addr)
|
|
{
|
|
case 0x04000241:
|
|
return WRAMCnt;
|
|
}
|
|
}
|
|
|
|
printf("unknown arm7 read8 %08X\n", addr);
|
|
return 0;
|
|
}
|
|
|
|
u16 ARM7Read16(u32 addr)
|
|
{
|
|
if (addr < 0x00004000)
|
|
{
|
|
return *(u16*)&ARM7BIOS[addr];
|
|
}
|
|
|
|
switch (addr & 0xFF800000)
|
|
{
|
|
case 0x02000000:
|
|
return *(u16*)&MainRAM[addr & 0x3FFFFF];
|
|
|
|
case 0x03000000:
|
|
if (SWRAM_ARM7) return *(u16*)&SWRAM_ARM7[addr & SWRAM_ARM7Mask];
|
|
else return *(u16*)&ARM7WRAM[addr & 0xFFFF];
|
|
|
|
case 0x03800000:
|
|
return *(u16*)&ARM7WRAM[addr & 0xFFFF];
|
|
|
|
case 0x04000000:
|
|
switch (addr)
|
|
{
|
|
case 0x04000180: return IPCSync7;
|
|
}
|
|
}
|
|
|
|
printf("unknown arm7 read16 %08X\n", addr);
|
|
return 0;
|
|
}
|
|
|
|
u32 ARM7Read32(u32 addr)
|
|
{
|
|
if (addr < 0x00004000)
|
|
{
|
|
return *(u32*)&ARM7BIOS[addr];
|
|
}
|
|
|
|
switch (addr & 0xFF800000)
|
|
{
|
|
case 0x02000000:
|
|
return *(u32*)&MainRAM[addr & 0x3FFFFF];
|
|
|
|
case 0x03000000:
|
|
if (SWRAM_ARM7) return *(u32*)&SWRAM_ARM7[addr & SWRAM_ARM7Mask];
|
|
else return *(u32*)&ARM7WRAM[addr & 0xFFFF];
|
|
|
|
case 0x03800000:
|
|
return *(u32*)&ARM7WRAM[addr & 0xFFFF];
|
|
|
|
case 0x04000000:
|
|
switch (addr)
|
|
{
|
|
case 0x040001A4:
|
|
return 0x00800000; // hax
|
|
}
|
|
}
|
|
if ((addr&0xFF000000) == 0xEA000000) Halt();
|
|
printf("unknown arm7 read32 %08X | %08X\n", addr, ARM7->R[15]);
|
|
return 0;
|
|
}
|
|
|
|
void ARM7Write8(u32 addr, u8 val)
|
|
{
|
|
switch (addr & 0xFF800000)
|
|
{
|
|
case 0x02000000:
|
|
*(u8*)&MainRAM[addr & 0x3FFFFF] = val;
|
|
return;
|
|
|
|
case 0x03000000:
|
|
if (SWRAM_ARM7) *(u8*)&SWRAM_ARM7[addr & SWRAM_ARM7Mask] = val;
|
|
else *(u8*)&ARM7WRAM[addr & 0xFFFF] = val;
|
|
return;
|
|
|
|
case 0x03800000:
|
|
*(u8*)&ARM7WRAM[addr & 0xFFFF] = val;
|
|
return;
|
|
}
|
|
|
|
printf("unknown arm7 write8 %08X %02X | %08X\n", addr, val, ARM7->R[15]);
|
|
}
|
|
|
|
void ARM7Write16(u32 addr, u16 val)
|
|
{
|
|
switch (addr & 0xFF800000)
|
|
{
|
|
case 0x02000000:
|
|
*(u16*)&MainRAM[addr & 0x3FFFFF] = val;
|
|
return;
|
|
|
|
case 0x03000000:
|
|
if (SWRAM_ARM7) *(u16*)&SWRAM_ARM7[addr & SWRAM_ARM7Mask] = val;
|
|
else *(u16*)&ARM7WRAM[addr & 0xFFFF] = val;
|
|
return;
|
|
|
|
case 0x03800000:
|
|
*(u16*)&ARM7WRAM[addr & 0xFFFF] = val;
|
|
return;
|
|
|
|
case 0x04000000:
|
|
switch (addr)
|
|
{
|
|
case 0x04000180:
|
|
IPCSync9 &= 0xFFF0;
|
|
IPCSync9 |= ((val & 0x0F00) >> 8);
|
|
IPCSync7 &= 0xB0FF;
|
|
IPCSync7 |= (val & 0x4F00);
|
|
if ((val & 0x2000) && (IPCSync9 & 0x4000))
|
|
{
|
|
printf("ARM7 IPCSYNC IRQ TODO\n");
|
|
}
|
|
return;
|
|
}
|
|
}
|
|
|
|
printf("unknown arm7 write16 %08X %04X | %08X\n", addr, val, ARM7->R[15]);
|
|
}
|
|
|
|
void ARM7Write32(u32 addr, u32 val)
|
|
{
|
|
switch (addr & 0xFF800000)
|
|
{
|
|
case 0x02000000:
|
|
*(u32*)&MainRAM[addr & 0x3FFFFF] = val;
|
|
return;
|
|
|
|
case 0x03000000:
|
|
if (SWRAM_ARM7) *(u32*)&SWRAM_ARM7[addr & SWRAM_ARM7Mask] = val;
|
|
else *(u32*)&ARM7WRAM[addr & 0xFFFF] = val;
|
|
return;
|
|
|
|
case 0x03800000:
|
|
*(u32*)&ARM7WRAM[addr & 0xFFFF] = val;
|
|
return;
|
|
}
|
|
|
|
printf("unknown arm7 write32 %08X %08X | %08X\n", addr, val, ARM7->R[15]);
|
|
}
|
|
|
|
}
|