2016-12-05 17:02:29 +00:00
|
|
|
/*
|
|
|
|
Copyright 2016-2017 StapleButter
|
|
|
|
|
|
|
|
This file is part of melonDS.
|
|
|
|
|
|
|
|
melonDS 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 3 of the License, or (at your option)
|
|
|
|
any later version.
|
|
|
|
|
|
|
|
melonDS 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 melonDS. If not, see http://www.gnu.org/licenses/.
|
|
|
|
*/
|
|
|
|
|
2016-05-16 15:48:40 +00:00
|
|
|
#include <stdio.h>
|
2016-12-03 12:42:27 +00:00
|
|
|
#include <string.h>
|
2016-05-16 15:48:40 +00:00
|
|
|
#include "NDS.h"
|
2016-11-03 00:38:58 +00:00
|
|
|
#include "ARM.h"
|
2016-12-03 15:13:04 +00:00
|
|
|
#include "CP15.h"
|
2017-01-22 19:34:59 +00:00
|
|
|
#include "NDSCart.h"
|
2017-01-18 00:33:06 +00:00
|
|
|
#include "DMA.h"
|
2017-01-17 00:58:25 +00:00
|
|
|
#include "FIFO.h"
|
2017-01-18 03:03:19 +00:00
|
|
|
#include "GPU.h"
|
2017-04-06 17:44:34 +00:00
|
|
|
#include "SPU.h"
|
2016-12-04 02:20:50 +00:00
|
|
|
#include "SPI.h"
|
2017-01-20 00:18:30 +00:00
|
|
|
#include "RTC.h"
|
2016-12-06 16:32:51 +00:00
|
|
|
#include "Wifi.h"
|
2017-09-21 01:59:12 +00:00
|
|
|
#include "Platform.h"
|
2016-05-16 15:48:40 +00:00
|
|
|
|
|
|
|
|
|
|
|
namespace NDS
|
|
|
|
{
|
|
|
|
|
2017-01-20 14:13:44 +00:00
|
|
|
// TODO LIST
|
|
|
|
// * stick all the variables in a big structure?
|
|
|
|
// would make it easier to deal with savestates
|
2017-01-17 00:58:25 +00:00
|
|
|
|
2016-11-03 00:38:58 +00:00
|
|
|
ARM* ARM9;
|
|
|
|
ARM* ARM7;
|
|
|
|
|
2017-01-31 02:54:51 +00:00
|
|
|
/*s32 ARM9Cycles, ARM7Cycles;
|
2016-12-05 16:08:24 +00:00
|
|
|
s32 CompensatedCycles;
|
2017-01-31 02:54:51 +00:00
|
|
|
s32 SchedCycles;*/
|
|
|
|
s32 CurIterationCycles;
|
|
|
|
s32 ARM7Offset;
|
|
|
|
|
|
|
|
SchedEvent SchedList[Event_MAX];
|
|
|
|
u32 SchedListMask;
|
2016-11-24 17:31:49 +00:00
|
|
|
|
2017-02-17 04:33:37 +00:00
|
|
|
u32 CPUStop;
|
|
|
|
|
2016-11-03 00:38:58 +00:00
|
|
|
u8 ARM9BIOS[0x1000];
|
|
|
|
u8 ARM7BIOS[0x4000];
|
2016-05-16 15:48:40 +00:00
|
|
|
|
2016-12-03 12:42:27 +00:00
|
|
|
u8 MainRAM[0x400000];
|
2016-12-03 15:13:04 +00:00
|
|
|
|
2016-12-03 17:29:19 +00:00
|
|
|
u8 SharedWRAM[0x8000];
|
|
|
|
u8 WRAMCnt;
|
|
|
|
u8* SWRAM_ARM9;
|
|
|
|
u8* SWRAM_ARM7;
|
|
|
|
u32 SWRAM_ARM9Mask;
|
|
|
|
u32 SWRAM_ARM7Mask;
|
|
|
|
|
2016-12-03 12:42:27 +00:00
|
|
|
u8 ARM7WRAM[0x10000];
|
|
|
|
|
2017-01-22 19:34:59 +00:00
|
|
|
u16 ExMemCnt[2];
|
|
|
|
|
|
|
|
u8 ROMSeed0[2*8];
|
|
|
|
u8 ROMSeed1[2*8];
|
|
|
|
|
2016-12-03 15:13:04 +00:00
|
|
|
// IO shit
|
2016-12-04 02:20:50 +00:00
|
|
|
u32 IME[2];
|
|
|
|
u32 IE[2], IF[2];
|
|
|
|
|
2017-01-18 01:20:45 +00:00
|
|
|
u8 PostFlag9;
|
|
|
|
u8 PostFlag7;
|
2016-12-06 16:32:51 +00:00
|
|
|
u16 PowerControl9;
|
|
|
|
u16 PowerControl7;
|
|
|
|
|
2017-02-05 16:15:17 +00:00
|
|
|
u16 ARM7BIOSProt;
|
|
|
|
|
2016-12-05 16:08:24 +00:00
|
|
|
Timer Timers[8];
|
2017-04-24 23:14:26 +00:00
|
|
|
u8 TimerCheckMask[2];
|
2016-12-05 16:08:24 +00:00
|
|
|
|
2017-01-18 00:33:06 +00:00
|
|
|
DMA* DMAs[8];
|
|
|
|
u32 DMA9Fill[4];
|
|
|
|
|
2016-12-03 15:13:04 +00:00
|
|
|
u16 IPCSync9, IPCSync7;
|
2017-01-17 00:58:25 +00:00
|
|
|
u16 IPCFIFOCnt9, IPCFIFOCnt7;
|
2017-02-07 21:41:33 +00:00
|
|
|
FIFO<u32>* IPCFIFO9; // FIFO in which the ARM9 writes
|
|
|
|
FIFO<u32>* IPCFIFO7;
|
2016-12-03 15:13:04 +00:00
|
|
|
|
2017-01-18 01:20:45 +00:00
|
|
|
u16 DivCnt;
|
|
|
|
u32 DivNumerator[2];
|
|
|
|
u32 DivDenominator[2];
|
|
|
|
u32 DivQuotient[2];
|
|
|
|
u32 DivRemainder[2];
|
|
|
|
|
2017-01-31 20:53:45 +00:00
|
|
|
u16 SqrtCnt;
|
|
|
|
u32 SqrtVal[2];
|
|
|
|
u32 SqrtRes;
|
|
|
|
|
2016-12-23 20:22:22 +00:00
|
|
|
u32 KeyInput;
|
2017-06-04 20:34:31 +00:00
|
|
|
u16 KeyCnt;
|
|
|
|
u16 RCnt;
|
2016-12-23 20:22:22 +00:00
|
|
|
|
2016-11-24 17:31:49 +00:00
|
|
|
bool Running;
|
|
|
|
|
2016-05-16 15:48:40 +00:00
|
|
|
|
2017-02-07 21:23:46 +00:00
|
|
|
bool Init()
|
2016-05-16 15:48:40 +00:00
|
|
|
{
|
2016-11-03 00:38:58 +00:00
|
|
|
ARM9 = new ARM(0);
|
|
|
|
ARM7 = new ARM(1);
|
|
|
|
|
2017-01-18 00:33:06 +00:00
|
|
|
DMAs[0] = new DMA(0, 0);
|
|
|
|
DMAs[1] = new DMA(0, 1);
|
|
|
|
DMAs[2] = new DMA(0, 2);
|
|
|
|
DMAs[3] = new DMA(0, 3);
|
|
|
|
DMAs[4] = new DMA(1, 0);
|
|
|
|
DMAs[5] = new DMA(1, 1);
|
|
|
|
DMAs[6] = new DMA(1, 2);
|
|
|
|
DMAs[7] = new DMA(1, 3);
|
|
|
|
|
2017-02-07 21:41:33 +00:00
|
|
|
IPCFIFO9 = new FIFO<u32>(16);
|
|
|
|
IPCFIFO7 = new FIFO<u32>(16);
|
2017-01-17 00:58:25 +00:00
|
|
|
|
2017-02-07 21:23:46 +00:00
|
|
|
if (!NDSCart::Init()) return false;
|
|
|
|
if (!GPU::Init()) return false;
|
2017-04-06 17:44:34 +00:00
|
|
|
if (!SPU::Init()) return false;
|
2017-02-07 21:23:46 +00:00
|
|
|
if (!SPI::Init()) return false;
|
|
|
|
if (!RTC::Init()) return false;
|
2017-05-11 17:57:49 +00:00
|
|
|
if (!Wifi::Init()) return false;
|
2016-12-04 02:20:50 +00:00
|
|
|
|
2017-02-07 21:23:46 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DeInit()
|
|
|
|
{
|
|
|
|
delete ARM9;
|
|
|
|
delete ARM7;
|
|
|
|
|
|
|
|
for (int i = 0; i < 8; i++)
|
|
|
|
delete DMAs[i];
|
|
|
|
|
|
|
|
delete IPCFIFO9;
|
|
|
|
delete IPCFIFO7;
|
|
|
|
|
|
|
|
NDSCart::DeInit();
|
|
|
|
GPU::DeInit();
|
2017-04-06 17:44:34 +00:00
|
|
|
SPU::DeInit();
|
2017-02-07 21:23:46 +00:00
|
|
|
SPI::DeInit();
|
|
|
|
RTC::DeInit();
|
2017-05-11 17:57:49 +00:00
|
|
|
Wifi::DeInit();
|
2016-05-16 15:48:40 +00:00
|
|
|
}
|
|
|
|
|
2016-12-06 16:32:51 +00:00
|
|
|
|
2017-02-01 20:35:00 +00:00
|
|
|
void SetupDirectBoot()
|
|
|
|
{
|
2016-12-06 16:32:51 +00:00
|
|
|
u32 bootparams[8];
|
2017-02-01 20:35:00 +00:00
|
|
|
memcpy(bootparams, &NDSCart::CartROM[0x20], 8*4);
|
2016-12-06 16:32:51 +00:00
|
|
|
|
|
|
|
printf("ARM9: offset=%08X entry=%08X RAM=%08X size=%08X\n",
|
|
|
|
bootparams[0], bootparams[1], bootparams[2], bootparams[3]);
|
|
|
|
printf("ARM7: offset=%08X entry=%08X RAM=%08X size=%08X\n",
|
|
|
|
bootparams[4], bootparams[5], bootparams[6], bootparams[7]);
|
|
|
|
|
2017-01-31 02:54:51 +00:00
|
|
|
MapSharedWRAM(3);
|
|
|
|
|
2016-12-06 16:32:51 +00:00
|
|
|
for (u32 i = 0; i < bootparams[3]; i+=4)
|
|
|
|
{
|
2017-02-01 20:35:00 +00:00
|
|
|
u32 tmp = *(u32*)&NDSCart::CartROM[bootparams[0]+i];
|
2016-12-06 16:32:51 +00:00
|
|
|
ARM9Write32(bootparams[2]+i, tmp);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (u32 i = 0; i < bootparams[7]; i+=4)
|
|
|
|
{
|
2017-02-01 20:35:00 +00:00
|
|
|
u32 tmp = *(u32*)&NDSCart::CartROM[bootparams[4]+i];
|
2016-12-23 20:22:22 +00:00
|
|
|
ARM7Write32(bootparams[6]+i, tmp);
|
2016-12-06 16:32:51 +00:00
|
|
|
}
|
|
|
|
|
2017-01-31 02:54:51 +00:00
|
|
|
for (u32 i = 0; i < 0x170; i+=4)
|
|
|
|
{
|
2017-02-01 20:35:00 +00:00
|
|
|
u32 tmp = *(u32*)&NDSCart::CartROM[i];
|
2017-01-31 02:54:51 +00:00
|
|
|
ARM9Write32(0x027FFE00+i, tmp);
|
|
|
|
}
|
|
|
|
|
|
|
|
ARM9Write32(0x027FF800, 0x00001FC2);
|
|
|
|
ARM9Write32(0x027FF804, 0x00001FC2);
|
2017-02-05 15:50:20 +00:00
|
|
|
ARM9Write16(0x027FF808, *(u16*)&NDSCart::CartROM[0x15E]);
|
|
|
|
ARM9Write16(0x027FF80A, *(u16*)&NDSCart::CartROM[0x6C]);
|
|
|
|
|
|
|
|
ARM9Write16(0x027FF850, 0x5835);
|
|
|
|
|
2017-01-31 02:54:51 +00:00
|
|
|
ARM9Write32(0x027FFC00, 0x00001FC2);
|
|
|
|
ARM9Write32(0x027FFC04, 0x00001FC2);
|
2017-02-05 15:50:20 +00:00
|
|
|
ARM9Write16(0x027FFC08, *(u16*)&NDSCart::CartROM[0x15E]);
|
|
|
|
ARM9Write16(0x027FFC0A, *(u16*)&NDSCart::CartROM[0x6C]);
|
2017-01-31 02:54:51 +00:00
|
|
|
|
2017-02-05 15:50:20 +00:00
|
|
|
ARM9Write16(0x027FFC10, 0x5835);
|
2017-01-31 02:54:51 +00:00
|
|
|
ARM9Write16(0x027FFC30, 0xFFFF);
|
2017-02-05 15:50:20 +00:00
|
|
|
ARM9Write16(0x027FFC40, 0x0001);
|
2017-01-31 02:54:51 +00:00
|
|
|
|
2016-12-23 20:22:22 +00:00
|
|
|
CP15::Write(0x910, 0x0300000A);
|
|
|
|
CP15::Write(0x911, 0x00000020);
|
|
|
|
CP15::Write(0x100, 0x00050000);
|
|
|
|
|
2017-03-20 23:53:04 +00:00
|
|
|
ARM9->R[12] = bootparams[1];
|
|
|
|
ARM9->R[13] = 0x03002F7C;
|
|
|
|
ARM9->R[14] = bootparams[1];
|
|
|
|
ARM9->R_IRQ[0] = 0x03003F80;
|
|
|
|
ARM9->R_SVC[0] = 0x03003FC0;
|
|
|
|
|
|
|
|
ARM7->R[12] = bootparams[5];
|
|
|
|
ARM7->R[13] = 0x0380FD80;
|
|
|
|
ARM7->R[14] = bootparams[5];
|
|
|
|
ARM7->R_IRQ[0] = 0x0380FF80;
|
|
|
|
ARM7->R_SVC[0] = 0x0380FFC0;
|
|
|
|
|
2016-12-06 16:32:51 +00:00
|
|
|
ARM9->JumpTo(bootparams[1]);
|
|
|
|
ARM7->JumpTo(bootparams[5]);
|
2017-02-01 20:57:25 +00:00
|
|
|
|
2017-04-26 14:17:03 +00:00
|
|
|
PostFlag9 = 0x01;
|
|
|
|
PostFlag7 = 0x01;
|
|
|
|
|
2017-02-01 20:57:25 +00:00
|
|
|
PowerControl9 = 0x820F;
|
|
|
|
GPU::DisplaySwap(PowerControl9);
|
2017-02-05 16:15:17 +00:00
|
|
|
|
2017-06-04 23:58:14 +00:00
|
|
|
// checkme
|
|
|
|
RCnt = 0x8000;
|
|
|
|
|
2017-04-07 15:37:49 +00:00
|
|
|
SPU::SetBias(0x200);
|
|
|
|
|
2017-02-05 16:15:17 +00:00
|
|
|
ARM7BIOSProt = 0x1204;
|
2017-03-20 23:53:04 +00:00
|
|
|
|
|
|
|
SPI_Firmware::SetupDirectBoot();
|
2016-12-06 16:32:51 +00:00
|
|
|
}
|
|
|
|
|
2016-05-16 15:48:40 +00:00
|
|
|
void Reset()
|
|
|
|
{
|
|
|
|
FILE* f;
|
2017-01-18 00:33:06 +00:00
|
|
|
u32 i;
|
2016-05-16 15:48:40 +00:00
|
|
|
|
|
|
|
f = fopen("bios9.bin", "rb");
|
|
|
|
if (!f)
|
2017-05-09 02:52:17 +00:00
|
|
|
{
|
2016-05-16 15:48:40 +00:00
|
|
|
printf("ARM9 BIOS not found\n");
|
2017-05-09 02:52:17 +00:00
|
|
|
|
|
|
|
for (i = 0; i < 16; i++)
|
|
|
|
((u32*)ARM9BIOS)[i] = 0xE7FFDEFF;
|
|
|
|
}
|
2016-05-16 15:48:40 +00:00
|
|
|
else
|
|
|
|
{
|
2016-11-03 00:38:58 +00:00
|
|
|
fseek(f, 0, SEEK_SET);
|
|
|
|
fread(ARM9BIOS, 0x1000, 1, f);
|
2016-05-16 15:48:40 +00:00
|
|
|
|
2017-02-01 18:30:57 +00:00
|
|
|
printf("ARM9 BIOS loaded\n");
|
2016-05-16 15:48:40 +00:00
|
|
|
fclose(f);
|
|
|
|
}
|
2016-11-03 00:38:58 +00:00
|
|
|
|
|
|
|
f = fopen("bios7.bin", "rb");
|
|
|
|
if (!f)
|
2017-05-09 02:52:17 +00:00
|
|
|
{
|
2016-11-03 00:38:58 +00:00
|
|
|
printf("ARM7 BIOS not found\n");
|
2017-05-09 02:52:17 +00:00
|
|
|
|
|
|
|
for (i = 0; i < 16; i++)
|
|
|
|
((u32*)ARM7BIOS)[i] = 0xE7FFDEFF;
|
|
|
|
}
|
2016-11-03 00:38:58 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
fseek(f, 0, SEEK_SET);
|
|
|
|
fread(ARM7BIOS, 0x4000, 1, f);
|
|
|
|
|
2017-02-01 18:30:57 +00:00
|
|
|
printf("ARM7 BIOS loaded\n");
|
2016-11-03 00:38:58 +00:00
|
|
|
fclose(f);
|
|
|
|
}
|
2016-11-24 17:31:49 +00:00
|
|
|
|
2016-12-03 12:42:27 +00:00
|
|
|
memset(MainRAM, 0, 0x400000);
|
2016-12-03 17:29:19 +00:00
|
|
|
memset(SharedWRAM, 0, 0x8000);
|
2016-12-03 12:42:27 +00:00
|
|
|
memset(ARM7WRAM, 0, 0x10000);
|
2016-12-03 15:13:04 +00:00
|
|
|
|
2016-12-06 16:32:51 +00:00
|
|
|
MapSharedWRAM(0);
|
2016-12-03 17:29:19 +00:00
|
|
|
|
2017-01-22 19:34:59 +00:00
|
|
|
ExMemCnt[0] = 0;
|
|
|
|
ExMemCnt[1] = 0;
|
|
|
|
memset(ROMSeed0, 0, 2*8);
|
|
|
|
memset(ROMSeed1, 0, 2*8);
|
|
|
|
|
2016-12-04 02:20:50 +00:00
|
|
|
IME[0] = 0;
|
2017-11-09 15:02:37 +00:00
|
|
|
IE[0] = 0;
|
|
|
|
IF[0] = 0;
|
2016-12-04 02:20:50 +00:00
|
|
|
IME[1] = 0;
|
2017-11-09 15:02:37 +00:00
|
|
|
IE[1] = 0;
|
|
|
|
IF[1] = 0;
|
2016-12-04 02:20:50 +00:00
|
|
|
|
2017-01-18 01:20:45 +00:00
|
|
|
PostFlag9 = 0x00;
|
|
|
|
PostFlag7 = 0x00;
|
2016-12-06 16:32:51 +00:00
|
|
|
PowerControl9 = 0x0001;
|
|
|
|
PowerControl7 = 0x0001;
|
|
|
|
|
2017-02-05 16:15:17 +00:00
|
|
|
ARM7BIOSProt = 0;
|
|
|
|
|
2016-12-03 15:13:04 +00:00
|
|
|
IPCSync9 = 0;
|
|
|
|
IPCSync7 = 0;
|
2017-01-17 00:58:25 +00:00
|
|
|
IPCFIFOCnt9 = 0;
|
|
|
|
IPCFIFOCnt7 = 0;
|
|
|
|
IPCFIFO9->Clear();
|
|
|
|
IPCFIFO7->Clear();
|
2016-12-03 12:42:27 +00:00
|
|
|
|
2017-01-18 01:20:45 +00:00
|
|
|
DivCnt = 0;
|
2017-01-31 20:53:45 +00:00
|
|
|
SqrtCnt = 0;
|
2017-01-18 01:20:45 +00:00
|
|
|
|
2016-11-24 17:31:49 +00:00
|
|
|
ARM9->Reset();
|
|
|
|
ARM7->Reset();
|
2016-12-03 15:13:04 +00:00
|
|
|
CP15::Reset();
|
2016-11-24 17:31:49 +00:00
|
|
|
|
2017-02-17 04:33:37 +00:00
|
|
|
CPUStop = 0;
|
|
|
|
|
2016-12-05 16:08:24 +00:00
|
|
|
memset(Timers, 0, 8*sizeof(Timer));
|
2017-04-24 23:14:26 +00:00
|
|
|
TimerCheckMask[0] = 0;
|
|
|
|
TimerCheckMask[1] = 0;
|
2016-12-05 16:08:24 +00:00
|
|
|
|
2017-01-18 00:33:06 +00:00
|
|
|
for (i = 0; i < 8; i++) DMAs[i]->Reset();
|
|
|
|
memset(DMA9Fill, 0, 4*4);
|
|
|
|
|
2017-01-31 02:54:51 +00:00
|
|
|
memset(SchedList, 0, sizeof(SchedList));
|
|
|
|
SchedListMask = 0;
|
2016-12-05 16:08:24 +00:00
|
|
|
|
2017-01-31 02:54:51 +00:00
|
|
|
CurIterationCycles = 0;
|
|
|
|
ARM7Offset = 0;
|
2016-12-05 16:08:24 +00:00
|
|
|
|
2016-12-23 20:22:22 +00:00
|
|
|
KeyInput = 0x007F03FF;
|
2017-06-04 20:34:31 +00:00
|
|
|
KeyCnt = 0;
|
|
|
|
RCnt = 0;
|
2016-12-23 20:22:22 +00:00
|
|
|
|
2017-04-06 17:44:34 +00:00
|
|
|
NDSCart::Reset();
|
|
|
|
GPU::Reset();
|
|
|
|
SPU::Reset();
|
|
|
|
SPI::Reset();
|
|
|
|
RTC::Reset();
|
|
|
|
Wifi::Reset();
|
2017-03-19 18:07:39 +00:00
|
|
|
}
|
|
|
|
|
2017-09-21 01:59:12 +00:00
|
|
|
void Stop()
|
|
|
|
{
|
|
|
|
printf("Stopping: shutdown\n");
|
2017-10-01 01:19:39 +00:00
|
|
|
Running = false;
|
2017-09-21 01:59:12 +00:00
|
|
|
Platform::StopEmu();
|
2017-09-21 02:08:03 +00:00
|
|
|
GPU::Stop();
|
2017-09-21 01:59:12 +00:00
|
|
|
SPU::Stop();
|
|
|
|
}
|
|
|
|
|
2017-10-10 22:01:05 +00:00
|
|
|
bool LoadROM(const char* path, bool direct)
|
2017-03-19 18:07:39 +00:00
|
|
|
{
|
|
|
|
if (NDSCart::LoadROM(path, direct))
|
2017-10-10 22:01:05 +00:00
|
|
|
{
|
2017-03-19 18:07:39 +00:00
|
|
|
Running = true;
|
2017-10-10 22:01:05 +00:00
|
|
|
return true;
|
|
|
|
}
|
2017-03-29 16:59:20 +00:00
|
|
|
else
|
2017-10-10 22:01:05 +00:00
|
|
|
{
|
2017-03-29 16:59:20 +00:00
|
|
|
printf("Failed to load ROM %s\n", path);
|
2017-10-10 22:01:05 +00:00
|
|
|
return false;
|
|
|
|
}
|
2017-03-29 16:59:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void LoadBIOS()
|
|
|
|
{
|
|
|
|
Reset();
|
|
|
|
Running = true;
|
2016-11-24 17:31:49 +00:00
|
|
|
}
|
|
|
|
|
2017-01-31 02:54:51 +00:00
|
|
|
|
|
|
|
void CalcIterationCycles()
|
|
|
|
{
|
|
|
|
CurIterationCycles = 16;
|
|
|
|
|
|
|
|
for (int i = 0; i < Event_MAX; i++)
|
|
|
|
{
|
|
|
|
if (!(SchedListMask & (1<<i)))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (SchedList[i].WaitCycles < CurIterationCycles)
|
|
|
|
CurIterationCycles = SchedList[i].WaitCycles;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void RunSystem(s32 cycles)
|
|
|
|
{
|
|
|
|
for (int i = 0; i < Event_MAX; i++)
|
|
|
|
{
|
|
|
|
if (!(SchedListMask & (1<<i)))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
SchedList[i].WaitCycles -= cycles;
|
2017-11-09 15:02:37 +00:00
|
|
|
|
2017-01-31 02:54:51 +00:00
|
|
|
if (SchedList[i].WaitCycles < 1)
|
|
|
|
{
|
|
|
|
SchedListMask &= ~(1<<i);
|
|
|
|
SchedList[i].Func(SchedList[i].Param);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-05-10 00:21:02 +00:00
|
|
|
u32 RunFrame()
|
2016-11-24 17:31:49 +00:00
|
|
|
{
|
2017-05-10 00:21:02 +00:00
|
|
|
if (!Running) return 263; // dorp
|
2016-12-05 22:17:03 +00:00
|
|
|
|
2017-01-18 03:03:19 +00:00
|
|
|
GPU::StartFrame();
|
2016-12-05 22:17:03 +00:00
|
|
|
|
2017-05-10 00:21:02 +00:00
|
|
|
while (Running && GPU::TotalScanlines==0)
|
2017-01-31 02:54:51 +00:00
|
|
|
{
|
2017-02-17 04:33:37 +00:00
|
|
|
s32 ndscyclestorun;
|
|
|
|
s32 ndscycles = 0;
|
2017-01-31 02:54:51 +00:00
|
|
|
|
2017-07-21 08:19:56 +00:00
|
|
|
// TODO: give it some margin, so it can directly do 17 cycles instead of 16 then 1
|
2017-02-17 04:33:37 +00:00
|
|
|
CalcIterationCycles();
|
2017-01-31 02:54:51 +00:00
|
|
|
|
2017-03-02 23:48:26 +00:00
|
|
|
if (CPUStop & 0xFFFF)
|
2017-02-17 04:33:37 +00:00
|
|
|
{
|
|
|
|
s32 cycles = CurIterationCycles;
|
|
|
|
cycles = DMAs[0]->Run(cycles);
|
|
|
|
if (cycles > 0) cycles = DMAs[1]->Run(cycles);
|
|
|
|
if (cycles > 0) cycles = DMAs[2]->Run(cycles);
|
|
|
|
if (cycles > 0) cycles = DMAs[3]->Run(cycles);
|
|
|
|
ndscyclestorun = CurIterationCycles - cycles;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ARM9->CyclesToRun = CurIterationCycles << 1;
|
|
|
|
ARM9->Execute();
|
|
|
|
ndscyclestorun = ARM9->Cycles >> 1;
|
|
|
|
}
|
2017-01-31 02:54:51 +00:00
|
|
|
|
2017-03-02 23:48:26 +00:00
|
|
|
if (CPUStop & 0xFFFF0000)
|
2017-02-17 04:33:37 +00:00
|
|
|
{
|
|
|
|
s32 cycles = ndscyclestorun - ARM7Offset;
|
|
|
|
cycles = DMAs[4]->Run(cycles);
|
|
|
|
if (cycles > 0) cycles = DMAs[5]->Run(cycles);
|
|
|
|
if (cycles > 0) cycles = DMAs[6]->Run(cycles);
|
|
|
|
if (cycles > 0) cycles = DMAs[7]->Run(cycles);
|
2017-06-12 17:00:32 +00:00
|
|
|
ARM7Offset = -cycles;
|
2017-02-17 04:33:37 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ARM7->CyclesToRun = ndscyclestorun - ARM7Offset;
|
|
|
|
ARM7->Execute();
|
|
|
|
ARM7Offset = ARM7->Cycles - ARM7->CyclesToRun;
|
|
|
|
}
|
2017-01-31 02:54:51 +00:00
|
|
|
|
|
|
|
RunSystem(ndscyclestorun);
|
2017-02-08 20:30:10 +00:00
|
|
|
//GPU3D::Run(ndscyclestorun);
|
2016-12-05 16:08:24 +00:00
|
|
|
}
|
2017-05-10 00:21:02 +00:00
|
|
|
|
|
|
|
return GPU::TotalScanlines;
|
2016-12-05 16:08:24 +00:00
|
|
|
}
|
|
|
|
|
2017-01-31 02:54:51 +00:00
|
|
|
void Reschedule()
|
|
|
|
{
|
|
|
|
CalcIterationCycles();
|
|
|
|
|
|
|
|
ARM9->CyclesToRun = CurIterationCycles << 1;
|
|
|
|
//ARM7->CyclesToRun = CurIterationCycles - ARM7Offset;
|
|
|
|
//ARM7->CyclesToRun = (ARM9->Cycles >> 1) - ARM7->Cycles - ARM7Offset;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ScheduleEvent(u32 id, bool periodic, s32 delay, void (*func)(u32), u32 param)
|
|
|
|
{
|
|
|
|
if (SchedListMask & (1<<id))
|
|
|
|
{
|
|
|
|
printf("!! EVENT %d ALREADY SCHEDULED\n", id);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
SchedEvent* evt = &SchedList[id];
|
|
|
|
|
|
|
|
if (periodic) evt->WaitCycles += delay;
|
|
|
|
else evt->WaitCycles = delay + (ARM9->Cycles >> 1);
|
|
|
|
|
|
|
|
evt->Func = func;
|
|
|
|
evt->Param = param;
|
|
|
|
|
|
|
|
SchedListMask |= (1<<id);
|
|
|
|
|
|
|
|
Reschedule();
|
|
|
|
}
|
|
|
|
|
|
|
|
void CancelEvent(u32 id)
|
|
|
|
{
|
|
|
|
SchedListMask &= ~(1<<id);
|
|
|
|
}
|
|
|
|
|
2016-11-24 17:31:49 +00:00
|
|
|
|
2016-12-23 20:22:22 +00:00
|
|
|
void PressKey(u32 key)
|
|
|
|
{
|
|
|
|
KeyInput &= ~(1 << key);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ReleaseKey(u32 key)
|
|
|
|
{
|
|
|
|
KeyInput |= (1 << key);
|
|
|
|
}
|
|
|
|
|
2017-01-31 23:24:36 +00:00
|
|
|
void TouchScreen(u16 x, u16 y)
|
|
|
|
{
|
|
|
|
SPI_TSC::SetTouchCoords(x, y);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ReleaseScreen()
|
|
|
|
{
|
|
|
|
SPI_TSC::SetTouchCoords(0x000, 0xFFF);
|
|
|
|
}
|
|
|
|
|
2016-12-23 20:22:22 +00:00
|
|
|
|
2017-09-30 18:05:56 +00:00
|
|
|
void SetKeyMask(u32 mask)
|
|
|
|
{
|
|
|
|
u32 key_lo = mask & 0x3FF;
|
|
|
|
u32 key_hi = (mask >> 10) & 0x3;
|
|
|
|
|
|
|
|
KeyInput &= 0xFFFCFC00;
|
|
|
|
KeyInput |= key_lo | (key_hi << 16);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-11-24 17:31:49 +00:00
|
|
|
void Halt()
|
|
|
|
{
|
2016-12-06 16:32:51 +00:00
|
|
|
printf("Halt()\n");
|
2016-11-24 17:31:49 +00:00
|
|
|
Running = false;
|
2016-11-03 00:38:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-12-06 16:32:51 +00:00
|
|
|
void MapSharedWRAM(u8 val)
|
2016-12-03 17:29:19 +00:00
|
|
|
{
|
2016-12-06 16:32:51 +00:00
|
|
|
WRAMCnt = val;
|
|
|
|
|
2016-12-03 17:29:19 +00:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-03-02 23:48:26 +00:00
|
|
|
void SetIRQ(u32 cpu, u32 irq)
|
2016-12-04 02:20:50 +00:00
|
|
|
{
|
2017-03-02 23:48:26 +00:00
|
|
|
IF[cpu] |= (1 << irq);
|
|
|
|
}
|
2016-12-06 16:32:51 +00:00
|
|
|
|
2017-03-02 23:48:26 +00:00
|
|
|
void ClearIRQ(u32 cpu, u32 irq)
|
|
|
|
{
|
|
|
|
IF[cpu] &= ~(1 << irq);
|
2016-12-04 02:20:50 +00:00
|
|
|
}
|
|
|
|
|
2016-12-06 16:32:51 +00:00
|
|
|
bool HaltInterrupted(u32 cpu)
|
|
|
|
{
|
|
|
|
if (cpu == 0)
|
|
|
|
{
|
|
|
|
if (!(IME[0] & 0x1))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (IF[cpu] & IE[cpu])
|
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-03-02 23:48:26 +00:00
|
|
|
void StopCPU(u32 cpu, u32 mask)
|
|
|
|
{
|
2017-04-12 15:53:15 +00:00
|
|
|
if (cpu)
|
|
|
|
{
|
|
|
|
CPUStop |= (mask << 16);
|
2017-04-13 02:16:57 +00:00
|
|
|
ARM7->Halt(2);
|
2017-04-12 15:53:15 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
CPUStop |= mask;
|
2017-04-13 02:16:57 +00:00
|
|
|
ARM9->Halt(2);
|
2017-04-12 15:53:15 +00:00
|
|
|
}
|
2017-03-02 23:48:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ResumeCPU(u32 cpu, u32 mask)
|
2017-02-17 04:33:37 +00:00
|
|
|
{
|
2017-03-02 23:48:26 +00:00
|
|
|
if (cpu) mask <<= 16;
|
|
|
|
CPUStop &= ~mask;
|
2017-02-17 04:33:37 +00:00
|
|
|
}
|
|
|
|
|
2017-06-25 15:35:45 +00:00
|
|
|
u32 GetPC(u32 cpu)
|
|
|
|
{
|
|
|
|
return cpu ? ARM7->R[15] : ARM9->R[15];
|
|
|
|
}
|
|
|
|
|
2016-12-04 02:20:50 +00:00
|
|
|
|
2016-12-03 00:31:33 +00:00
|
|
|
|
2017-03-20 16:39:42 +00:00
|
|
|
void HandleTimerOverflow(u32 tid)
|
|
|
|
{
|
|
|
|
Timer* timer = &Timers[tid];
|
|
|
|
|
|
|
|
timer->Counter += timer->Reload << 16;
|
|
|
|
if (timer->Cnt & (1<<6))
|
|
|
|
SetIRQ(tid >> 2, IRQ_Timer0 + (tid & 0x3));
|
|
|
|
|
|
|
|
if ((tid & 0x3) == 3)
|
|
|
|
return;
|
|
|
|
|
|
|
|
for (;;)
|
|
|
|
{
|
|
|
|
tid++;
|
|
|
|
|
|
|
|
timer = &Timers[tid];
|
|
|
|
|
|
|
|
if ((timer->Cnt & 0x84) != 0x84)
|
|
|
|
break;
|
|
|
|
|
|
|
|
timer->Counter += 0x10000;
|
|
|
|
if (timer->Counter >> 16)
|
|
|
|
break;
|
|
|
|
|
|
|
|
timer->Counter = timer->Reload << 16;
|
|
|
|
if (timer->Cnt & (1<<6))
|
|
|
|
SetIRQ(tid >> 2, IRQ_Timer0 + (tid & 0x3));
|
|
|
|
|
|
|
|
if ((tid & 0x3) == 3)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void RunTimer(u32 tid, s32 cycles)
|
|
|
|
{
|
|
|
|
Timer* timer = &Timers[tid];
|
2017-04-24 23:14:26 +00:00
|
|
|
//if ((timer->Cnt & 0x84) != 0x80)
|
|
|
|
// return;
|
2017-03-20 16:39:42 +00:00
|
|
|
|
|
|
|
u32 oldcount = timer->Counter;
|
|
|
|
timer->Counter += (cycles << timer->CycleShift);
|
|
|
|
if (timer->Counter < oldcount)
|
|
|
|
HandleTimerOverflow(tid);
|
|
|
|
}
|
|
|
|
|
|
|
|
void RunTimingCriticalDevices(u32 cpu, s32 cycles)
|
|
|
|
{
|
2017-04-24 23:14:26 +00:00
|
|
|
register u32 timermask = TimerCheckMask[cpu];
|
|
|
|
|
|
|
|
if (timermask & 0x1) RunTimer((cpu<<2)+0, cycles);
|
|
|
|
if (timermask & 0x2) RunTimer((cpu<<2)+1, cycles);
|
|
|
|
if (timermask & 0x4) RunTimer((cpu<<2)+2, cycles);
|
|
|
|
if (timermask & 0x8) RunTimer((cpu<<2)+3, cycles);
|
2017-03-20 16:39:42 +00:00
|
|
|
|
|
|
|
if (cpu == 0)
|
|
|
|
{
|
|
|
|
GPU3D::Run(cycles);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2017-06-26 09:02:10 +00:00
|
|
|
bool DMAsInMode(u32 cpu, u32 mode)
|
|
|
|
{
|
|
|
|
cpu <<= 2;
|
|
|
|
if (DMAs[cpu+0]->IsInMode(mode)) return true;
|
|
|
|
if (DMAs[cpu+1]->IsInMode(mode)) return true;
|
|
|
|
if (DMAs[cpu+2]->IsInMode(mode)) return true;
|
|
|
|
if (DMAs[cpu+3]->IsInMode(mode)) return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-01-22 19:34:59 +00:00
|
|
|
void CheckDMAs(u32 cpu, u32 mode)
|
|
|
|
{
|
|
|
|
cpu <<= 2;
|
|
|
|
DMAs[cpu+0]->StartIfNeeded(mode);
|
|
|
|
DMAs[cpu+1]->StartIfNeeded(mode);
|
|
|
|
DMAs[cpu+2]->StartIfNeeded(mode);
|
|
|
|
DMAs[cpu+3]->StartIfNeeded(mode);
|
|
|
|
}
|
|
|
|
|
2017-03-20 21:18:35 +00:00
|
|
|
void StopDMAs(u32 cpu, u32 mode)
|
|
|
|
{
|
|
|
|
cpu <<= 2;
|
|
|
|
DMAs[cpu+0]->StopIfNeeded(mode);
|
|
|
|
DMAs[cpu+1]->StopIfNeeded(mode);
|
|
|
|
DMAs[cpu+2]->StopIfNeeded(mode);
|
|
|
|
DMAs[cpu+3]->StopIfNeeded(mode);
|
|
|
|
}
|
|
|
|
|
2017-01-22 19:34:59 +00:00
|
|
|
|
|
|
|
|
2017-03-20 16:39:42 +00:00
|
|
|
|
2017-01-31 02:54:51 +00:00
|
|
|
const s32 TimerPrescaler[4] = {0, 6, 8, 10};
|
2016-12-05 16:08:24 +00:00
|
|
|
|
2017-01-31 02:54:51 +00:00
|
|
|
u16 TimerGetCounter(u32 timer)
|
|
|
|
{
|
|
|
|
u32 ret = Timers[timer].Counter;
|
|
|
|
|
|
|
|
return ret >> 16;
|
|
|
|
}
|
|
|
|
|
2016-12-05 16:08:24 +00:00
|
|
|
void TimerStart(u32 id, u16 cnt)
|
|
|
|
{
|
|
|
|
Timer* timer = &Timers[id];
|
2017-01-31 02:54:51 +00:00
|
|
|
u16 curstart = timer->Cnt & (1<<7);
|
2016-12-05 16:08:24 +00:00
|
|
|
u16 newstart = cnt & (1<<7);
|
|
|
|
|
2017-01-31 02:54:51 +00:00
|
|
|
timer->Cnt = cnt;
|
2017-03-20 16:39:42 +00:00
|
|
|
timer->CycleShift = 16 - TimerPrescaler[cnt & 0x03];
|
2016-12-05 16:08:24 +00:00
|
|
|
|
|
|
|
if ((!curstart) && newstart)
|
|
|
|
{
|
2017-01-31 02:54:51 +00:00
|
|
|
timer->Counter = timer->Reload << 16;
|
2016-12-05 16:08:24 +00:00
|
|
|
}
|
2017-04-24 23:14:26 +00:00
|
|
|
|
|
|
|
if ((cnt & 0x84) == 0x80)
|
|
|
|
TimerCheckMask[id>>2] |= (1<<(id&0x3));
|
|
|
|
else
|
|
|
|
TimerCheckMask[id>>2] &= ~(1<<(id&0x3));
|
2016-12-05 16:08:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2017-07-15 17:46:27 +00:00
|
|
|
void DivDone(u32 param)
|
2017-01-18 01:20:45 +00:00
|
|
|
{
|
2017-07-15 17:46:27 +00:00
|
|
|
DivCnt &= ~0xC000;
|
2017-01-18 01:20:45 +00:00
|
|
|
|
|
|
|
switch (DivCnt & 0x0003)
|
|
|
|
{
|
|
|
|
case 0x0000:
|
|
|
|
{
|
|
|
|
s32 num = (s32)DivNumerator[0];
|
|
|
|
s32 den = (s32)DivDenominator[0];
|
|
|
|
if (den == 0)
|
|
|
|
{
|
|
|
|
DivQuotient[0] = (num<0) ? 1:-1;
|
|
|
|
DivQuotient[1] = (num<0) ? -1:1;
|
|
|
|
*(s64*)&DivRemainder[0] = num;
|
|
|
|
}
|
|
|
|
else if (num == -0x80000000 && den == -1)
|
|
|
|
{
|
|
|
|
*(s64*)&DivQuotient[0] = 0x80000000;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
*(s64*)&DivQuotient[0] = (s64)(num / den);
|
|
|
|
*(s64*)&DivRemainder[0] = (s64)(num % den);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x0001:
|
|
|
|
case 0x0003:
|
|
|
|
{
|
|
|
|
s64 num = *(s64*)&DivNumerator[0];
|
|
|
|
s32 den = (s32)DivDenominator[0];
|
|
|
|
if (den == 0)
|
|
|
|
{
|
|
|
|
*(s64*)&DivQuotient[0] = (num<0) ? 1:-1;
|
|
|
|
*(s64*)&DivRemainder[0] = num;
|
|
|
|
}
|
|
|
|
else if (num == -0x8000000000000000 && den == -1)
|
|
|
|
{
|
|
|
|
*(s64*)&DivQuotient[0] = 0x8000000000000000;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
*(s64*)&DivQuotient[0] = (s64)(num / den);
|
|
|
|
*(s64*)&DivRemainder[0] = (s64)(num % den);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x0002:
|
|
|
|
{
|
|
|
|
s64 num = *(s64*)&DivNumerator[0];
|
|
|
|
s64 den = *(s64*)&DivDenominator[0];
|
|
|
|
if (den == 0)
|
|
|
|
{
|
|
|
|
*(s64*)&DivQuotient[0] = (num<0) ? 1:-1;
|
|
|
|
*(s64*)&DivRemainder[0] = num;
|
|
|
|
}
|
|
|
|
else if (num == -0x8000000000000000 && den == -1)
|
|
|
|
{
|
|
|
|
*(s64*)&DivQuotient[0] = 0x8000000000000000;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
*(s64*)&DivQuotient[0] = (s64)(num / den);
|
|
|
|
*(s64*)&DivRemainder[0] = (s64)(num % den);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((DivDenominator[0] | DivDenominator[1]) == 0)
|
2017-07-15 17:46:27 +00:00
|
|
|
DivCnt |= 0x4000;
|
2017-01-18 01:20:45 +00:00
|
|
|
}
|
|
|
|
|
2017-07-15 17:46:27 +00:00
|
|
|
void StartDiv()
|
2017-01-31 20:53:45 +00:00
|
|
|
{
|
2017-07-15 17:46:27 +00:00
|
|
|
NDS::CancelEvent(NDS::Event_Div);
|
|
|
|
DivCnt |= 0x8000;
|
|
|
|
NDS::ScheduleEvent(NDS::Event_Div, false, ((DivCnt&0x3)==0) ? 18:34, DivDone, 0);
|
|
|
|
}
|
2017-01-31 20:53:45 +00:00
|
|
|
|
2017-07-15 17:46:27 +00:00
|
|
|
// http://stackoverflow.com/questions/1100090/looking-for-an-efficient-integer-square-root-algorithm-for-arm-thumb2
|
|
|
|
void SqrtDone(u32 param)
|
|
|
|
{
|
2017-01-31 20:53:45 +00:00
|
|
|
u64 val;
|
|
|
|
u32 res = 0;
|
|
|
|
u64 rem = 0;
|
|
|
|
u32 prod = 0;
|
|
|
|
u32 nbits, topshift;
|
|
|
|
|
2017-07-15 17:46:27 +00:00
|
|
|
SqrtCnt &= ~0x8000;
|
|
|
|
|
2017-01-31 20:53:45 +00:00
|
|
|
if (SqrtCnt & 0x0001)
|
|
|
|
{
|
|
|
|
val = *(u64*)&SqrtVal[0];
|
|
|
|
nbits = 32;
|
|
|
|
topshift = 62;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
val = (u64)SqrtVal[0]; // 32bit
|
|
|
|
nbits = 16;
|
|
|
|
topshift = 30;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (u32 i = 0; i < nbits; i++)
|
|
|
|
{
|
|
|
|
rem = (rem << 2) + ((val >> topshift) & 0x3);
|
|
|
|
val <<= 2;
|
|
|
|
res <<= 1;
|
|
|
|
|
|
|
|
prod = (res << 1) + 1;
|
|
|
|
if (rem >= prod)
|
|
|
|
{
|
|
|
|
rem -= prod;
|
|
|
|
res++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
SqrtRes = res;
|
|
|
|
}
|
|
|
|
|
2017-07-15 17:46:27 +00:00
|
|
|
void StartSqrt()
|
|
|
|
{
|
|
|
|
NDS::CancelEvent(NDS::Event_Sqrt);
|
|
|
|
SqrtCnt |= 0x8000;
|
|
|
|
NDS::ScheduleEvent(NDS::Event_Sqrt, false, 13, SqrtDone, 0);
|
|
|
|
}
|
|
|
|
|
2017-01-18 01:20:45 +00:00
|
|
|
|
|
|
|
|
2016-12-06 16:32:51 +00:00
|
|
|
void debug(u32 param)
|
|
|
|
{
|
2017-02-03 15:57:31 +00:00
|
|
|
printf("ARM9 PC=%08X LR=%08X %08X\n", ARM9->R[15], ARM9->R[14], ARM9->R_IRQ[1]);
|
|
|
|
printf("ARM7 PC=%08X LR=%08X %08X\n", ARM7->R[15], ARM7->R[14], ARM7->R_IRQ[1]);
|
2017-02-27 23:50:54 +00:00
|
|
|
|
2017-04-10 16:47:11 +00:00
|
|
|
printf("ARM9 IME=%08X IE=%08X IF=%08X\n", IME[0], IE[0], IF[0]);
|
|
|
|
printf("ARM7 IME=%08X IE=%08X IF=%08X\n", IME[1], IE[1], IF[1]);
|
|
|
|
|
|
|
|
//for (int i = 0; i < 9; i++)
|
|
|
|
// printf("VRAM %c: %02X\n", 'A'+i, GPU::VRAMCNT[i]);
|
2017-04-26 14:17:03 +00:00
|
|
|
|
2017-06-12 17:00:32 +00:00
|
|
|
/*FILE*
|
|
|
|
shit = fopen("debug/pictochat.bin", "wb");
|
|
|
|
for (u32 i = 0x02000000; i < 0x02400000; i+=4)
|
2017-04-26 14:17:03 +00:00
|
|
|
{
|
|
|
|
u32 val = ARM7Read32(i);
|
|
|
|
fwrite(&val, 4, 1, shit);
|
|
|
|
}
|
2017-05-09 01:54:37 +00:00
|
|
|
for (u32 i = 0x037F0000; i < 0x03810000; i+=4)
|
|
|
|
{
|
|
|
|
u32 val = ARM7Read32(i);
|
|
|
|
fwrite(&val, 4, 1, shit);
|
|
|
|
}
|
|
|
|
fclose(shit);*/
|
2016-12-06 16:32:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2016-12-03 00:31:33 +00:00
|
|
|
u8 ARM9Read8(u32 addr)
|
|
|
|
{
|
|
|
|
if ((addr & 0xFFFFF000) == 0xFFFF0000)
|
|
|
|
{
|
|
|
|
return *(u8*)&ARM9BIOS[addr & 0xFFF];
|
|
|
|
}
|
|
|
|
|
2016-12-03 12:42:27 +00:00
|
|
|
switch (addr & 0xFF000000)
|
|
|
|
{
|
|
|
|
case 0x02000000:
|
|
|
|
return *(u8*)&MainRAM[addr & 0x3FFFFF];
|
2016-12-03 17:29:19 +00:00
|
|
|
|
|
|
|
case 0x03000000:
|
|
|
|
if (SWRAM_ARM9) return *(u8*)&SWRAM_ARM9[addr & SWRAM_ARM9Mask];
|
|
|
|
else return 0;
|
|
|
|
|
|
|
|
case 0x04000000:
|
2017-01-17 01:29:25 +00:00
|
|
|
return ARM9IORead8(addr);
|
2016-12-05 22:17:03 +00:00
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
case 0x05000000:
|
2017-01-18 03:03:19 +00:00
|
|
|
return *(u8*)&GPU::Palette[addr & 0x7FF];
|
2016-12-06 16:32:51 +00:00
|
|
|
|
2016-12-05 22:17:03 +00:00
|
|
|
case 0x06000000:
|
|
|
|
{
|
|
|
|
switch (addr & 0x00E00000)
|
|
|
|
{
|
2017-02-27 20:26:11 +00:00
|
|
|
case 0x00000000: return GPU::ReadVRAM_ABG<u8>(addr);
|
|
|
|
case 0x00200000: return GPU::ReadVRAM_BBG<u8>(addr);
|
|
|
|
case 0x00400000: return GPU::ReadVRAM_AOBJ<u8>(addr);
|
|
|
|
case 0x00600000: return GPU::ReadVRAM_BOBJ<u8>(addr);
|
|
|
|
default: return GPU::ReadVRAM_LCDC<u8>(addr);
|
2016-12-05 22:17:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
2016-12-06 16:32:51 +00:00
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
case 0x07000000:
|
2017-01-18 03:03:19 +00:00
|
|
|
return *(u8*)&GPU::OAM[addr & 0x7FF];
|
2017-01-18 16:57:12 +00:00
|
|
|
|
|
|
|
case 0x08000000:
|
|
|
|
case 0x09000000:
|
2017-11-09 15:02:37 +00:00
|
|
|
printf("GBA read8 %08X\n", addr);
|
2017-01-18 16:57:12 +00:00
|
|
|
return 0xFF;
|
2016-12-03 12:42:27 +00:00
|
|
|
}
|
|
|
|
|
2016-12-03 00:31:33 +00:00
|
|
|
printf("unknown arm9 read8 %08X\n", addr);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
u16 ARM9Read16(u32 addr)
|
2016-11-03 00:38:58 +00:00
|
|
|
{
|
2016-12-03 00:31:33 +00:00
|
|
|
if ((addr & 0xFFFFF000) == 0xFFFF0000)
|
|
|
|
{
|
|
|
|
return *(u16*)&ARM9BIOS[addr & 0xFFF];
|
|
|
|
}
|
|
|
|
|
2016-12-03 12:42:27 +00:00
|
|
|
switch (addr & 0xFF000000)
|
|
|
|
{
|
|
|
|
case 0x02000000:
|
|
|
|
return *(u16*)&MainRAM[addr & 0x3FFFFF];
|
2016-12-03 15:13:04 +00:00
|
|
|
|
2016-12-03 17:29:19 +00:00
|
|
|
case 0x03000000:
|
|
|
|
if (SWRAM_ARM9) return *(u16*)&SWRAM_ARM9[addr & SWRAM_ARM9Mask];
|
|
|
|
else return 0;
|
|
|
|
|
2016-12-03 15:13:04 +00:00
|
|
|
case 0x04000000:
|
2017-01-17 01:29:25 +00:00
|
|
|
return ARM9IORead16(addr);
|
2016-12-06 16:32:51 +00:00
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
case 0x05000000:
|
2017-01-18 03:03:19 +00:00
|
|
|
return *(u16*)&GPU::Palette[addr & 0x7FF];
|
2016-12-06 16:32:51 +00:00
|
|
|
|
2016-12-05 22:17:03 +00:00
|
|
|
case 0x06000000:
|
|
|
|
{
|
|
|
|
switch (addr & 0x00E00000)
|
|
|
|
{
|
2017-02-27 20:26:11 +00:00
|
|
|
case 0x00000000: return GPU::ReadVRAM_ABG<u16>(addr);
|
|
|
|
case 0x00200000: return GPU::ReadVRAM_BBG<u16>(addr);
|
|
|
|
case 0x00400000: return GPU::ReadVRAM_AOBJ<u16>(addr);
|
|
|
|
case 0x00600000: return GPU::ReadVRAM_BOBJ<u16>(addr);
|
|
|
|
default: return GPU::ReadVRAM_LCDC<u16>(addr);
|
2016-12-05 22:17:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
2016-12-06 16:32:51 +00:00
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
case 0x07000000:
|
2017-01-18 03:03:19 +00:00
|
|
|
return *(u16*)&GPU::OAM[addr & 0x7FF];
|
2017-01-18 16:57:12 +00:00
|
|
|
|
|
|
|
case 0x08000000:
|
|
|
|
case 0x09000000:
|
2017-11-09 15:02:37 +00:00
|
|
|
printf("GBA read16 %08X\n", addr);
|
2017-01-18 16:57:12 +00:00
|
|
|
return 0xFFFF;
|
2016-12-03 12:42:27 +00:00
|
|
|
}
|
|
|
|
|
2017-04-27 16:45:43 +00:00
|
|
|
//printf("unknown arm9 read16 %08X %08X\n", addr, ARM9->R[15]);
|
2016-12-03 00:31:33 +00:00
|
|
|
return 0;
|
|
|
|
}
|
2016-11-03 00:38:58 +00:00
|
|
|
|
2016-12-03 00:31:33 +00:00
|
|
|
u32 ARM9Read32(u32 addr)
|
|
|
|
{
|
2016-11-03 00:38:58 +00:00
|
|
|
if ((addr & 0xFFFFF000) == 0xFFFF0000)
|
|
|
|
{
|
|
|
|
return *(u32*)&ARM9BIOS[addr & 0xFFF];
|
|
|
|
}
|
2016-12-05 16:08:24 +00:00
|
|
|
|
2016-12-03 12:42:27 +00:00
|
|
|
switch (addr & 0xFF000000)
|
|
|
|
{
|
|
|
|
case 0x02000000:
|
|
|
|
return *(u32*)&MainRAM[addr & 0x3FFFFF];
|
2016-12-03 17:29:19 +00:00
|
|
|
|
|
|
|
case 0x03000000:
|
|
|
|
if (SWRAM_ARM9) return *(u32*)&SWRAM_ARM9[addr & SWRAM_ARM9Mask];
|
|
|
|
else return 0;
|
2016-12-04 02:20:50 +00:00
|
|
|
|
|
|
|
case 0x04000000:
|
2017-01-17 01:29:25 +00:00
|
|
|
return ARM9IORead32(addr);
|
2016-12-05 16:08:24 +00:00
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
case 0x05000000:
|
2017-01-18 03:03:19 +00:00
|
|
|
return *(u32*)&GPU::Palette[addr & 0x7FF];
|
2016-12-06 16:32:51 +00:00
|
|
|
|
2016-12-05 22:17:03 +00:00
|
|
|
case 0x06000000:
|
|
|
|
{
|
|
|
|
switch (addr & 0x00E00000)
|
|
|
|
{
|
2017-02-27 20:26:11 +00:00
|
|
|
case 0x00000000: return GPU::ReadVRAM_ABG<u32>(addr);
|
|
|
|
case 0x00200000: return GPU::ReadVRAM_BBG<u32>(addr);
|
|
|
|
case 0x00400000: return GPU::ReadVRAM_AOBJ<u32>(addr);
|
|
|
|
case 0x00600000: return GPU::ReadVRAM_BOBJ<u32>(addr);
|
|
|
|
default: return GPU::ReadVRAM_LCDC<u32>(addr);
|
2016-12-05 22:17:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
2016-12-06 16:32:51 +00:00
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
case 0x07000000:
|
2017-01-18 03:03:19 +00:00
|
|
|
return *(u32*)&GPU::OAM[addr & 0x7FF];
|
2017-01-18 16:57:12 +00:00
|
|
|
|
|
|
|
case 0x08000000:
|
|
|
|
case 0x09000000:
|
2017-11-09 15:02:37 +00:00
|
|
|
printf("GBA read32 %08X\n", addr);
|
2017-01-18 16:57:12 +00:00
|
|
|
return 0xFFFFFFFF;
|
2016-12-03 12:42:27 +00:00
|
|
|
}
|
|
|
|
|
2016-12-05 16:08:24 +00:00
|
|
|
printf("unknown arm9 read32 %08X | %08X %08X %08X\n", addr, ARM9->R[15], ARM9->R[12], ARM9Read32(0x027FF820));
|
2016-11-03 00:38:58 +00:00
|
|
|
return 0;
|
2016-05-16 15:48:40 +00:00
|
|
|
}
|
|
|
|
|
2016-12-03 00:31:33 +00:00
|
|
|
void ARM9Write8(u32 addr, u8 val)
|
|
|
|
{
|
2016-12-03 12:42:27 +00:00
|
|
|
switch (addr & 0xFF000000)
|
|
|
|
{
|
|
|
|
case 0x02000000:
|
|
|
|
*(u8*)&MainRAM[addr & 0x3FFFFF] = val;
|
|
|
|
return;
|
2016-12-03 17:29:19 +00:00
|
|
|
|
|
|
|
case 0x03000000:
|
|
|
|
if (SWRAM_ARM9) *(u8*)&SWRAM_ARM9[addr & SWRAM_ARM9Mask] = val;
|
|
|
|
return;
|
|
|
|
|
|
|
|
case 0x04000000:
|
2017-01-17 01:29:25 +00:00
|
|
|
ARM9IOWrite8(addr, val);
|
|
|
|
return;
|
2016-12-06 16:32:51 +00:00
|
|
|
|
|
|
|
case 0x05000000:
|
|
|
|
case 0x06000000:
|
|
|
|
case 0x07000000:
|
|
|
|
return;
|
2016-12-03 12:42:27 +00:00
|
|
|
}
|
|
|
|
|
2016-12-03 00:31:33 +00:00
|
|
|
printf("unknown arm9 write8 %08X %02X\n", addr, val);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ARM9Write16(u32 addr, u16 val)
|
|
|
|
{
|
2016-12-03 12:42:27 +00:00
|
|
|
switch (addr & 0xFF000000)
|
|
|
|
{
|
|
|
|
case 0x02000000:
|
|
|
|
*(u16*)&MainRAM[addr & 0x3FFFFF] = val;
|
|
|
|
return;
|
2016-12-03 15:13:04 +00:00
|
|
|
|
2016-12-03 17:29:19 +00:00
|
|
|
case 0x03000000:
|
|
|
|
if (SWRAM_ARM9) *(u16*)&SWRAM_ARM9[addr & SWRAM_ARM9Mask] = val;
|
|
|
|
return;
|
|
|
|
|
2016-12-03 15:13:04 +00:00
|
|
|
case 0x04000000:
|
2017-01-17 01:29:25 +00:00
|
|
|
ARM9IOWrite16(addr, val);
|
|
|
|
return;
|
2016-12-05 22:17:03 +00:00
|
|
|
|
2016-12-06 16:32:51 +00:00
|
|
|
case 0x05000000:
|
2017-01-18 03:03:19 +00:00
|
|
|
*(u16*)&GPU::Palette[addr & 0x7FF] = val;
|
2016-12-06 16:32:51 +00:00
|
|
|
return;
|
|
|
|
|
2016-12-05 22:17:03 +00:00
|
|
|
case 0x06000000:
|
2017-02-27 20:26:11 +00:00
|
|
|
switch (addr & 0x00E00000)
|
2016-12-05 22:17:03 +00:00
|
|
|
{
|
2017-02-27 20:26:11 +00:00
|
|
|
case 0x00000000: GPU::WriteVRAM_ABG<u16>(addr, val); break;
|
|
|
|
case 0x00200000: GPU::WriteVRAM_BBG<u16>(addr, val); break;
|
|
|
|
case 0x00400000: GPU::WriteVRAM_AOBJ<u16>(addr, val); break;
|
|
|
|
case 0x00600000: GPU::WriteVRAM_BOBJ<u16>(addr, val); break;
|
|
|
|
default: GPU::WriteVRAM_LCDC<u16>(addr, val); break;
|
2016-12-05 22:17:03 +00:00
|
|
|
}
|
|
|
|
return;
|
2016-12-06 16:32:51 +00:00
|
|
|
|
|
|
|
case 0x07000000:
|
2017-01-18 03:03:19 +00:00
|
|
|
*(u16*)&GPU::OAM[addr & 0x7FF] = val;
|
2016-12-06 16:32:51 +00:00
|
|
|
return;
|
2016-12-03 12:42:27 +00:00
|
|
|
}
|
|
|
|
|
2017-11-09 15:02:37 +00:00
|
|
|
printf("unknown arm9 write16 %08X %04X\n", addr, val);
|
2016-12-03 00:31:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ARM9Write32(u32 addr, u32 val)
|
|
|
|
{
|
2016-12-03 12:42:27 +00:00
|
|
|
switch (addr & 0xFF000000)
|
|
|
|
{
|
|
|
|
case 0x02000000:
|
|
|
|
*(u32*)&MainRAM[addr & 0x3FFFFF] = val;
|
|
|
|
return;
|
2016-12-03 17:29:19 +00:00
|
|
|
|
|
|
|
case 0x03000000:
|
|
|
|
if (SWRAM_ARM9) *(u32*)&SWRAM_ARM9[addr & SWRAM_ARM9Mask] = val;
|
|
|
|
return;
|
2016-12-04 02:20:50 +00:00
|
|
|
|
|
|
|
case 0x04000000:
|
2017-01-17 01:29:25 +00:00
|
|
|
ARM9IOWrite32(addr, val);
|
|
|
|
return;
|
2016-12-05 22:17:03 +00:00
|
|
|
|
2016-12-06 16:32:51 +00:00
|
|
|
case 0x05000000:
|
2017-01-18 03:03:19 +00:00
|
|
|
*(u32*)&GPU::Palette[addr & 0x7FF] = val;
|
2016-12-06 16:32:51 +00:00
|
|
|
return;
|
|
|
|
|
2016-12-05 22:17:03 +00:00
|
|
|
case 0x06000000:
|
2017-02-27 20:26:11 +00:00
|
|
|
switch (addr & 0x00E00000)
|
2016-12-05 22:17:03 +00:00
|
|
|
{
|
2017-02-27 20:26:11 +00:00
|
|
|
case 0x00000000: GPU::WriteVRAM_ABG<u32>(addr, val); break;
|
|
|
|
case 0x00200000: GPU::WriteVRAM_BBG<u32>(addr, val); break;
|
|
|
|
case 0x00400000: GPU::WriteVRAM_AOBJ<u32>(addr, val); break;
|
|
|
|
case 0x00600000: GPU::WriteVRAM_BOBJ<u32>(addr, val); break;
|
|
|
|
default: GPU::WriteVRAM_LCDC<u32>(addr, val); break;
|
2016-12-05 22:17:03 +00:00
|
|
|
}
|
|
|
|
return;
|
2016-12-06 16:32:51 +00:00
|
|
|
|
|
|
|
case 0x07000000:
|
2017-01-18 03:03:19 +00:00
|
|
|
*(u32*)&GPU::OAM[addr & 0x7FF] = val;
|
2016-12-06 16:32:51 +00:00
|
|
|
return;
|
2016-12-03 12:42:27 +00:00
|
|
|
}
|
|
|
|
|
2016-12-03 15:13:04 +00:00
|
|
|
printf("unknown arm9 write32 %08X %08X | %08X\n", addr, val, ARM9->R[15]);
|
2016-12-03 00:31:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
u8 ARM7Read8(u32 addr)
|
|
|
|
{
|
|
|
|
if (addr < 0x00004000)
|
|
|
|
{
|
2017-02-05 16:15:17 +00:00
|
|
|
if (ARM7->R[15] >= 0x4000)
|
|
|
|
return 0xFF;
|
|
|
|
if (addr < ARM7BIOSProt && ARM7->R[15] >= ARM7BIOSProt)
|
|
|
|
return 0xFF;
|
|
|
|
|
2016-12-03 00:31:33 +00:00
|
|
|
return *(u8*)&ARM7BIOS[addr];
|
|
|
|
}
|
|
|
|
|
2016-12-03 12:42:27 +00:00
|
|
|
switch (addr & 0xFF800000)
|
|
|
|
{
|
|
|
|
case 0x02000000:
|
2017-02-01 20:35:00 +00:00
|
|
|
case 0x02800000:
|
2016-12-03 12:42:27 +00:00
|
|
|
return *(u8*)&MainRAM[addr & 0x3FFFFF];
|
|
|
|
|
2016-12-03 17:29:19 +00:00
|
|
|
case 0x03000000:
|
|
|
|
if (SWRAM_ARM7) return *(u8*)&SWRAM_ARM7[addr & SWRAM_ARM7Mask];
|
|
|
|
else return *(u8*)&ARM7WRAM[addr & 0xFFFF];
|
|
|
|
|
2016-12-03 12:42:27 +00:00
|
|
|
case 0x03800000:
|
|
|
|
return *(u8*)&ARM7WRAM[addr & 0xFFFF];
|
2016-12-03 17:29:19 +00:00
|
|
|
|
|
|
|
case 0x04000000:
|
2017-01-17 01:29:25 +00:00
|
|
|
return ARM7IORead8(addr);
|
2016-12-05 22:17:03 +00:00
|
|
|
|
|
|
|
case 0x06000000:
|
|
|
|
case 0x06800000:
|
2017-02-27 20:26:11 +00:00
|
|
|
return GPU::ReadVRAM_ARM7<u8>(addr);
|
2016-12-03 12:42:27 +00:00
|
|
|
}
|
|
|
|
|
2017-01-18 00:33:06 +00:00
|
|
|
printf("unknown arm7 read8 %08X %08X %08X/%08X\n", addr, ARM7->R[15], ARM7->R[0], ARM7->R[1]);
|
2016-12-03 00:31:33 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
u16 ARM7Read16(u32 addr)
|
|
|
|
{
|
|
|
|
if (addr < 0x00004000)
|
|
|
|
{
|
2017-02-05 16:15:17 +00:00
|
|
|
if (ARM7->R[15] >= 0x4000)
|
|
|
|
return 0xFFFF;
|
|
|
|
if (addr < ARM7BIOSProt && ARM7->R[15] >= ARM7BIOSProt)
|
|
|
|
return 0xFFFF;
|
|
|
|
|
2016-12-03 00:31:33 +00:00
|
|
|
return *(u16*)&ARM7BIOS[addr];
|
|
|
|
}
|
|
|
|
|
2016-12-03 12:42:27 +00:00
|
|
|
switch (addr & 0xFF800000)
|
|
|
|
{
|
|
|
|
case 0x02000000:
|
2017-02-01 20:35:00 +00:00
|
|
|
case 0x02800000:
|
2016-12-03 12:42:27 +00:00
|
|
|
return *(u16*)&MainRAM[addr & 0x3FFFFF];
|
|
|
|
|
2016-12-03 17:29:19 +00:00
|
|
|
case 0x03000000:
|
|
|
|
if (SWRAM_ARM7) return *(u16*)&SWRAM_ARM7[addr & SWRAM_ARM7Mask];
|
|
|
|
else return *(u16*)&ARM7WRAM[addr & 0xFFFF];
|
|
|
|
|
2016-12-03 12:42:27 +00:00
|
|
|
case 0x03800000:
|
|
|
|
return *(u16*)&ARM7WRAM[addr & 0xFFFF];
|
2016-12-03 15:13:04 +00:00
|
|
|
|
|
|
|
case 0x04000000:
|
2017-01-17 01:29:25 +00:00
|
|
|
return ARM7IORead16(addr);
|
2016-12-05 22:17:03 +00:00
|
|
|
|
|
|
|
case 0x04800000:
|
2016-12-06 16:32:51 +00:00
|
|
|
return Wifi::Read(addr);
|
2016-12-05 22:17:03 +00:00
|
|
|
|
|
|
|
case 0x06000000:
|
|
|
|
case 0x06800000:
|
2017-02-27 20:26:11 +00:00
|
|
|
return GPU::ReadVRAM_ARM7<u16>(addr);
|
2016-12-03 12:42:27 +00:00
|
|
|
}
|
|
|
|
|
2016-12-05 16:08:24 +00:00
|
|
|
printf("unknown arm7 read16 %08X %08X\n", addr, ARM7->R[15]);
|
2016-12-03 00:31:33 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-11-03 00:38:58 +00:00
|
|
|
u32 ARM7Read32(u32 addr)
|
|
|
|
{
|
|
|
|
if (addr < 0x00004000)
|
|
|
|
{
|
2017-02-05 16:15:17 +00:00
|
|
|
if (ARM7->R[15] >= 0x4000)
|
|
|
|
return 0xFFFFFFFF;
|
|
|
|
if (addr < ARM7BIOSProt && ARM7->R[15] >= ARM7BIOSProt)
|
|
|
|
return 0xFFFFFFFF;
|
|
|
|
|
2016-11-03 00:38:58 +00:00
|
|
|
return *(u32*)&ARM7BIOS[addr];
|
|
|
|
}
|
|
|
|
|
2016-12-03 12:42:27 +00:00
|
|
|
switch (addr & 0xFF800000)
|
|
|
|
{
|
|
|
|
case 0x02000000:
|
2017-02-01 20:35:00 +00:00
|
|
|
case 0x02800000:
|
2016-12-03 12:42:27 +00:00
|
|
|
return *(u32*)&MainRAM[addr & 0x3FFFFF];
|
|
|
|
|
2016-12-03 17:29:19 +00:00
|
|
|
case 0x03000000:
|
|
|
|
if (SWRAM_ARM7) return *(u32*)&SWRAM_ARM7[addr & SWRAM_ARM7Mask];
|
|
|
|
else return *(u32*)&ARM7WRAM[addr & 0xFFFF];
|
|
|
|
|
2016-12-03 12:42:27 +00:00
|
|
|
case 0x03800000:
|
|
|
|
return *(u32*)&ARM7WRAM[addr & 0xFFFF];
|
2016-12-03 17:29:19 +00:00
|
|
|
|
|
|
|
case 0x04000000:
|
2017-01-17 01:29:25 +00:00
|
|
|
return ARM7IORead32(addr);
|
2016-12-05 22:17:03 +00:00
|
|
|
|
2017-05-01 17:29:25 +00:00
|
|
|
case 0x04800000:
|
|
|
|
return Wifi::Read(addr) | (Wifi::Read(addr+2) << 16);
|
|
|
|
|
2016-12-05 22:17:03 +00:00
|
|
|
case 0x06000000:
|
|
|
|
case 0x06800000:
|
2017-02-27 20:26:11 +00:00
|
|
|
return GPU::ReadVRAM_ARM7<u32>(addr);
|
2016-12-03 12:42:27 +00:00
|
|
|
}
|
2016-12-05 16:08:24 +00:00
|
|
|
|
2016-12-03 16:58:24 +00:00
|
|
|
printf("unknown arm7 read32 %08X | %08X\n", addr, ARM7->R[15]);
|
2016-11-03 00:38:58 +00:00
|
|
|
return 0;
|
|
|
|
}
|
2016-05-16 15:48:40 +00:00
|
|
|
|
2016-12-03 00:31:33 +00:00
|
|
|
void ARM7Write8(u32 addr, u8 val)
|
|
|
|
{
|
2016-12-03 12:42:27 +00:00
|
|
|
switch (addr & 0xFF800000)
|
|
|
|
{
|
|
|
|
case 0x02000000:
|
2017-02-01 20:35:00 +00:00
|
|
|
case 0x02800000:
|
2016-12-03 12:42:27 +00:00
|
|
|
*(u8*)&MainRAM[addr & 0x3FFFFF] = val;
|
|
|
|
return;
|
|
|
|
|
2016-12-03 17:29:19 +00:00
|
|
|
case 0x03000000:
|
|
|
|
if (SWRAM_ARM7) *(u8*)&SWRAM_ARM7[addr & SWRAM_ARM7Mask] = val;
|
|
|
|
else *(u8*)&ARM7WRAM[addr & 0xFFFF] = val;
|
|
|
|
return;
|
|
|
|
|
2016-12-03 12:42:27 +00:00
|
|
|
case 0x03800000:
|
|
|
|
*(u8*)&ARM7WRAM[addr & 0xFFFF] = val;
|
|
|
|
return;
|
2016-12-04 02:20:50 +00:00
|
|
|
|
|
|
|
case 0x04000000:
|
2017-01-17 01:29:25 +00:00
|
|
|
ARM7IOWrite8(addr, val);
|
|
|
|
return;
|
2016-12-05 22:17:03 +00:00
|
|
|
|
|
|
|
case 0x06000000:
|
|
|
|
case 0x06800000:
|
2017-02-27 20:26:11 +00:00
|
|
|
GPU::WriteVRAM_ARM7<u8>(addr, val);
|
2016-12-05 22:17:03 +00:00
|
|
|
return;
|
2016-12-03 12:42:27 +00:00
|
|
|
}
|
|
|
|
|
2017-04-12 14:58:09 +00:00
|
|
|
printf("unknown arm7 write8 %08X %02X @ %08X\n", addr, val, ARM7->R[15]);
|
2016-12-03 00:31:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ARM7Write16(u32 addr, u16 val)
|
2017-06-12 23:01:46 +00:00
|
|
|
{
|
2016-12-03 12:42:27 +00:00
|
|
|
switch (addr & 0xFF800000)
|
|
|
|
{
|
|
|
|
case 0x02000000:
|
2017-02-01 20:35:00 +00:00
|
|
|
case 0x02800000:
|
2016-12-03 12:42:27 +00:00
|
|
|
*(u16*)&MainRAM[addr & 0x3FFFFF] = val;
|
|
|
|
return;
|
|
|
|
|
2016-12-03 17:29:19 +00:00
|
|
|
case 0x03000000:
|
|
|
|
if (SWRAM_ARM7) *(u16*)&SWRAM_ARM7[addr & SWRAM_ARM7Mask] = val;
|
|
|
|
else *(u16*)&ARM7WRAM[addr & 0xFFFF] = val;
|
|
|
|
return;
|
|
|
|
|
2016-12-03 12:42:27 +00:00
|
|
|
case 0x03800000:
|
|
|
|
*(u16*)&ARM7WRAM[addr & 0xFFFF] = val;
|
|
|
|
return;
|
2016-12-03 15:13:04 +00:00
|
|
|
|
|
|
|
case 0x04000000:
|
2017-01-17 01:29:25 +00:00
|
|
|
ARM7IOWrite16(addr, val);
|
|
|
|
return;
|
2016-12-05 22:17:03 +00:00
|
|
|
|
|
|
|
case 0x04800000:
|
2016-12-06 16:32:51 +00:00
|
|
|
Wifi::Write(addr, val);
|
2016-12-05 22:17:03 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
case 0x06000000:
|
|
|
|
case 0x06800000:
|
2017-02-27 20:26:11 +00:00
|
|
|
GPU::WriteVRAM_ARM7<u16>(addr, val);
|
2016-12-05 22:17:03 +00:00
|
|
|
return;
|
2016-12-03 12:42:27 +00:00
|
|
|
}
|
|
|
|
|
2017-06-05 12:10:12 +00:00
|
|
|
//printf("unknown arm7 write16 %08X %04X @ %08X\n", addr, val, ARM7->R[15]);
|
2016-05-16 15:48:40 +00:00
|
|
|
}
|
|
|
|
|
2016-12-03 00:31:33 +00:00
|
|
|
void ARM7Write32(u32 addr, u32 val)
|
2016-05-16 15:48:40 +00:00
|
|
|
{
|
2016-12-03 12:42:27 +00:00
|
|
|
switch (addr & 0xFF800000)
|
|
|
|
{
|
|
|
|
case 0x02000000:
|
2017-02-01 20:35:00 +00:00
|
|
|
case 0x02800000:
|
2016-12-03 12:42:27 +00:00
|
|
|
*(u32*)&MainRAM[addr & 0x3FFFFF] = val;
|
|
|
|
return;
|
|
|
|
|
2016-12-03 17:29:19 +00:00
|
|
|
case 0x03000000:
|
|
|
|
if (SWRAM_ARM7) *(u32*)&SWRAM_ARM7[addr & SWRAM_ARM7Mask] = val;
|
|
|
|
else *(u32*)&ARM7WRAM[addr & 0xFFFF] = val;
|
|
|
|
return;
|
|
|
|
|
2016-12-03 12:42:27 +00:00
|
|
|
case 0x03800000:
|
|
|
|
*(u32*)&ARM7WRAM[addr & 0xFFFF] = val;
|
|
|
|
return;
|
2016-12-04 02:20:50 +00:00
|
|
|
|
|
|
|
case 0x04000000:
|
2017-01-17 01:29:25 +00:00
|
|
|
ARM7IOWrite32(addr, val);
|
|
|
|
return;
|
2016-12-05 22:17:03 +00:00
|
|
|
|
2017-05-01 17:29:25 +00:00
|
|
|
case 0x04800000:
|
|
|
|
Wifi::Write(addr, val & 0xFFFF);
|
|
|
|
Wifi::Write(addr+2, val >> 16);
|
|
|
|
return;
|
|
|
|
|
2016-12-05 22:17:03 +00:00
|
|
|
case 0x06000000:
|
|
|
|
case 0x06800000:
|
2017-02-27 20:26:11 +00:00
|
|
|
GPU::WriteVRAM_ARM7<u32>(addr, val);
|
2016-12-05 22:17:03 +00:00
|
|
|
return;
|
2016-12-03 12:42:27 +00:00
|
|
|
}
|
|
|
|
|
2017-06-05 12:10:12 +00:00
|
|
|
//printf("unknown arm7 write32 %08X %08X @ %08X\n", addr, val, ARM7->R[15]);
|
2016-05-16 15:48:40 +00:00
|
|
|
}
|
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
2017-10-02 02:27:50 +00:00
|
|
|
#define CASE_READ8_16BIT(addr, val) \
|
|
|
|
case (addr): return (val) & 0xFF; \
|
|
|
|
case (addr+1): return (val) >> 8;
|
|
|
|
|
|
|
|
#define CASE_READ8_32BIT(addr, val) \
|
|
|
|
case (addr): return (val) & 0xFF; \
|
|
|
|
case (addr+1): return ((val) >> 8) & 0xFF; \
|
|
|
|
case (addr+2): return ((val) >> 16) & 0xFF; \
|
|
|
|
case (addr+3): return (val) >> 24;
|
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
u8 ARM9IORead8(u32 addr)
|
|
|
|
{
|
|
|
|
switch (addr)
|
|
|
|
{
|
2017-04-23 23:30:30 +00:00
|
|
|
case 0x04000130: return KeyInput & 0xFF;
|
|
|
|
case 0x04000131: return (KeyInput >> 8) & 0xFF;
|
2017-06-04 20:34:31 +00:00
|
|
|
case 0x04000132: return KeyCnt & 0xFF;
|
|
|
|
case 0x04000133: return KeyCnt >> 8;
|
2017-04-23 23:30:30 +00:00
|
|
|
|
2017-01-31 16:34:17 +00:00
|
|
|
case 0x040001A2: return NDSCart::ReadSPIData();
|
|
|
|
|
2017-03-30 23:50:01 +00:00
|
|
|
case 0x040001A8: return NDSCart::ROMCommand[0];
|
|
|
|
case 0x040001A9: return NDSCart::ROMCommand[1];
|
|
|
|
case 0x040001AA: return NDSCart::ROMCommand[2];
|
|
|
|
case 0x040001AB: return NDSCart::ROMCommand[3];
|
|
|
|
case 0x040001AC: return NDSCart::ROMCommand[4];
|
|
|
|
case 0x040001AD: return NDSCart::ROMCommand[5];
|
|
|
|
case 0x040001AE: return NDSCart::ROMCommand[6];
|
|
|
|
case 0x040001AF: return NDSCart::ROMCommand[7];
|
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
case 0x04000208: return IME[0];
|
|
|
|
|
2017-01-18 03:03:19 +00:00
|
|
|
case 0x04000240: return GPU::VRAMCNT[0];
|
|
|
|
case 0x04000241: return GPU::VRAMCNT[1];
|
|
|
|
case 0x04000242: return GPU::VRAMCNT[2];
|
|
|
|
case 0x04000243: return GPU::VRAMCNT[3];
|
|
|
|
case 0x04000244: return GPU::VRAMCNT[4];
|
|
|
|
case 0x04000245: return GPU::VRAMCNT[5];
|
|
|
|
case 0x04000246: return GPU::VRAMCNT[6];
|
2017-01-17 01:29:25 +00:00
|
|
|
case 0x04000247: return WRAMCnt;
|
2017-01-18 03:03:19 +00:00
|
|
|
case 0x04000248: return GPU::VRAMCNT[7];
|
|
|
|
case 0x04000249: return GPU::VRAMCNT[8];
|
2017-01-17 01:29:25 +00:00
|
|
|
|
2017-10-02 02:27:50 +00:00
|
|
|
CASE_READ8_16BIT(0x04000280, DivCnt)
|
|
|
|
CASE_READ8_32BIT(0x04000290, DivNumerator[0])
|
|
|
|
CASE_READ8_32BIT(0x04000294, DivNumerator[1])
|
|
|
|
CASE_READ8_32BIT(0x04000298, DivDenominator[0])
|
|
|
|
CASE_READ8_32BIT(0x0400029C, DivDenominator[1])
|
|
|
|
CASE_READ8_32BIT(0x040002A0, DivQuotient[0])
|
|
|
|
CASE_READ8_32BIT(0x040002A4, DivQuotient[1])
|
|
|
|
CASE_READ8_32BIT(0x040002A8, DivRemainder[0])
|
|
|
|
CASE_READ8_32BIT(0x040002AC, DivRemainder[1])
|
|
|
|
|
|
|
|
CASE_READ8_16BIT(0x040002B0, SqrtCnt)
|
|
|
|
CASE_READ8_32BIT(0x040002B4, SqrtRes)
|
|
|
|
CASE_READ8_32BIT(0x040002B8, SqrtVal[0])
|
|
|
|
CASE_READ8_32BIT(0x040002BC, SqrtVal[1])
|
|
|
|
|
2017-01-18 01:20:45 +00:00
|
|
|
case 0x04000300: return PostFlag9;
|
2017-01-17 01:29:25 +00:00
|
|
|
}
|
|
|
|
|
2017-01-18 03:03:19 +00:00
|
|
|
if (addr >= 0x04000000 && addr < 0x04000060)
|
|
|
|
{
|
|
|
|
return GPU::GPU2D_A->Read8(addr);
|
|
|
|
}
|
|
|
|
if (addr >= 0x04001000 && addr < 0x04001060)
|
|
|
|
{
|
|
|
|
return GPU::GPU2D_B->Read8(addr);
|
|
|
|
}
|
2017-02-07 22:31:21 +00:00
|
|
|
if (addr >= 0x04000320 && addr < 0x040006A4)
|
|
|
|
{
|
|
|
|
return GPU3D::Read8(addr);
|
|
|
|
}
|
2017-01-18 03:03:19 +00:00
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
printf("unknown ARM9 IO read8 %08X\n", addr);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
u16 ARM9IORead16(u32 addr)
|
|
|
|
{
|
|
|
|
switch (addr)
|
|
|
|
{
|
2017-01-18 03:03:19 +00:00
|
|
|
case 0x04000004: return GPU::DispStat[0];
|
|
|
|
case 0x04000006: return GPU::VCount;
|
2017-01-17 01:29:25 +00:00
|
|
|
|
2017-03-07 00:13:00 +00:00
|
|
|
case 0x04000060: return GPU3D::Read16(addr);
|
2017-03-04 13:47:20 +00:00
|
|
|
case 0x04000064:
|
|
|
|
case 0x04000066: return GPU::GPU2D_A->Read16(addr);
|
2017-01-31 23:24:36 +00:00
|
|
|
|
2017-01-31 16:34:17 +00:00
|
|
|
case 0x040000B8: return DMAs[0]->Cnt & 0xFFFF;
|
|
|
|
case 0x040000BA: return DMAs[0]->Cnt >> 16;
|
|
|
|
case 0x040000C4: return DMAs[1]->Cnt & 0xFFFF;
|
|
|
|
case 0x040000C6: return DMAs[1]->Cnt >> 16;
|
|
|
|
case 0x040000D0: return DMAs[2]->Cnt & 0xFFFF;
|
|
|
|
case 0x040000D2: return DMAs[2]->Cnt >> 16;
|
|
|
|
case 0x040000DC: return DMAs[3]->Cnt & 0xFFFF;
|
|
|
|
case 0x040000DE: return DMAs[3]->Cnt >> 16;
|
|
|
|
|
2017-01-18 00:33:06 +00:00
|
|
|
case 0x040000E0: return ((u16*)DMA9Fill)[0];
|
|
|
|
case 0x040000E2: return ((u16*)DMA9Fill)[1];
|
|
|
|
case 0x040000E4: return ((u16*)DMA9Fill)[2];
|
|
|
|
case 0x040000E6: return ((u16*)DMA9Fill)[3];
|
|
|
|
case 0x040000E8: return ((u16*)DMA9Fill)[4];
|
|
|
|
case 0x040000EA: return ((u16*)DMA9Fill)[5];
|
|
|
|
case 0x040000EC: return ((u16*)DMA9Fill)[6];
|
|
|
|
case 0x040000EE: return ((u16*)DMA9Fill)[7];
|
|
|
|
|
2017-01-31 02:54:51 +00:00
|
|
|
case 0x04000100: return TimerGetCounter(0);
|
|
|
|
case 0x04000102: return Timers[0].Cnt;
|
|
|
|
case 0x04000104: return TimerGetCounter(1);
|
|
|
|
case 0x04000106: return Timers[1].Cnt;
|
|
|
|
case 0x04000108: return TimerGetCounter(2);
|
|
|
|
case 0x0400010A: return Timers[2].Cnt;
|
|
|
|
case 0x0400010C: return TimerGetCounter(3);
|
|
|
|
case 0x0400010E: return Timers[3].Cnt;
|
2017-01-17 01:29:25 +00:00
|
|
|
|
|
|
|
case 0x04000130: return KeyInput & 0xFFFF;
|
2017-06-04 20:34:31 +00:00
|
|
|
case 0x04000132: return KeyCnt;
|
2017-01-17 01:29:25 +00:00
|
|
|
|
|
|
|
case 0x04000180: return IPCSync9;
|
|
|
|
case 0x04000184:
|
|
|
|
{
|
|
|
|
u16 val = IPCFIFOCnt9;
|
|
|
|
if (IPCFIFO9->IsEmpty()) val |= 0x0001;
|
|
|
|
else if (IPCFIFO9->IsFull()) val |= 0x0002;
|
|
|
|
if (IPCFIFO7->IsEmpty()) val |= 0x0100;
|
|
|
|
else if (IPCFIFO7->IsFull()) val |= 0x0200;
|
|
|
|
return val;
|
|
|
|
}
|
|
|
|
|
2017-01-22 19:34:59 +00:00
|
|
|
case 0x040001A0: return NDSCart::SPICnt;
|
2017-01-31 16:34:17 +00:00
|
|
|
case 0x040001A2: return NDSCart::ReadSPIData();
|
2017-01-17 01:29:25 +00:00
|
|
|
|
2017-03-30 23:50:01 +00:00
|
|
|
case 0x040001A8: return NDSCart::ROMCommand[0] |
|
|
|
|
(NDSCart::ROMCommand[1] << 8);
|
|
|
|
case 0x040001AA: return NDSCart::ROMCommand[2] |
|
|
|
|
(NDSCart::ROMCommand[3] << 8);
|
|
|
|
case 0x040001AC: return NDSCart::ROMCommand[4] |
|
|
|
|
(NDSCart::ROMCommand[5] << 8);
|
|
|
|
case 0x040001AE: return NDSCart::ROMCommand[6] |
|
|
|
|
(NDSCart::ROMCommand[7] << 8);
|
|
|
|
|
2017-01-22 19:34:59 +00:00
|
|
|
case 0x04000204: return ExMemCnt[0];
|
2017-01-17 01:29:25 +00:00
|
|
|
case 0x04000208: return IME[0];
|
2017-06-23 23:21:09 +00:00
|
|
|
case 0x04000210: return IE[0] & 0xFFFF;
|
|
|
|
case 0x04000212: return IE[0] >> 16;
|
2017-01-17 01:29:25 +00:00
|
|
|
|
2017-03-04 13:47:20 +00:00
|
|
|
case 0x04000240: return GPU::VRAMCNT[0] | (GPU::VRAMCNT[1] << 8);
|
|
|
|
case 0x04000242: return GPU::VRAMCNT[2] | (GPU::VRAMCNT[3] << 8);
|
|
|
|
case 0x04000244: return GPU::VRAMCNT[4] | (GPU::VRAMCNT[5] << 8);
|
|
|
|
case 0x04000246: return GPU::VRAMCNT[6] | (WRAMCnt << 8);
|
|
|
|
case 0x04000248: return GPU::VRAMCNT[7] | (GPU::VRAMCNT[8] << 8);
|
|
|
|
|
2017-01-18 01:20:45 +00:00
|
|
|
case 0x04000280: return DivCnt;
|
2017-07-05 16:46:19 +00:00
|
|
|
case 0x04000290: return DivNumerator[0] & 0xFFFF;
|
|
|
|
case 0x04000292: return DivNumerator[0] >> 16;
|
|
|
|
case 0x04000294: return DivNumerator[1] & 0xFFFF;
|
|
|
|
case 0x04000296: return DivNumerator[1] >> 16;
|
|
|
|
case 0x04000298: return DivDenominator[0] & 0xFFFF;
|
|
|
|
case 0x0400029A: return DivDenominator[0] >> 16;
|
|
|
|
case 0x0400029C: return DivDenominator[1] & 0xFFFF;
|
|
|
|
case 0x0400029E: return DivDenominator[1] >> 16;
|
|
|
|
case 0x040002A0: return DivQuotient[0] & 0xFFFF;
|
|
|
|
case 0x040002A2: return DivQuotient[0] >> 16;
|
|
|
|
case 0x040002A4: return DivQuotient[1] & 0xFFFF;
|
|
|
|
case 0x040002A6: return DivQuotient[1] >> 16;
|
|
|
|
case 0x040002A8: return DivRemainder[0] & 0xFFFF;
|
|
|
|
case 0x040002AA: return DivRemainder[0] >> 16;
|
|
|
|
case 0x040002AC: return DivRemainder[1] & 0xFFFF;
|
|
|
|
case 0x040002AE: return DivRemainder[1] >> 16;
|
2017-01-18 01:20:45 +00:00
|
|
|
|
2017-01-31 20:53:45 +00:00
|
|
|
case 0x040002B0: return SqrtCnt;
|
2017-07-05 16:46:19 +00:00
|
|
|
case 0x040002B4: return SqrtRes & 0xFFFF;
|
|
|
|
case 0x040002B6: return SqrtRes >> 16;
|
|
|
|
case 0x040002B8: return SqrtVal[0] & 0xFFFF;
|
|
|
|
case 0x040002BA: return SqrtVal[0] >> 16;
|
|
|
|
case 0x040002BC: return SqrtVal[1] & 0xFFFF;
|
|
|
|
case 0x040002BE: return SqrtVal[1] >> 16;
|
2017-01-31 20:53:45 +00:00
|
|
|
|
2017-01-18 01:20:45 +00:00
|
|
|
case 0x04000300: return PostFlag9;
|
2017-01-17 01:29:25 +00:00
|
|
|
case 0x04000304: return PowerControl9;
|
|
|
|
}
|
|
|
|
|
2017-04-09 01:47:47 +00:00
|
|
|
if ((addr >= 0x04000000 && addr < 0x04000060) || (addr == 0x0400006C))
|
2017-01-18 03:03:19 +00:00
|
|
|
{
|
|
|
|
return GPU::GPU2D_A->Read16(addr);
|
|
|
|
}
|
2017-04-09 01:47:47 +00:00
|
|
|
if ((addr >= 0x04001000 && addr < 0x04001060) || (addr == 0x0400106C))
|
2017-01-18 03:03:19 +00:00
|
|
|
{
|
|
|
|
return GPU::GPU2D_B->Read16(addr);
|
|
|
|
}
|
2017-02-07 22:31:21 +00:00
|
|
|
if (addr >= 0x04000320 && addr < 0x040006A4)
|
|
|
|
{
|
|
|
|
return GPU3D::Read16(addr);
|
|
|
|
}
|
2017-01-18 03:03:19 +00:00
|
|
|
|
2017-01-18 16:57:12 +00:00
|
|
|
printf("unknown ARM9 IO read16 %08X %08X\n", addr, ARM9->R[15]);
|
2017-01-17 01:29:25 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
u32 ARM9IORead32(u32 addr)
|
|
|
|
{
|
|
|
|
switch (addr)
|
|
|
|
{
|
2017-01-18 03:03:19 +00:00
|
|
|
case 0x04000004: return GPU::DispStat[0] | (GPU::VCount << 16);
|
2017-01-17 01:29:25 +00:00
|
|
|
|
2017-03-07 00:13:00 +00:00
|
|
|
case 0x04000060: return GPU3D::Read32(addr);
|
2017-03-04 13:47:20 +00:00
|
|
|
case 0x04000064: return GPU::GPU2D_A->Read32(addr);
|
|
|
|
|
2017-01-18 00:33:06 +00:00
|
|
|
case 0x040000B0: return DMAs[0]->SrcAddr;
|
|
|
|
case 0x040000B4: return DMAs[0]->DstAddr;
|
|
|
|
case 0x040000B8: return DMAs[0]->Cnt;
|
|
|
|
case 0x040000BC: return DMAs[1]->SrcAddr;
|
|
|
|
case 0x040000C0: return DMAs[1]->DstAddr;
|
|
|
|
case 0x040000C4: return DMAs[1]->Cnt;
|
|
|
|
case 0x040000C8: return DMAs[2]->SrcAddr;
|
|
|
|
case 0x040000CC: return DMAs[2]->DstAddr;
|
|
|
|
case 0x040000D0: return DMAs[2]->Cnt;
|
|
|
|
case 0x040000D4: return DMAs[3]->SrcAddr;
|
|
|
|
case 0x040000D8: return DMAs[3]->DstAddr;
|
|
|
|
case 0x040000DC: return DMAs[3]->Cnt;
|
|
|
|
|
|
|
|
case 0x040000E0: return DMA9Fill[0];
|
|
|
|
case 0x040000E4: return DMA9Fill[1];
|
|
|
|
case 0x040000E8: return DMA9Fill[2];
|
|
|
|
case 0x040000EC: return DMA9Fill[3];
|
|
|
|
|
2017-11-09 15:02:37 +00:00
|
|
|
case 0x040000F4: return 0; // ???? Golden Sun Dark Dawn keeps reading this
|
|
|
|
|
2017-01-31 02:54:51 +00:00
|
|
|
case 0x04000100: return TimerGetCounter(0) | (Timers[0].Cnt << 16);
|
|
|
|
case 0x04000104: return TimerGetCounter(1) | (Timers[1].Cnt << 16);
|
|
|
|
case 0x04000108: return TimerGetCounter(2) | (Timers[2].Cnt << 16);
|
|
|
|
case 0x0400010C: return TimerGetCounter(3) | (Timers[3].Cnt << 16);
|
2017-01-17 01:29:25 +00:00
|
|
|
|
2017-06-04 20:34:31 +00:00
|
|
|
case 0x04000130: return (KeyInput & 0xFFFF) | (KeyCnt << 16);
|
|
|
|
|
2017-01-31 16:34:17 +00:00
|
|
|
case 0x040001A0: return NDSCart::SPICnt | (NDSCart::ReadSPIData() << 16);
|
2017-01-22 19:34:59 +00:00
|
|
|
case 0x040001A4: return NDSCart::ROMCnt;
|
|
|
|
|
2017-03-30 23:50:01 +00:00
|
|
|
case 0x040001A8: return NDSCart::ROMCommand[0] |
|
|
|
|
(NDSCart::ROMCommand[1] << 8) |
|
|
|
|
(NDSCart::ROMCommand[2] << 16) |
|
|
|
|
(NDSCart::ROMCommand[3] << 24);
|
|
|
|
case 0x040001AC: return NDSCart::ROMCommand[4] |
|
|
|
|
(NDSCart::ROMCommand[5] << 8) |
|
|
|
|
(NDSCart::ROMCommand[6] << 16) |
|
|
|
|
(NDSCart::ROMCommand[7] << 24);
|
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
case 0x04000208: return IME[0];
|
|
|
|
case 0x04000210: return IE[0];
|
|
|
|
case 0x04000214: return IF[0];
|
|
|
|
|
2017-03-04 13:47:20 +00:00
|
|
|
case 0x04000240: return GPU::VRAMCNT[0] | (GPU::VRAMCNT[1] << 8) | (GPU::VRAMCNT[2] << 16) | (GPU::VRAMCNT[3] << 24);
|
|
|
|
case 0x04000244: return GPU::VRAMCNT[4] | (GPU::VRAMCNT[5] << 8) | (GPU::VRAMCNT[6] << 16) | (WRAMCnt << 24);
|
|
|
|
case 0x04000248: return GPU::VRAMCNT[7] | (GPU::VRAMCNT[8] << 8);
|
|
|
|
|
2017-04-07 15:37:49 +00:00
|
|
|
case 0x04000280: return DivCnt;
|
2017-01-18 01:20:45 +00:00
|
|
|
case 0x04000290: return DivNumerator[0];
|
|
|
|
case 0x04000294: return DivNumerator[1];
|
|
|
|
case 0x04000298: return DivDenominator[0];
|
|
|
|
case 0x0400029C: return DivDenominator[1];
|
|
|
|
case 0x040002A0: return DivQuotient[0];
|
|
|
|
case 0x040002A4: return DivQuotient[1];
|
|
|
|
case 0x040002A8: return DivRemainder[0];
|
|
|
|
case 0x040002AC: return DivRemainder[1];
|
|
|
|
|
2017-04-07 15:37:49 +00:00
|
|
|
case 0x040002B0: return SqrtCnt;
|
2017-01-31 20:53:45 +00:00
|
|
|
case 0x040002B4: return SqrtRes;
|
|
|
|
case 0x040002B8: return SqrtVal[0];
|
|
|
|
case 0x040002BC: return SqrtVal[1];
|
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
case 0x04100000:
|
|
|
|
if (IPCFIFOCnt9 & 0x8000)
|
|
|
|
{
|
|
|
|
u32 ret;
|
|
|
|
if (IPCFIFO7->IsEmpty())
|
|
|
|
{
|
|
|
|
IPCFIFOCnt9 |= 0x4000;
|
|
|
|
ret = IPCFIFO7->Peek();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ret = IPCFIFO7->Read();
|
|
|
|
|
|
|
|
if (IPCFIFO7->IsEmpty() && (IPCFIFOCnt7 & 0x0004))
|
2017-03-02 23:48:26 +00:00
|
|
|
SetIRQ(1, IRQ_IPCSendDone);
|
2017-01-17 01:29:25 +00:00
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return IPCFIFO7->Peek();
|
2017-01-22 19:34:59 +00:00
|
|
|
|
|
|
|
case 0x04100010:
|
2017-01-31 16:34:17 +00:00
|
|
|
if (!(ExMemCnt[0] & (1<<11))) return NDSCart::ReadROMData();
|
2017-01-22 19:34:59 +00:00
|
|
|
return 0;
|
2017-01-17 01:29:25 +00:00
|
|
|
}
|
|
|
|
|
2017-04-09 01:47:47 +00:00
|
|
|
if ((addr >= 0x04000000 && addr < 0x04000060) || (addr == 0x0400006C))
|
2017-01-18 03:03:19 +00:00
|
|
|
{
|
|
|
|
return GPU::GPU2D_A->Read32(addr);
|
|
|
|
}
|
2017-04-09 01:47:47 +00:00
|
|
|
if ((addr >= 0x04001000 && addr < 0x04001060) || (addr == 0x0400106C))
|
2017-01-18 03:03:19 +00:00
|
|
|
{
|
|
|
|
return GPU::GPU2D_B->Read32(addr);
|
|
|
|
}
|
2017-02-01 20:57:25 +00:00
|
|
|
if (addr >= 0x04000320 && addr < 0x040006A4)
|
|
|
|
{
|
2017-02-07 22:31:21 +00:00
|
|
|
return GPU3D::Read32(addr);
|
2017-02-01 20:57:25 +00:00
|
|
|
}
|
2017-01-18 03:03:19 +00:00
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
printf("unknown ARM9 IO read32 %08X\n", addr);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ARM9IOWrite8(u32 addr, u8 val)
|
|
|
|
{
|
|
|
|
switch (addr)
|
|
|
|
{
|
2017-07-23 13:31:09 +00:00
|
|
|
case 0x0400006C:
|
|
|
|
case 0x0400006D: GPU::GPU2D_A->Write8(addr, val); return;
|
|
|
|
case 0x0400106C:
|
|
|
|
case 0x0400106D: GPU::GPU2D_B->Write8(addr, val); return;
|
|
|
|
|
2017-06-04 20:34:31 +00:00
|
|
|
case 0x04000132:
|
|
|
|
KeyCnt = (KeyCnt & 0xFF00) | val;
|
|
|
|
return;
|
|
|
|
case 0x04000133:
|
|
|
|
KeyCnt = (KeyCnt & 0x00FF) | (val << 8);
|
|
|
|
return;
|
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
case 0x040001A0:
|
2017-01-22 19:34:59 +00:00
|
|
|
if (!(ExMemCnt[0] & (1<<11)))
|
|
|
|
{
|
2017-01-31 16:34:17 +00:00
|
|
|
NDSCart::WriteSPICnt((NDSCart::SPICnt & 0xFF00) | val);
|
2017-01-22 19:34:59 +00:00
|
|
|
}
|
2017-01-17 01:29:25 +00:00
|
|
|
return;
|
|
|
|
case 0x040001A1:
|
2017-01-22 19:34:59 +00:00
|
|
|
if (!(ExMemCnt[0] & (1<<11)))
|
|
|
|
{
|
2017-01-31 16:34:17 +00:00
|
|
|
NDSCart::WriteSPICnt((NDSCart::SPICnt & 0x00FF) | (val << 8));
|
2017-01-22 19:34:59 +00:00
|
|
|
}
|
2017-01-17 01:29:25 +00:00
|
|
|
return;
|
2017-01-31 16:34:17 +00:00
|
|
|
case 0x040001A2:
|
|
|
|
NDSCart::WriteSPIData(val);
|
|
|
|
return;
|
2017-01-17 01:29:25 +00:00
|
|
|
|
2017-01-22 19:34:59 +00:00
|
|
|
case 0x040001A8: NDSCart::ROMCommand[0] = val; return;
|
|
|
|
case 0x040001A9: NDSCart::ROMCommand[1] = val; return;
|
|
|
|
case 0x040001AA: NDSCart::ROMCommand[2] = val; return;
|
|
|
|
case 0x040001AB: NDSCart::ROMCommand[3] = val; return;
|
|
|
|
case 0x040001AC: NDSCart::ROMCommand[4] = val; return;
|
|
|
|
case 0x040001AD: NDSCart::ROMCommand[5] = val; return;
|
|
|
|
case 0x040001AE: NDSCart::ROMCommand[6] = val; return;
|
|
|
|
case 0x040001AF: NDSCart::ROMCommand[7] = val; return;
|
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
case 0x04000208: IME[0] = val & 0x1; return;
|
|
|
|
|
2017-01-18 03:03:19 +00:00
|
|
|
case 0x04000240: GPU::MapVRAM_AB(0, val); return;
|
|
|
|
case 0x04000241: GPU::MapVRAM_AB(1, val); return;
|
|
|
|
case 0x04000242: GPU::MapVRAM_CD(2, val); return;
|
|
|
|
case 0x04000243: GPU::MapVRAM_CD(3, val); return;
|
|
|
|
case 0x04000244: GPU::MapVRAM_E(4, val); return;
|
|
|
|
case 0x04000245: GPU::MapVRAM_FG(5, val); return;
|
|
|
|
case 0x04000246: GPU::MapVRAM_FG(6, val); return;
|
2017-01-17 01:29:25 +00:00
|
|
|
case 0x04000247: MapSharedWRAM(val); return;
|
2017-01-18 03:03:19 +00:00
|
|
|
case 0x04000248: GPU::MapVRAM_H(7, val); return;
|
|
|
|
case 0x04000249: GPU::MapVRAM_I(8, val); return;
|
2017-01-18 01:20:45 +00:00
|
|
|
|
|
|
|
case 0x04000300:
|
|
|
|
if (PostFlag9 & 0x01) val |= 0x01;
|
|
|
|
PostFlag9 = val & 0x03;
|
|
|
|
return;
|
2017-01-17 01:29:25 +00:00
|
|
|
}
|
|
|
|
|
2017-01-18 03:03:19 +00:00
|
|
|
if (addr >= 0x04000000 && addr < 0x04000060)
|
|
|
|
{
|
|
|
|
GPU::GPU2D_A->Write8(addr, val);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (addr >= 0x04001000 && addr < 0x04001060)
|
|
|
|
{
|
|
|
|
GPU::GPU2D_B->Write8(addr, val);
|
|
|
|
return;
|
|
|
|
}
|
2017-02-07 22:31:21 +00:00
|
|
|
if (addr >= 0x04000320 && addr < 0x040006A4)
|
|
|
|
{
|
|
|
|
GPU3D::Write8(addr, val);
|
|
|
|
return;
|
|
|
|
}
|
2017-01-18 03:03:19 +00:00
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
printf("unknown ARM9 IO write8 %08X %02X\n", addr, val);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ARM9IOWrite16(u32 addr, u16 val)
|
|
|
|
{
|
|
|
|
switch (addr)
|
|
|
|
{
|
2017-01-18 03:03:19 +00:00
|
|
|
case 0x04000004: GPU::SetDispStat(0, val); return;
|
2017-05-10 00:21:02 +00:00
|
|
|
case 0x04000006: GPU::SetVCount(val); return;
|
2017-01-17 01:29:25 +00:00
|
|
|
|
2017-03-07 00:13:00 +00:00
|
|
|
case 0x04000060: GPU3D::Write16(addr, val); return;
|
2017-01-31 23:24:36 +00:00
|
|
|
|
2017-06-26 09:02:10 +00:00
|
|
|
case 0x04000068:
|
|
|
|
case 0x0400006A: GPU::GPU2D_A->Write16(addr, val); return;
|
|
|
|
|
2017-07-23 13:31:09 +00:00
|
|
|
case 0x0400006C: GPU::GPU2D_A->Write16(addr, val); return;
|
|
|
|
case 0x0400106C: GPU::GPU2D_B->Write16(addr, val); return;
|
|
|
|
|
2017-01-31 16:34:17 +00:00
|
|
|
case 0x040000B8: DMAs[0]->WriteCnt((DMAs[0]->Cnt & 0xFFFF0000) | val); return;
|
|
|
|
case 0x040000BA: DMAs[0]->WriteCnt((DMAs[0]->Cnt & 0x0000FFFF) | (val << 16)); return;
|
|
|
|
case 0x040000C4: DMAs[1]->WriteCnt((DMAs[1]->Cnt & 0xFFFF0000) | val); return;
|
|
|
|
case 0x040000C6: DMAs[1]->WriteCnt((DMAs[1]->Cnt & 0x0000FFFF) | (val << 16)); return;
|
|
|
|
case 0x040000D0: DMAs[2]->WriteCnt((DMAs[2]->Cnt & 0xFFFF0000) | val); return;
|
|
|
|
case 0x040000D2: DMAs[2]->WriteCnt((DMAs[2]->Cnt & 0x0000FFFF) | (val << 16)); return;
|
|
|
|
case 0x040000DC: DMAs[3]->WriteCnt((DMAs[3]->Cnt & 0xFFFF0000) | val); return;
|
|
|
|
case 0x040000DE: DMAs[3]->WriteCnt((DMAs[3]->Cnt & 0x0000FFFF) | (val << 16)); return;
|
|
|
|
|
2017-04-25 14:13:16 +00:00
|
|
|
case 0x040000E0: DMA9Fill[0] = (DMA9Fill[0] & 0xFFFF0000) | val; return;
|
|
|
|
case 0x040000E2: DMA9Fill[0] = (DMA9Fill[0] & 0x0000FFFF) | (val << 16); return;
|
|
|
|
case 0x040000E4: DMA9Fill[1] = (DMA9Fill[1] & 0xFFFF0000) | val; return;
|
|
|
|
case 0x040000E6: DMA9Fill[1] = (DMA9Fill[1] & 0x0000FFFF) | (val << 16); return;
|
|
|
|
case 0x040000E8: DMA9Fill[2] = (DMA9Fill[2] & 0xFFFF0000) | val; return;
|
|
|
|
case 0x040000EA: DMA9Fill[2] = (DMA9Fill[2] & 0x0000FFFF) | (val << 16); return;
|
|
|
|
case 0x040000EC: DMA9Fill[3] = (DMA9Fill[3] & 0xFFFF0000) | val; return;
|
|
|
|
case 0x040000EE: DMA9Fill[3] = (DMA9Fill[3] & 0x0000FFFF) | (val << 16); return;
|
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
case 0x04000100: Timers[0].Reload = val; return;
|
|
|
|
case 0x04000102: TimerStart(0, val); return;
|
|
|
|
case 0x04000104: Timers[1].Reload = val; return;
|
|
|
|
case 0x04000106: TimerStart(1, val); return;
|
|
|
|
case 0x04000108: Timers[2].Reload = val; return;
|
|
|
|
case 0x0400010A: TimerStart(2, val); return;
|
|
|
|
case 0x0400010C: Timers[3].Reload = val; return;
|
|
|
|
case 0x0400010E: TimerStart(3, val); return;
|
|
|
|
|
2017-06-04 20:34:31 +00:00
|
|
|
case 0x04000132:
|
|
|
|
KeyCnt = val;
|
|
|
|
return;
|
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
case 0x04000180:
|
|
|
|
IPCSync7 &= 0xFFF0;
|
|
|
|
IPCSync7 |= ((val & 0x0F00) >> 8);
|
|
|
|
IPCSync9 &= 0xB0FF;
|
|
|
|
IPCSync9 |= (val & 0x4F00);
|
|
|
|
if ((val & 0x2000) && (IPCSync7 & 0x4000))
|
|
|
|
{
|
2017-03-02 23:48:26 +00:00
|
|
|
SetIRQ(1, IRQ_IPCSync);
|
2017-01-17 01:29:25 +00:00
|
|
|
}
|
|
|
|
return;
|
|
|
|
|
|
|
|
case 0x04000184:
|
|
|
|
if (val & 0x0008)
|
|
|
|
IPCFIFO9->Clear();
|
|
|
|
if ((val & 0x0004) && (!(IPCFIFOCnt9 & 0x0004)) && IPCFIFO9->IsEmpty())
|
2017-03-02 23:48:26 +00:00
|
|
|
SetIRQ(0, IRQ_IPCSendDone);
|
2017-01-17 01:29:25 +00:00
|
|
|
if ((val & 0x0400) && (!(IPCFIFOCnt9 & 0x0400)) && (!IPCFIFO7->IsEmpty()))
|
2017-03-02 23:48:26 +00:00
|
|
|
SetIRQ(0, IRQ_IPCRecv);
|
2017-01-17 01:29:25 +00:00
|
|
|
if (val & 0x4000)
|
|
|
|
IPCFIFOCnt9 &= ~0x4000;
|
|
|
|
IPCFIFOCnt9 = val & 0x8404;
|
|
|
|
return;
|
|
|
|
|
|
|
|
case 0x040001A0:
|
2017-01-31 16:34:17 +00:00
|
|
|
if (!(ExMemCnt[0] & (1<<11))) NDSCart::WriteSPICnt(val);
|
|
|
|
return;
|
|
|
|
case 0x040001A2:
|
|
|
|
NDSCart::WriteSPIData(val & 0xFF);
|
2017-01-22 19:34:59 +00:00
|
|
|
return;
|
|
|
|
|
2017-03-30 23:50:01 +00:00
|
|
|
case 0x040001A8:
|
|
|
|
NDSCart::ROMCommand[0] = val & 0xFF;
|
|
|
|
NDSCart::ROMCommand[1] = val >> 8;
|
|
|
|
return;
|
|
|
|
case 0x040001AA:
|
|
|
|
NDSCart::ROMCommand[2] = val & 0xFF;
|
|
|
|
NDSCart::ROMCommand[3] = val >> 8;
|
|
|
|
return;
|
|
|
|
case 0x040001AC:
|
|
|
|
NDSCart::ROMCommand[4] = val & 0xFF;
|
|
|
|
NDSCart::ROMCommand[5] = val >> 8;
|
|
|
|
return;
|
|
|
|
case 0x040001AE:
|
|
|
|
NDSCart::ROMCommand[6] = val & 0xFF;
|
|
|
|
NDSCart::ROMCommand[7] = val >> 8;
|
|
|
|
return;
|
|
|
|
|
2017-01-22 19:34:59 +00:00
|
|
|
case 0x040001B8: ROMSeed0[4] = val & 0x7F; return;
|
|
|
|
case 0x040001BA: ROMSeed1[4] = val & 0x7F; return;
|
|
|
|
|
|
|
|
case 0x04000204:
|
|
|
|
ExMemCnt[0] = val;
|
|
|
|
ExMemCnt[1] = (ExMemCnt[1] & 0x007F) | (val & 0xFF80);
|
2017-01-17 01:29:25 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
case 0x04000208: IME[0] = val & 0x1; return;
|
2017-06-23 23:21:09 +00:00
|
|
|
case 0x04000210: IE[0] = (IE[0] & 0xFFFF0000) | val; return;
|
|
|
|
case 0x04000212: IE[0] = (IE[0] & 0x0000FFFF) | (val << 16); return;
|
|
|
|
// TODO: what happens when writing to IF this way??
|
2017-01-17 01:29:25 +00:00
|
|
|
|
|
|
|
case 0x04000240:
|
2017-01-18 03:03:19 +00:00
|
|
|
GPU::MapVRAM_AB(0, val & 0xFF);
|
|
|
|
GPU::MapVRAM_AB(1, val >> 8);
|
2017-01-17 01:29:25 +00:00
|
|
|
return;
|
|
|
|
case 0x04000242:
|
2017-01-18 03:03:19 +00:00
|
|
|
GPU::MapVRAM_CD(2, val & 0xFF);
|
|
|
|
GPU::MapVRAM_CD(3, val >> 8);
|
2017-01-17 01:29:25 +00:00
|
|
|
return;
|
|
|
|
case 0x04000244:
|
2017-01-18 03:03:19 +00:00
|
|
|
GPU::MapVRAM_E(4, val & 0xFF);
|
|
|
|
GPU::MapVRAM_FG(5, val >> 8);
|
2017-01-17 01:29:25 +00:00
|
|
|
return;
|
|
|
|
case 0x04000246:
|
2017-01-18 03:03:19 +00:00
|
|
|
GPU::MapVRAM_FG(6, val & 0xFF);
|
2017-01-17 01:29:25 +00:00
|
|
|
MapSharedWRAM(val >> 8);
|
|
|
|
return;
|
|
|
|
case 0x04000248:
|
2017-01-18 03:03:19 +00:00
|
|
|
GPU::MapVRAM_H(7, val & 0xFF);
|
|
|
|
GPU::MapVRAM_I(8, val >> 8);
|
2017-01-17 01:29:25 +00:00
|
|
|
return;
|
|
|
|
|
2017-01-18 01:20:45 +00:00
|
|
|
case 0x04000280: DivCnt = val; StartDiv(); return;
|
|
|
|
|
2017-01-31 20:53:45 +00:00
|
|
|
case 0x040002B0: SqrtCnt = val; StartSqrt(); return;
|
|
|
|
|
2017-01-18 01:20:45 +00:00
|
|
|
case 0x04000300:
|
|
|
|
if (PostFlag9 & 0x01) val |= 0x01;
|
|
|
|
PostFlag9 = val & 0x03;
|
|
|
|
return;
|
|
|
|
|
2017-02-01 20:57:25 +00:00
|
|
|
case 0x04000304:
|
|
|
|
PowerControl9 = val;
|
|
|
|
GPU::DisplaySwap(PowerControl9>>15);
|
|
|
|
return;
|
2017-01-17 01:29:25 +00:00
|
|
|
}
|
|
|
|
|
2017-07-23 13:31:09 +00:00
|
|
|
if (addr >= 0x04000000 && addr < 0x04000060)
|
2017-01-18 03:03:19 +00:00
|
|
|
{
|
|
|
|
GPU::GPU2D_A->Write16(addr, val);
|
|
|
|
return;
|
|
|
|
}
|
2017-07-23 13:31:09 +00:00
|
|
|
if (addr >= 0x04001000 && addr < 0x04001060)
|
2017-01-18 03:03:19 +00:00
|
|
|
{
|
|
|
|
GPU::GPU2D_B->Write16(addr, val);
|
|
|
|
return;
|
|
|
|
}
|
2017-02-01 20:57:25 +00:00
|
|
|
if (addr >= 0x04000320 && addr < 0x040006A4)
|
|
|
|
{
|
2017-02-07 22:31:21 +00:00
|
|
|
GPU3D::Write16(addr, val);
|
2017-02-01 20:57:25 +00:00
|
|
|
return;
|
|
|
|
}
|
2017-01-18 03:03:19 +00:00
|
|
|
|
2017-02-05 15:50:20 +00:00
|
|
|
printf("unknown ARM9 IO write16 %08X %04X %08X\n", addr, val, ARM9->R[14]);
|
2017-01-17 01:29:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ARM9IOWrite32(u32 addr, u32 val)
|
|
|
|
{
|
|
|
|
switch (addr)
|
|
|
|
{
|
2017-03-07 00:13:00 +00:00
|
|
|
case 0x04000060: GPU3D::Write32(addr, val); return;
|
2017-06-26 09:02:10 +00:00
|
|
|
case 0x04000064:
|
|
|
|
case 0x04000068: GPU::GPU2D_A->Write32(addr, val); return;
|
2017-01-31 23:24:36 +00:00
|
|
|
|
2017-07-23 13:31:09 +00:00
|
|
|
case 0x0400006C: GPU::GPU2D_A->Write16(addr, val&0xFFFF); return;
|
|
|
|
case 0x0400106C: GPU::GPU2D_B->Write16(addr, val&0xFFFF); return;
|
|
|
|
|
2017-01-18 00:33:06 +00:00
|
|
|
case 0x040000B0: DMAs[0]->SrcAddr = val; return;
|
|
|
|
case 0x040000B4: DMAs[0]->DstAddr = val; return;
|
|
|
|
case 0x040000B8: DMAs[0]->WriteCnt(val); return;
|
|
|
|
case 0x040000BC: DMAs[1]->SrcAddr = val; return;
|
|
|
|
case 0x040000C0: DMAs[1]->DstAddr = val; return;
|
|
|
|
case 0x040000C4: DMAs[1]->WriteCnt(val); return;
|
|
|
|
case 0x040000C8: DMAs[2]->SrcAddr = val; return;
|
|
|
|
case 0x040000CC: DMAs[2]->DstAddr = val; return;
|
|
|
|
case 0x040000D0: DMAs[2]->WriteCnt(val); return;
|
|
|
|
case 0x040000D4: DMAs[3]->SrcAddr = val; return;
|
|
|
|
case 0x040000D8: DMAs[3]->DstAddr = val; return;
|
|
|
|
case 0x040000DC: DMAs[3]->WriteCnt(val); return;
|
|
|
|
|
|
|
|
case 0x040000E0: DMA9Fill[0] = val; return;
|
|
|
|
case 0x040000E4: DMA9Fill[1] = val; return;
|
|
|
|
case 0x040000E8: DMA9Fill[2] = val; return;
|
|
|
|
case 0x040000EC: DMA9Fill[3] = val; return;
|
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
case 0x04000100:
|
|
|
|
Timers[0].Reload = val & 0xFFFF;
|
|
|
|
TimerStart(0, val>>16);
|
|
|
|
return;
|
|
|
|
case 0x04000104:
|
|
|
|
Timers[1].Reload = val & 0xFFFF;
|
|
|
|
TimerStart(1, val>>16);
|
|
|
|
return;
|
|
|
|
case 0x04000108:
|
|
|
|
Timers[2].Reload = val & 0xFFFF;
|
|
|
|
TimerStart(2, val>>16);
|
|
|
|
return;
|
|
|
|
case 0x0400010C:
|
|
|
|
Timers[3].Reload = val & 0xFFFF;
|
|
|
|
TimerStart(3, val>>16);
|
|
|
|
return;
|
|
|
|
|
2017-06-04 20:34:31 +00:00
|
|
|
case 0x04000130:
|
|
|
|
KeyCnt = val >> 16;
|
|
|
|
return;
|
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
case 0x04000188:
|
|
|
|
if (IPCFIFOCnt9 & 0x8000)
|
2017-06-18 14:26:59 +00:00
|
|
|
{
|
2017-01-17 01:29:25 +00:00
|
|
|
if (IPCFIFO9->IsFull())
|
|
|
|
IPCFIFOCnt9 |= 0x4000;
|
|
|
|
else
|
|
|
|
{
|
2017-01-17 04:29:38 +00:00
|
|
|
bool wasempty = IPCFIFO9->IsEmpty();
|
2017-01-31 03:06:05 +00:00
|
|
|
IPCFIFO9->Write(val);
|
2017-01-17 04:29:38 +00:00
|
|
|
if ((IPCFIFOCnt7 & 0x0400) && wasempty)
|
2017-03-02 23:48:26 +00:00
|
|
|
SetIRQ(1, IRQ_IPCRecv);
|
2017-01-17 01:29:25 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
|
|
|
|
case 0x040001A0:
|
2017-01-22 19:34:59 +00:00
|
|
|
if (!(ExMemCnt[0] & (1<<11)))
|
|
|
|
{
|
2017-01-31 16:34:17 +00:00
|
|
|
NDSCart::WriteSPICnt(val & 0xFFFF);
|
|
|
|
NDSCart::WriteSPIData((val >> 16) & 0xFF);
|
2017-01-22 19:34:59 +00:00
|
|
|
}
|
2017-01-17 01:29:25 +00:00
|
|
|
return;
|
|
|
|
case 0x040001A4:
|
2017-01-31 16:34:17 +00:00
|
|
|
if (!(ExMemCnt[0] & (1<<11))) NDSCart::WriteROMCnt(val);
|
2017-01-17 01:29:25 +00:00
|
|
|
return;
|
|
|
|
|
2017-03-30 23:50:01 +00:00
|
|
|
case 0x040001A8:
|
|
|
|
NDSCart::ROMCommand[0] = val & 0xFF;
|
|
|
|
NDSCart::ROMCommand[1] = (val >> 8) & 0xFF;
|
|
|
|
NDSCart::ROMCommand[2] = (val >> 16) & 0xFF;
|
|
|
|
NDSCart::ROMCommand[3] = val >> 24;
|
|
|
|
return;
|
|
|
|
case 0x040001AC:
|
|
|
|
NDSCart::ROMCommand[4] = val & 0xFF;
|
|
|
|
NDSCart::ROMCommand[5] = (val >> 8) & 0xFF;
|
|
|
|
NDSCart::ROMCommand[6] = (val >> 16) & 0xFF;
|
|
|
|
NDSCart::ROMCommand[7] = val >> 24;
|
|
|
|
return;
|
|
|
|
|
2017-01-22 19:34:59 +00:00
|
|
|
case 0x040001B0: *(u32*)&ROMSeed0[0] = val; return;
|
|
|
|
case 0x040001B4: *(u32*)&ROMSeed1[0] = val; return;
|
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
case 0x04000208: IME[0] = val & 0x1; return;
|
2017-02-07 23:52:37 +00:00
|
|
|
case 0x04000210: IE[0] = val; return;
|
|
|
|
case 0x04000214: IF[0] &= ~val; GPU3D::CheckFIFOIRQ(); return;
|
2017-01-17 01:29:25 +00:00
|
|
|
|
|
|
|
case 0x04000240:
|
2017-01-18 03:03:19 +00:00
|
|
|
GPU::MapVRAM_AB(0, val & 0xFF);
|
|
|
|
GPU::MapVRAM_AB(1, (val >> 8) & 0xFF);
|
|
|
|
GPU::MapVRAM_CD(2, (val >> 16) & 0xFF);
|
|
|
|
GPU::MapVRAM_CD(3, val >> 24);
|
2017-01-17 01:29:25 +00:00
|
|
|
return;
|
|
|
|
case 0x04000244:
|
2017-01-18 03:03:19 +00:00
|
|
|
GPU::MapVRAM_E(4, val & 0xFF);
|
|
|
|
GPU::MapVRAM_FG(5, (val >> 8) & 0xFF);
|
|
|
|
GPU::MapVRAM_FG(6, (val >> 16) & 0xFF);
|
2017-01-17 01:29:25 +00:00
|
|
|
MapSharedWRAM(val >> 24);
|
|
|
|
return;
|
|
|
|
case 0x04000248:
|
2017-01-18 03:03:19 +00:00
|
|
|
GPU::MapVRAM_H(7, val & 0xFF);
|
|
|
|
GPU::MapVRAM_I(8, (val >> 8) & 0xFF);
|
2017-01-17 01:29:25 +00:00
|
|
|
return;
|
2017-01-18 01:20:45 +00:00
|
|
|
|
2017-04-28 15:23:18 +00:00
|
|
|
case 0x04000280: DivCnt = val; StartDiv(); return;
|
|
|
|
|
|
|
|
case 0x040002B0: SqrtCnt = val; StartSqrt(); return;
|
|
|
|
|
2017-01-18 01:20:45 +00:00
|
|
|
case 0x04000290: DivNumerator[0] = val; StartDiv(); return;
|
|
|
|
case 0x04000294: DivNumerator[1] = val; StartDiv(); return;
|
|
|
|
case 0x04000298: DivDenominator[0] = val; StartDiv(); return;
|
|
|
|
case 0x0400029C: DivDenominator[1] = val; StartDiv(); return;
|
2017-01-31 20:53:45 +00:00
|
|
|
|
|
|
|
case 0x040002B8: SqrtVal[0] = val; StartSqrt(); return;
|
|
|
|
case 0x040002BC: SqrtVal[1] = val; StartSqrt(); return;
|
2017-03-23 16:14:48 +00:00
|
|
|
|
|
|
|
case 0x04000304:
|
|
|
|
PowerControl9 = val & 0xFFFF;
|
|
|
|
GPU::DisplaySwap(PowerControl9>>15);
|
|
|
|
return;
|
2017-01-17 01:29:25 +00:00
|
|
|
}
|
|
|
|
|
2017-01-18 03:03:19 +00:00
|
|
|
if (addr >= 0x04000000 && addr < 0x04000060)
|
|
|
|
{
|
|
|
|
GPU::GPU2D_A->Write32(addr, val);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (addr >= 0x04001000 && addr < 0x04001060)
|
|
|
|
{
|
|
|
|
GPU::GPU2D_B->Write32(addr, val);
|
|
|
|
return;
|
|
|
|
}
|
2017-01-31 17:41:31 +00:00
|
|
|
if (addr >= 0x04000320 && addr < 0x040006A4)
|
|
|
|
{
|
2017-02-07 22:31:21 +00:00
|
|
|
GPU3D::Write32(addr, val);
|
2017-01-31 17:41:31 +00:00
|
|
|
return;
|
|
|
|
}
|
2017-01-18 03:03:19 +00:00
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
printf("unknown ARM9 IO write32 %08X %08X\n", addr, val);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
u8 ARM7IORead8(u32 addr)
|
|
|
|
{
|
|
|
|
switch (addr)
|
|
|
|
{
|
2017-04-23 23:30:30 +00:00
|
|
|
case 0x04000130: return KeyInput & 0xFF;
|
|
|
|
case 0x04000131: return (KeyInput >> 8) & 0xFF;
|
2017-06-04 20:34:31 +00:00
|
|
|
case 0x04000132: return KeyCnt & 0xFF;
|
|
|
|
case 0x04000133: return KeyCnt >> 8;
|
|
|
|
case 0x04000134: return RCnt & 0xFF;
|
|
|
|
case 0x04000135: return RCnt >> 8;
|
2017-04-23 23:30:30 +00:00
|
|
|
case 0x04000136: return (KeyInput >> 16) & 0xFF;
|
|
|
|
case 0x04000137: return KeyInput >> 24;
|
|
|
|
|
2017-01-20 00:18:30 +00:00
|
|
|
case 0x04000138: return RTC::Read() & 0xFF;
|
2017-01-17 01:29:25 +00:00
|
|
|
|
2017-01-31 16:34:17 +00:00
|
|
|
case 0x040001A2: return NDSCart::ReadSPIData();
|
|
|
|
|
2017-03-30 23:50:01 +00:00
|
|
|
case 0x040001A8: return NDSCart::ROMCommand[0];
|
|
|
|
case 0x040001A9: return NDSCart::ROMCommand[1];
|
|
|
|
case 0x040001AA: return NDSCart::ROMCommand[2];
|
|
|
|
case 0x040001AB: return NDSCart::ROMCommand[3];
|
|
|
|
case 0x040001AC: return NDSCart::ROMCommand[4];
|
|
|
|
case 0x040001AD: return NDSCart::ROMCommand[5];
|
|
|
|
case 0x040001AE: return NDSCart::ROMCommand[6];
|
|
|
|
case 0x040001AF: return NDSCart::ROMCommand[7];
|
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
case 0x040001C2: return SPI::ReadData();
|
|
|
|
|
|
|
|
case 0x04000208: return IME[1];
|
|
|
|
|
2017-01-18 03:03:19 +00:00
|
|
|
case 0x04000240: return GPU::VRAMSTAT;
|
2017-01-17 01:29:25 +00:00
|
|
|
case 0x04000241: return WRAMCnt;
|
|
|
|
|
2017-01-18 01:20:45 +00:00
|
|
|
case 0x04000300: return PostFlag7;
|
2017-01-17 01:29:25 +00:00
|
|
|
}
|
2017-01-20 00:18:30 +00:00
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
if (addr >= 0x04000400 && addr < 0x04000520)
|
|
|
|
{
|
2017-04-06 17:44:34 +00:00
|
|
|
return SPU::Read8(addr);
|
2017-01-17 01:29:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
printf("unknown ARM7 IO read8 %08X\n", addr);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
u16 ARM7IORead16(u32 addr)
|
|
|
|
{
|
|
|
|
switch (addr)
|
|
|
|
{
|
2017-01-18 03:03:19 +00:00
|
|
|
case 0x04000004: return GPU::DispStat[1];
|
|
|
|
case 0x04000006: return GPU::VCount;
|
2017-01-17 01:29:25 +00:00
|
|
|
|
2017-01-31 16:34:17 +00:00
|
|
|
case 0x040000B8: return DMAs[4]->Cnt & 0xFFFF;
|
|
|
|
case 0x040000BA: return DMAs[4]->Cnt >> 16;
|
|
|
|
case 0x040000C4: return DMAs[5]->Cnt & 0xFFFF;
|
|
|
|
case 0x040000C6: return DMAs[5]->Cnt >> 16;
|
|
|
|
case 0x040000D0: return DMAs[6]->Cnt & 0xFFFF;
|
|
|
|
case 0x040000D2: return DMAs[6]->Cnt >> 16;
|
|
|
|
case 0x040000DC: return DMAs[7]->Cnt & 0xFFFF;
|
|
|
|
case 0x040000DE: return DMAs[7]->Cnt >> 16;
|
|
|
|
|
2017-01-31 02:54:51 +00:00
|
|
|
case 0x04000100: return TimerGetCounter(4);
|
|
|
|
case 0x04000102: return Timers[4].Cnt;
|
|
|
|
case 0x04000104: return TimerGetCounter(5);
|
|
|
|
case 0x04000106: return Timers[5].Cnt;
|
|
|
|
case 0x04000108: return TimerGetCounter(6);
|
|
|
|
case 0x0400010A: return Timers[6].Cnt;
|
|
|
|
case 0x0400010C: return TimerGetCounter(7);
|
|
|
|
case 0x0400010E: return Timers[7].Cnt;
|
2017-01-17 01:29:25 +00:00
|
|
|
|
|
|
|
case 0x04000130: return KeyInput & 0xFFFF;
|
2017-06-04 20:34:31 +00:00
|
|
|
case 0x04000132: return KeyCnt;
|
|
|
|
case 0x04000134: return RCnt;
|
2017-01-17 01:29:25 +00:00
|
|
|
case 0x04000136: return KeyInput >> 16;
|
|
|
|
|
2017-01-20 00:18:30 +00:00
|
|
|
case 0x04000138: return RTC::Read();
|
2017-01-17 01:29:25 +00:00
|
|
|
|
|
|
|
case 0x04000180: return IPCSync7;
|
|
|
|
case 0x04000184:
|
|
|
|
{
|
|
|
|
u16 val = IPCFIFOCnt7;
|
|
|
|
if (IPCFIFO7->IsEmpty()) val |= 0x0001;
|
|
|
|
else if (IPCFIFO7->IsFull()) val |= 0x0002;
|
|
|
|
if (IPCFIFO9->IsEmpty()) val |= 0x0100;
|
|
|
|
else if (IPCFIFO9->IsFull()) val |= 0x0200;
|
|
|
|
return val;
|
|
|
|
}
|
|
|
|
|
2017-01-22 19:34:59 +00:00
|
|
|
case 0x040001A0: return NDSCart::SPICnt;
|
2017-01-31 16:34:17 +00:00
|
|
|
case 0x040001A2: return NDSCart::ReadSPIData();
|
2017-01-22 19:34:59 +00:00
|
|
|
|
2017-03-30 23:50:01 +00:00
|
|
|
case 0x040001A8: return NDSCart::ROMCommand[0] |
|
|
|
|
(NDSCart::ROMCommand[1] << 8);
|
|
|
|
case 0x040001AA: return NDSCart::ROMCommand[2] |
|
|
|
|
(NDSCart::ROMCommand[3] << 8);
|
|
|
|
case 0x040001AC: return NDSCart::ROMCommand[4] |
|
|
|
|
(NDSCart::ROMCommand[5] << 8);
|
|
|
|
case 0x040001AE: return NDSCart::ROMCommand[6] |
|
|
|
|
(NDSCart::ROMCommand[7] << 8);
|
|
|
|
|
2017-01-31 16:34:17 +00:00
|
|
|
case 0x040001C0: return SPI::Cnt;
|
2017-01-17 01:29:25 +00:00
|
|
|
case 0x040001C2: return SPI::ReadData();
|
|
|
|
|
2017-01-22 19:34:59 +00:00
|
|
|
case 0x04000204: return ExMemCnt[1];
|
2017-01-17 01:29:25 +00:00
|
|
|
case 0x04000208: return IME[1];
|
2017-06-23 23:21:09 +00:00
|
|
|
case 0x04000210: return IE[1] & 0xFFFF;
|
|
|
|
case 0x04000212: return IE[1] >> 16;
|
2017-01-17 01:29:25 +00:00
|
|
|
|
2017-01-18 01:20:45 +00:00
|
|
|
case 0x04000300: return PostFlag7;
|
2017-01-17 01:29:25 +00:00
|
|
|
case 0x04000304: return PowerControl7;
|
2017-02-05 16:15:17 +00:00
|
|
|
case 0x04000308: return ARM7BIOSProt;
|
2017-01-17 01:29:25 +00:00
|
|
|
}
|
|
|
|
|
2017-01-20 00:18:30 +00:00
|
|
|
if (addr >= 0x04000400 && addr < 0x04000520)
|
|
|
|
{
|
2017-04-06 17:44:34 +00:00
|
|
|
return SPU::Read16(addr);
|
2017-01-20 00:18:30 +00:00
|
|
|
}
|
|
|
|
|
2017-01-18 16:57:12 +00:00
|
|
|
printf("unknown ARM7 IO read16 %08X %08X\n", addr, ARM9->R[15]);
|
2017-01-17 01:29:25 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
u32 ARM7IORead32(u32 addr)
|
|
|
|
{
|
|
|
|
switch (addr)
|
|
|
|
{
|
2017-01-18 03:03:19 +00:00
|
|
|
case 0x04000004: return GPU::DispStat[1] | (GPU::VCount << 16);
|
2017-01-17 01:29:25 +00:00
|
|
|
|
2017-01-18 00:33:06 +00:00
|
|
|
case 0x040000B0: return DMAs[4]->SrcAddr;
|
|
|
|
case 0x040000B4: return DMAs[4]->DstAddr;
|
|
|
|
case 0x040000B8: return DMAs[4]->Cnt;
|
|
|
|
case 0x040000BC: return DMAs[5]->SrcAddr;
|
|
|
|
case 0x040000C0: return DMAs[5]->DstAddr;
|
|
|
|
case 0x040000C4: return DMAs[5]->Cnt;
|
|
|
|
case 0x040000C8: return DMAs[6]->SrcAddr;
|
|
|
|
case 0x040000CC: return DMAs[6]->DstAddr;
|
|
|
|
case 0x040000D0: return DMAs[6]->Cnt;
|
|
|
|
case 0x040000D4: return DMAs[7]->SrcAddr;
|
|
|
|
case 0x040000D8: return DMAs[7]->DstAddr;
|
|
|
|
case 0x040000DC: return DMAs[7]->Cnt;
|
|
|
|
|
2017-01-31 02:54:51 +00:00
|
|
|
case 0x04000100: return TimerGetCounter(4) | (Timers[4].Cnt << 16);
|
|
|
|
case 0x04000104: return TimerGetCounter(5) | (Timers[5].Cnt << 16);
|
|
|
|
case 0x04000108: return TimerGetCounter(6) | (Timers[6].Cnt << 16);
|
|
|
|
case 0x0400010C: return TimerGetCounter(7) | (Timers[7].Cnt << 16);
|
2017-01-17 01:29:25 +00:00
|
|
|
|
2017-06-04 20:34:31 +00:00
|
|
|
case 0x04000130: return (KeyInput & 0xFFFF) | (KeyCnt << 16);
|
|
|
|
case 0x04000134: return RCnt | (KeyCnt & 0xFFFF0000);
|
|
|
|
case 0x04000138: return RTC::Read();
|
|
|
|
|
2017-01-31 16:34:17 +00:00
|
|
|
case 0x040001A0: return NDSCart::SPICnt | (NDSCart::ReadSPIData() << 16);
|
2017-01-22 19:34:59 +00:00
|
|
|
case 0x040001A4: return NDSCart::ROMCnt;
|
2017-01-17 01:29:25 +00:00
|
|
|
|
2017-03-30 23:50:01 +00:00
|
|
|
case 0x040001A8: return NDSCart::ROMCommand[0] |
|
|
|
|
(NDSCart::ROMCommand[1] << 8) |
|
|
|
|
(NDSCart::ROMCommand[2] << 16) |
|
|
|
|
(NDSCart::ROMCommand[3] << 24);
|
|
|
|
case 0x040001AC: return NDSCart::ROMCommand[4] |
|
|
|
|
(NDSCart::ROMCommand[5] << 8) |
|
|
|
|
(NDSCart::ROMCommand[6] << 16) |
|
|
|
|
(NDSCart::ROMCommand[7] << 24);
|
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
case 0x040001C0:
|
2017-01-31 16:34:17 +00:00
|
|
|
return SPI::Cnt | (SPI::ReadData() << 16);
|
2017-01-17 01:29:25 +00:00
|
|
|
|
|
|
|
case 0x04000208: return IME[1];
|
|
|
|
case 0x04000210: return IE[1];
|
|
|
|
case 0x04000214: return IF[1];
|
|
|
|
|
2017-10-02 02:57:23 +00:00
|
|
|
case 0x04000308: return ARM7BIOSProt;
|
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
case 0x04100000:
|
|
|
|
if (IPCFIFOCnt7 & 0x8000)
|
|
|
|
{
|
|
|
|
u32 ret;
|
|
|
|
if (IPCFIFO9->IsEmpty())
|
|
|
|
{
|
|
|
|
IPCFIFOCnt7 |= 0x4000;
|
|
|
|
ret = IPCFIFO9->Peek();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ret = IPCFIFO9->Read();
|
|
|
|
|
|
|
|
if (IPCFIFO9->IsEmpty() && (IPCFIFOCnt9 & 0x0004))
|
2017-03-02 23:48:26 +00:00
|
|
|
SetIRQ(0, IRQ_IPCSendDone);
|
2017-01-17 01:29:25 +00:00
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return IPCFIFO9->Peek();
|
|
|
|
|
2017-01-22 19:34:59 +00:00
|
|
|
case 0x04100010:
|
2017-01-31 16:34:17 +00:00
|
|
|
if (ExMemCnt[0] & (1<<11)) return NDSCart::ReadROMData();
|
2017-01-22 19:34:59 +00:00
|
|
|
return 0;
|
2017-01-17 01:29:25 +00:00
|
|
|
}
|
|
|
|
|
2017-01-20 00:18:30 +00:00
|
|
|
if (addr >= 0x04000400 && addr < 0x04000520)
|
|
|
|
{
|
2017-04-06 17:44:34 +00:00
|
|
|
return SPU::Read32(addr);
|
2017-01-20 00:18:30 +00:00
|
|
|
}
|
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
printf("unknown ARM7 IO read32 %08X\n", addr);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ARM7IOWrite8(u32 addr, u8 val)
|
|
|
|
{
|
|
|
|
switch (addr)
|
|
|
|
{
|
2017-06-04 20:34:31 +00:00
|
|
|
case 0x04000132:
|
|
|
|
KeyCnt = (KeyCnt & 0xFF00) | val;
|
|
|
|
return;
|
|
|
|
case 0x04000133:
|
|
|
|
KeyCnt = (KeyCnt & 0x00FF) | (val << 8);
|
|
|
|
return;
|
|
|
|
case 0x04000134:
|
|
|
|
RCnt = (RCnt & 0xFF00) | val;
|
|
|
|
return;
|
|
|
|
case 0x04000135:
|
|
|
|
RCnt = (RCnt & 0x00FF) | (val << 8);
|
|
|
|
return;
|
|
|
|
|
2017-01-20 00:18:30 +00:00
|
|
|
case 0x04000138: RTC::Write(val, true); return;
|
2017-01-17 01:29:25 +00:00
|
|
|
|
|
|
|
case 0x040001A0:
|
2017-01-22 19:34:59 +00:00
|
|
|
if (ExMemCnt[0] & (1<<11))
|
|
|
|
{
|
2017-01-31 16:34:17 +00:00
|
|
|
NDSCart::WriteSPICnt((NDSCart::SPICnt & 0xFF00) | val);
|
2017-01-22 19:34:59 +00:00
|
|
|
}
|
2017-01-17 01:29:25 +00:00
|
|
|
return;
|
|
|
|
case 0x040001A1:
|
2017-01-22 19:34:59 +00:00
|
|
|
if (ExMemCnt[0] & (1<<11))
|
|
|
|
{
|
2017-01-31 16:34:17 +00:00
|
|
|
NDSCart::WriteSPICnt((NDSCart::SPICnt & 0x00FF) | (val << 8));
|
2017-01-22 19:34:59 +00:00
|
|
|
}
|
2017-01-17 01:29:25 +00:00
|
|
|
return;
|
2017-01-31 02:54:51 +00:00
|
|
|
case 0x040001A2:
|
2017-01-31 16:34:17 +00:00
|
|
|
NDSCart::WriteSPIData(val);
|
2017-01-31 02:54:51 +00:00
|
|
|
return;
|
2017-01-17 01:29:25 +00:00
|
|
|
|
2017-01-22 19:34:59 +00:00
|
|
|
case 0x040001A8: NDSCart::ROMCommand[0] = val; return;
|
|
|
|
case 0x040001A9: NDSCart::ROMCommand[1] = val; return;
|
|
|
|
case 0x040001AA: NDSCart::ROMCommand[2] = val; return;
|
|
|
|
case 0x040001AB: NDSCart::ROMCommand[3] = val; return;
|
|
|
|
case 0x040001AC: NDSCart::ROMCommand[4] = val; return;
|
|
|
|
case 0x040001AD: NDSCart::ROMCommand[5] = val; return;
|
|
|
|
case 0x040001AE: NDSCart::ROMCommand[6] = val; return;
|
|
|
|
case 0x040001AF: NDSCart::ROMCommand[7] = val; return;
|
2017-01-17 01:29:25 +00:00
|
|
|
|
|
|
|
case 0x040001C2:
|
|
|
|
SPI::WriteData(val);
|
|
|
|
return;
|
|
|
|
|
|
|
|
case 0x04000208: IME[1] = val & 0x1; return;
|
|
|
|
|
2017-01-18 01:20:45 +00:00
|
|
|
case 0x04000300:
|
|
|
|
if (ARM7->R[15] >= 0x4000)
|
|
|
|
return;
|
|
|
|
if (!(PostFlag7 & 0x01))
|
|
|
|
PostFlag7 = val & 0x01;
|
|
|
|
return;
|
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
case 0x04000301:
|
|
|
|
if (val == 0x80) ARM7->Halt(1);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-01-20 00:18:30 +00:00
|
|
|
if (addr >= 0x04000400 && addr < 0x04000520)
|
|
|
|
{
|
2017-04-06 17:44:34 +00:00
|
|
|
SPU::Write8(addr, val);
|
2017-01-20 00:18:30 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
printf("unknown ARM7 IO write8 %08X %02X\n", addr, val);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ARM7IOWrite16(u32 addr, u16 val)
|
|
|
|
{
|
|
|
|
switch (addr)
|
|
|
|
{
|
2017-01-18 03:03:19 +00:00
|
|
|
case 0x04000004: GPU::SetDispStat(1, val); return;
|
2017-05-10 00:21:02 +00:00
|
|
|
case 0x04000006: GPU::SetVCount(val); return;
|
2017-01-17 01:29:25 +00:00
|
|
|
|
2017-01-31 16:34:17 +00:00
|
|
|
case 0x040000B8: DMAs[4]->WriteCnt((DMAs[4]->Cnt & 0xFFFF0000) | val); return;
|
|
|
|
case 0x040000BA: DMAs[4]->WriteCnt((DMAs[4]->Cnt & 0x0000FFFF) | (val << 16)); return;
|
|
|
|
case 0x040000C4: DMAs[5]->WriteCnt((DMAs[5]->Cnt & 0xFFFF0000) | val); return;
|
|
|
|
case 0x040000C6: DMAs[5]->WriteCnt((DMAs[5]->Cnt & 0x0000FFFF) | (val << 16)); return;
|
|
|
|
case 0x040000D0: DMAs[6]->WriteCnt((DMAs[6]->Cnt & 0xFFFF0000) | val); return;
|
|
|
|
case 0x040000D2: DMAs[6]->WriteCnt((DMAs[6]->Cnt & 0x0000FFFF) | (val << 16)); return;
|
|
|
|
case 0x040000DC: DMAs[7]->WriteCnt((DMAs[7]->Cnt & 0xFFFF0000) | val); return;
|
|
|
|
case 0x040000DE: DMAs[7]->WriteCnt((DMAs[7]->Cnt & 0x0000FFFF) | (val << 16)); return;
|
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
case 0x04000100: Timers[4].Reload = val; return;
|
|
|
|
case 0x04000102: TimerStart(4, val); return;
|
|
|
|
case 0x04000104: Timers[5].Reload = val; return;
|
|
|
|
case 0x04000106: TimerStart(5, val); return;
|
|
|
|
case 0x04000108: Timers[6].Reload = val; return;
|
|
|
|
case 0x0400010A: TimerStart(6, val); return;
|
|
|
|
case 0x0400010C: Timers[7].Reload = val; return;
|
|
|
|
case 0x0400010E: TimerStart(7, val); return;
|
|
|
|
|
2017-06-04 20:34:31 +00:00
|
|
|
case 0x04000132: KeyCnt = val; return;
|
|
|
|
case 0x04000134: RCnt = val; return;
|
2017-01-17 01:29:25 +00:00
|
|
|
|
2017-01-20 00:18:30 +00:00
|
|
|
case 0x04000138: RTC::Write(val, false); return;
|
2017-01-17 01:29:25 +00:00
|
|
|
|
|
|
|
case 0x04000180:
|
|
|
|
IPCSync9 &= 0xFFF0;
|
|
|
|
IPCSync9 |= ((val & 0x0F00) >> 8);
|
|
|
|
IPCSync7 &= 0xB0FF;
|
|
|
|
IPCSync7 |= (val & 0x4F00);
|
|
|
|
if ((val & 0x2000) && (IPCSync9 & 0x4000))
|
|
|
|
{
|
2017-03-02 23:48:26 +00:00
|
|
|
SetIRQ(0, IRQ_IPCSync);
|
2017-01-17 01:29:25 +00:00
|
|
|
}
|
|
|
|
return;
|
|
|
|
|
|
|
|
case 0x04000184:
|
|
|
|
if (val & 0x0008)
|
|
|
|
IPCFIFO7->Clear();
|
|
|
|
if ((val & 0x0004) && (!(IPCFIFOCnt7 & 0x0004)) && IPCFIFO7->IsEmpty())
|
2017-03-02 23:48:26 +00:00
|
|
|
SetIRQ(1, IRQ_IPCSendDone);
|
2017-01-17 01:29:25 +00:00
|
|
|
if ((val & 0x0400) && (!(IPCFIFOCnt7 & 0x0400)) && (!IPCFIFO9->IsEmpty()))
|
2017-03-02 23:48:26 +00:00
|
|
|
SetIRQ(1, IRQ_IPCRecv);
|
2017-01-17 01:29:25 +00:00
|
|
|
if (val & 0x4000)
|
|
|
|
IPCFIFOCnt7 &= ~0x4000;
|
|
|
|
IPCFIFOCnt7 = val & 0x8404;
|
|
|
|
return;
|
|
|
|
|
|
|
|
case 0x040001A0:
|
2017-01-22 19:34:59 +00:00
|
|
|
if (ExMemCnt[0] & (1<<11))
|
2017-01-31 16:34:17 +00:00
|
|
|
NDSCart::WriteSPICnt(val);
|
2017-01-17 01:29:25 +00:00
|
|
|
return;
|
2017-01-31 02:54:51 +00:00
|
|
|
case 0x040001A2:
|
2017-01-31 16:34:17 +00:00
|
|
|
NDSCart::WriteSPIData(val & 0xFF);
|
2017-01-31 02:54:51 +00:00
|
|
|
return;
|
2017-01-17 01:29:25 +00:00
|
|
|
|
2017-03-30 23:50:01 +00:00
|
|
|
case 0x040001A8:
|
|
|
|
NDSCart::ROMCommand[0] = val & 0xFF;
|
|
|
|
NDSCart::ROMCommand[1] = val >> 8;
|
|
|
|
return;
|
|
|
|
case 0x040001AA:
|
|
|
|
NDSCart::ROMCommand[2] = val & 0xFF;
|
|
|
|
NDSCart::ROMCommand[3] = val >> 8;
|
|
|
|
return;
|
|
|
|
case 0x040001AC:
|
|
|
|
NDSCart::ROMCommand[4] = val & 0xFF;
|
|
|
|
NDSCart::ROMCommand[5] = val >> 8;
|
|
|
|
return;
|
|
|
|
case 0x040001AE:
|
|
|
|
NDSCart::ROMCommand[6] = val & 0xFF;
|
|
|
|
NDSCart::ROMCommand[7] = val >> 8;
|
|
|
|
return;
|
|
|
|
|
2017-01-22 19:34:59 +00:00
|
|
|
case 0x040001B8: ROMSeed0[12] = val & 0x7F; return;
|
|
|
|
case 0x040001BA: ROMSeed1[12] = val & 0x7F; return;
|
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
case 0x040001C0:
|
|
|
|
SPI::WriteCnt(val);
|
|
|
|
return;
|
|
|
|
case 0x040001C2:
|
|
|
|
SPI::WriteData(val & 0xFF);
|
|
|
|
return;
|
|
|
|
|
2017-01-22 19:34:59 +00:00
|
|
|
case 0x04000204:
|
|
|
|
ExMemCnt[1] = (ExMemCnt[1] & 0xFF80) | (val & 0x007F);
|
|
|
|
return;
|
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
case 0x04000208: IME[1] = val & 0x1; return;
|
2017-06-23 23:21:09 +00:00
|
|
|
case 0x04000210: IE[1] = (IE[1] & 0xFFFF0000) | val; return;
|
|
|
|
case 0x04000212: IE[1] = (IE[1] & 0x0000FFFF) | (val << 16); return;
|
|
|
|
// TODO: what happens when writing to IF this way??
|
2017-01-17 01:29:25 +00:00
|
|
|
|
2017-01-18 01:20:45 +00:00
|
|
|
case 0x04000300:
|
|
|
|
if (ARM7->R[15] >= 0x4000)
|
|
|
|
return;
|
|
|
|
if (!(PostFlag7 & 0x01))
|
|
|
|
PostFlag7 = val & 0x01;
|
|
|
|
return;
|
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
case 0x04000304: PowerControl7 = val; return;
|
|
|
|
|
2017-02-05 16:15:17 +00:00
|
|
|
case 0x04000308:
|
|
|
|
if (ARM7BIOSProt == 0)
|
2017-10-02 02:57:23 +00:00
|
|
|
ARM7BIOSProt = val & 0xFFFE;
|
2017-02-05 16:15:17 +00:00
|
|
|
return;
|
2017-01-17 01:29:25 +00:00
|
|
|
}
|
|
|
|
|
2017-01-20 00:18:30 +00:00
|
|
|
if (addr >= 0x04000400 && addr < 0x04000520)
|
|
|
|
{
|
2017-04-06 17:44:34 +00:00
|
|
|
SPU::Write16(addr, val);
|
2017-01-20 00:18:30 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
printf("unknown ARM7 IO write16 %08X %04X\n", addr, val);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ARM7IOWrite32(u32 addr, u32 val)
|
|
|
|
{
|
|
|
|
switch (addr)
|
|
|
|
{
|
2017-01-18 00:33:06 +00:00
|
|
|
case 0x040000B0: DMAs[4]->SrcAddr = val; return;
|
|
|
|
case 0x040000B4: DMAs[4]->DstAddr = val; return;
|
|
|
|
case 0x040000B8: DMAs[4]->WriteCnt(val); return;
|
|
|
|
case 0x040000BC: DMAs[5]->SrcAddr = val; return;
|
|
|
|
case 0x040000C0: DMAs[5]->DstAddr = val; return;
|
|
|
|
case 0x040000C4: DMAs[5]->WriteCnt(val); return;
|
|
|
|
case 0x040000C8: DMAs[6]->SrcAddr = val; return;
|
|
|
|
case 0x040000CC: DMAs[6]->DstAddr = val; return;
|
|
|
|
case 0x040000D0: DMAs[6]->WriteCnt(val); return;
|
|
|
|
case 0x040000D4: DMAs[7]->SrcAddr = val; return;
|
|
|
|
case 0x040000D8: DMAs[7]->DstAddr = val; return;
|
2017-01-31 02:54:51 +00:00
|
|
|
case 0x040000DC: DMAs[7]->WriteCnt(val); return;
|
2017-01-18 00:33:06 +00:00
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
case 0x04000100:
|
|
|
|
Timers[4].Reload = val & 0xFFFF;
|
|
|
|
TimerStart(4, val>>16);
|
|
|
|
return;
|
|
|
|
case 0x04000104:
|
|
|
|
Timers[5].Reload = val & 0xFFFF;
|
|
|
|
TimerStart(5, val>>16);
|
|
|
|
return;
|
|
|
|
case 0x04000108:
|
|
|
|
Timers[6].Reload = val & 0xFFFF;
|
|
|
|
TimerStart(6, val>>16);
|
|
|
|
return;
|
|
|
|
case 0x0400010C:
|
|
|
|
Timers[7].Reload = val & 0xFFFF;
|
|
|
|
TimerStart(7, val>>16);
|
|
|
|
return;
|
|
|
|
|
2017-06-04 20:34:31 +00:00
|
|
|
case 0x04000130: KeyCnt = val >> 16; return;
|
|
|
|
case 0x04000134: RCnt = val & 0xFFFF; return;
|
|
|
|
case 0x04000138: RTC::Write(val & 0xFFFF, false); return;
|
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
case 0x04000188:
|
|
|
|
if (IPCFIFOCnt7 & 0x8000)
|
2017-06-18 14:26:59 +00:00
|
|
|
{
|
2017-01-17 01:29:25 +00:00
|
|
|
if (IPCFIFO7->IsFull())
|
|
|
|
IPCFIFOCnt7 |= 0x4000;
|
|
|
|
else
|
|
|
|
{
|
2017-01-17 04:29:38 +00:00
|
|
|
bool wasempty = IPCFIFO7->IsEmpty();
|
2017-01-17 01:29:25 +00:00
|
|
|
IPCFIFO7->Write(val);
|
2017-01-17 04:29:38 +00:00
|
|
|
if ((IPCFIFOCnt9 & 0x0400) && wasempty)
|
2017-03-02 23:48:26 +00:00
|
|
|
SetIRQ(0, IRQ_IPCRecv);
|
2017-01-17 01:29:25 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
|
|
|
|
case 0x040001A0:
|
2017-01-22 19:34:59 +00:00
|
|
|
if (ExMemCnt[0] & (1<<11))
|
|
|
|
{
|
2017-01-31 16:34:17 +00:00
|
|
|
NDSCart::WriteSPICnt(val & 0xFFFF);
|
|
|
|
NDSCart::WriteSPIData((val >> 16) & 0xFF);
|
2017-01-22 19:34:59 +00:00
|
|
|
}
|
2017-01-17 01:29:25 +00:00
|
|
|
return;
|
|
|
|
case 0x040001A4:
|
2017-01-31 16:34:17 +00:00
|
|
|
if (ExMemCnt[0] & (1<<11)) NDSCart::WriteROMCnt(val);
|
2017-01-17 01:29:25 +00:00
|
|
|
return;
|
|
|
|
|
2017-03-30 23:50:01 +00:00
|
|
|
case 0x040001A8:
|
|
|
|
NDSCart::ROMCommand[0] = val & 0xFF;
|
|
|
|
NDSCart::ROMCommand[1] = (val >> 8) & 0xFF;
|
|
|
|
NDSCart::ROMCommand[2] = (val >> 16) & 0xFF;
|
|
|
|
NDSCart::ROMCommand[3] = val >> 24;
|
|
|
|
return;
|
|
|
|
case 0x040001AC:
|
|
|
|
NDSCart::ROMCommand[4] = val & 0xFF;
|
|
|
|
NDSCart::ROMCommand[5] = (val >> 8) & 0xFF;
|
|
|
|
NDSCart::ROMCommand[6] = (val >> 16) & 0xFF;
|
|
|
|
NDSCart::ROMCommand[7] = val >> 24;
|
|
|
|
return;
|
|
|
|
|
2017-01-22 19:34:59 +00:00
|
|
|
case 0x040001B0: *(u32*)&ROMSeed0[8] = val; return;
|
|
|
|
case 0x040001B4: *(u32*)&ROMSeed1[8] = val; return;
|
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
case 0x04000208: IME[1] = val & 0x1; return;
|
|
|
|
case 0x04000210: IE[1] = val; return;
|
|
|
|
case 0x04000214: IF[1] &= ~val; return;
|
2017-10-02 02:57:23 +00:00
|
|
|
|
|
|
|
case 0x04000308:
|
|
|
|
if (ARM7BIOSProt == 0)
|
|
|
|
ARM7BIOSProt = val & 0xFFFE;
|
|
|
|
return;
|
2017-01-17 01:29:25 +00:00
|
|
|
}
|
|
|
|
|
2017-01-20 00:18:30 +00:00
|
|
|
if (addr >= 0x04000400 && addr < 0x04000520)
|
|
|
|
{
|
2017-04-06 17:44:34 +00:00
|
|
|
SPU::Write32(addr, val);
|
2017-01-20 00:18:30 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-01-17 01:29:25 +00:00
|
|
|
printf("unknown ARM7 IO write32 %08X %08X\n", addr, val);
|
|
|
|
}
|
|
|
|
|
2016-05-16 15:48:40 +00:00
|
|
|
}
|