2016-12-05 17:02:29 +00:00
|
|
|
/*
|
2023-11-03 23:21:46 +00:00
|
|
|
Copyright 2016-2023 melonDS team
|
2016-12-05 17:02:29 +00:00
|
|
|
|
|
|
|
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-11-03 00:38:58 +00:00
|
|
|
|
|
|
|
#ifndef ARM_H
|
|
|
|
#define ARM_H
|
|
|
|
|
2018-12-04 16:54:10 +00:00
|
|
|
#include <algorithm>
|
|
|
|
|
2016-11-03 00:38:58 +00:00
|
|
|
#include "types.h"
|
|
|
|
#include "NDS.h"
|
|
|
|
|
2023-10-22 13:35:31 +00:00
|
|
|
#ifdef GDBSTUB_ENABLED
|
|
|
|
#include "debug/GdbStub.h"
|
|
|
|
#endif
|
|
|
|
|
2023-11-25 17:32:09 +00:00
|
|
|
namespace melonDS
|
|
|
|
{
|
2020-09-04 18:37:14 +00:00
|
|
|
inline u32 ROR(u32 x, u32 n)
|
|
|
|
{
|
|
|
|
return (x >> (n&0x1F)) | (x << ((32-n)&0x1F));
|
|
|
|
}
|
2016-11-24 17:31:49 +00:00
|
|
|
|
2018-12-04 16:54:10 +00:00
|
|
|
enum
|
|
|
|
{
|
|
|
|
RWFlags_Nonseq = (1<<5),
|
|
|
|
RWFlags_ForceUser = (1<<21),
|
|
|
|
};
|
|
|
|
|
2020-06-14 19:04:25 +00:00
|
|
|
const u32 ITCMPhysicalSize = 0x8000;
|
|
|
|
const u32 DTCMPhysicalSize = 0x4000;
|
|
|
|
|
2023-11-25 17:32:09 +00:00
|
|
|
|
2023-11-18 15:40:54 +00:00
|
|
|
class ARMJIT;
|
2023-11-09 20:54:51 +00:00
|
|
|
class GPU;
|
2023-11-18 15:40:54 +00:00
|
|
|
class ARMJIT_Memory;
|
|
|
|
|
2016-11-03 00:38:58 +00:00
|
|
|
class ARM
|
2023-10-22 13:35:31 +00:00
|
|
|
#ifdef GDBSTUB_ENABLED
|
|
|
|
: public Gdb::StubCallbacks
|
|
|
|
#endif
|
2016-11-03 00:38:58 +00:00
|
|
|
{
|
|
|
|
public:
|
2023-11-25 17:32:09 +00:00
|
|
|
ARM(u32 num, ARMJIT& jit, GPU& gpu);
|
2020-06-14 19:04:25 +00:00
|
|
|
virtual ~ARM(); // destroy shit
|
2016-11-03 00:38:58 +00:00
|
|
|
|
2018-12-04 16:54:10 +00:00
|
|
|
virtual void Reset();
|
|
|
|
|
|
|
|
virtual void DoSavestate(Savestate* file);
|
2016-11-24 17:31:49 +00:00
|
|
|
|
2019-08-17 14:50:48 +00:00
|
|
|
virtual void FillPipeline() = 0;
|
|
|
|
|
2018-12-04 16:54:10 +00:00
|
|
|
virtual void JumpTo(u32 addr, bool restorecpsr = false) = 0;
|
2016-11-24 23:08:53 +00:00
|
|
|
void RestoreCPSR();
|
|
|
|
|
2016-12-05 22:17:03 +00:00
|
|
|
void Halt(u32 halt)
|
|
|
|
{
|
2017-04-13 02:16:57 +00:00
|
|
|
if (halt==2 && Halted==1) return;
|
2016-12-05 22:17:03 +00:00
|
|
|
Halted = halt;
|
|
|
|
}
|
|
|
|
|
2019-01-05 04:28:58 +00:00
|
|
|
virtual void Execute() = 0;
|
2020-07-25 18:59:53 +00:00
|
|
|
#ifdef JIT_ENABLED
|
2019-07-14 02:33:36 +00:00
|
|
|
virtual void ExecuteJIT() = 0;
|
2019-07-14 17:24:00 +00:00
|
|
|
#endif
|
2016-11-03 00:38:58 +00:00
|
|
|
|
2016-11-24 23:08:53 +00:00
|
|
|
bool CheckCondition(u32 code)
|
|
|
|
{
|
|
|
|
if (code == 0xE) return true;
|
|
|
|
if (ConditionTable[code] & (1 << (CPSR>>28))) return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SetC(bool c)
|
|
|
|
{
|
|
|
|
if (c) CPSR |= 0x20000000;
|
|
|
|
else CPSR &= ~0x20000000;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SetNZ(bool n, bool z)
|
|
|
|
{
|
|
|
|
CPSR &= ~0xC0000000;
|
|
|
|
if (n) CPSR |= 0x80000000;
|
|
|
|
if (z) CPSR |= 0x40000000;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SetNZCV(bool n, bool z, bool c, bool v)
|
|
|
|
{
|
|
|
|
CPSR &= ~0xF0000000;
|
|
|
|
if (n) CPSR |= 0x80000000;
|
|
|
|
if (z) CPSR |= 0x40000000;
|
|
|
|
if (c) CPSR |= 0x20000000;
|
|
|
|
if (v) CPSR |= 0x10000000;
|
|
|
|
}
|
|
|
|
|
2023-10-22 13:35:31 +00:00
|
|
|
inline bool ModeIs(u32 mode)
|
|
|
|
{
|
|
|
|
u32 cm = CPSR & 0x1f;
|
|
|
|
mode &= 0x1f;
|
|
|
|
|
|
|
|
if (mode == cm) return true;
|
|
|
|
if (mode == 0x17) return cm >= 0x14 && cm <= 0x17; // abt
|
|
|
|
if (mode == 0x1b) return cm >= 0x18 && cm <= 0x1b; // und
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-10-28 19:15:12 +00:00
|
|
|
void UpdateMode(u32 oldmode, u32 newmode, bool phony = false);
|
2016-12-03 03:41:10 +00:00
|
|
|
|
2016-12-04 02:20:50 +00:00
|
|
|
void TriggerIRQ();
|
|
|
|
|
2018-11-04 22:21:58 +00:00
|
|
|
void SetupCodeMem(u32 addr);
|
|
|
|
|
2016-12-03 00:31:33 +00:00
|
|
|
|
2018-12-09 00:17:05 +00:00
|
|
|
virtual void DataRead8(u32 addr, u32* val) = 0;
|
|
|
|
virtual void DataRead16(u32 addr, u32* val) = 0;
|
|
|
|
virtual void DataRead32(u32 addr, u32* val) = 0;
|
|
|
|
virtual void DataRead32S(u32 addr, u32* val) = 0;
|
|
|
|
virtual void DataWrite8(u32 addr, u8 val) = 0;
|
|
|
|
virtual void DataWrite16(u32 addr, u16 val) = 0;
|
|
|
|
virtual void DataWrite32(u32 addr, u32 val) = 0;
|
|
|
|
virtual void DataWrite32S(u32 addr, u32 val) = 0;
|
2018-11-04 22:21:58 +00:00
|
|
|
|
2018-12-04 16:54:10 +00:00
|
|
|
virtual void AddCycles_C() = 0;
|
2018-12-09 00:17:05 +00:00
|
|
|
virtual void AddCycles_CI(s32 numI) = 0;
|
2018-12-04 16:54:10 +00:00
|
|
|
virtual void AddCycles_CDI() = 0;
|
|
|
|
virtual void AddCycles_CD() = 0;
|
2018-11-04 22:21:58 +00:00
|
|
|
|
2023-10-22 13:35:31 +00:00
|
|
|
void CheckGdbIncoming();
|
2017-01-30 17:36:11 +00:00
|
|
|
|
2018-12-04 16:54:10 +00:00
|
|
|
u32 Num;
|
|
|
|
|
|
|
|
s32 Cycles;
|
2019-10-18 11:29:17 +00:00
|
|
|
union
|
|
|
|
{
|
|
|
|
struct
|
|
|
|
{
|
|
|
|
u8 Halted;
|
|
|
|
u8 IRQ; // nonzero to trigger IRQ
|
|
|
|
u8 IdleLoop;
|
|
|
|
};
|
|
|
|
u32 StopExecution;
|
|
|
|
};
|
2019-06-08 20:16:51 +00:00
|
|
|
|
2018-12-09 00:17:05 +00:00
|
|
|
u32 CodeRegion;
|
|
|
|
s32 CodeCycles;
|
2018-12-04 16:54:10 +00:00
|
|
|
|
2018-12-09 00:17:05 +00:00
|
|
|
u32 DataRegion;
|
2018-12-04 16:54:10 +00:00
|
|
|
s32 DataCycles;
|
|
|
|
|
|
|
|
u32 R[16]; // heh
|
|
|
|
u32 CPSR;
|
|
|
|
u32 R_FIQ[8]; // holding SPSR too
|
|
|
|
u32 R_SVC[3];
|
|
|
|
u32 R_ABT[3];
|
|
|
|
u32 R_IRQ[3];
|
|
|
|
u32 R_UND[3];
|
|
|
|
u32 CurInstr;
|
|
|
|
u32 NextInstr[2];
|
|
|
|
|
|
|
|
u32 ExceptionBase;
|
|
|
|
|
|
|
|
NDS::MemRegion CodeMem;
|
|
|
|
|
2020-06-14 19:04:25 +00:00
|
|
|
#ifdef JIT_ENABLED
|
2020-06-30 21:50:41 +00:00
|
|
|
u32 FastBlockLookupStart, FastBlockLookupSize;
|
2020-06-14 19:04:25 +00:00
|
|
|
u64* FastBlockLookup;
|
|
|
|
#endif
|
|
|
|
|
2023-11-18 15:40:54 +00:00
|
|
|
static const u32 ConditionTable[16];
|
2023-10-22 13:35:31 +00:00
|
|
|
#ifdef GDBSTUB_ENABLED
|
|
|
|
Gdb::GdbStub GdbStub;
|
|
|
|
#endif
|
2020-06-01 18:36:30 +00:00
|
|
|
|
2023-11-25 17:32:09 +00:00
|
|
|
ARMJIT& JIT;
|
2020-06-01 18:36:30 +00:00
|
|
|
protected:
|
|
|
|
u8 (*BusRead8)(u32 addr);
|
|
|
|
u16 (*BusRead16)(u32 addr);
|
|
|
|
u32 (*BusRead32)(u32 addr);
|
|
|
|
void (*BusWrite8)(u32 addr, u8 val);
|
|
|
|
void (*BusWrite16)(u32 addr, u16 val);
|
|
|
|
void (*BusWrite32)(u32 addr, u32 val);
|
2023-10-22 13:35:31 +00:00
|
|
|
|
|
|
|
#ifdef GDBSTUB_ENABLED
|
|
|
|
bool IsSingleStep;
|
|
|
|
bool BreakReq;
|
|
|
|
bool BreakOnStartup;
|
|
|
|
|
|
|
|
public:
|
|
|
|
int GetCPU() const override { return Num ? 7 : 9; }
|
|
|
|
|
|
|
|
u32 ReadReg(Gdb::Register reg) override;
|
|
|
|
void WriteReg(Gdb::Register reg, u32 v) override;
|
|
|
|
u32 ReadMem(u32 addr, int size) override;
|
|
|
|
void WriteMem(u32 addr, int size, u32 v) override;
|
|
|
|
|
|
|
|
void ResetGdb() override;
|
|
|
|
int RemoteCmd(const u8* cmd, size_t len) override;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
#endif
|
|
|
|
|
|
|
|
void GdbCheckA();
|
|
|
|
void GdbCheckB();
|
|
|
|
void GdbCheckC();
|
2023-11-09 20:54:51 +00:00
|
|
|
private:
|
2023-11-25 17:32:09 +00:00
|
|
|
melonDS::GPU& GPU;
|
2018-12-04 16:54:10 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
class ARMv5 : public ARM
|
|
|
|
{
|
|
|
|
public:
|
2023-11-25 17:32:09 +00:00
|
|
|
ARMv5(ARMJIT& jit, melonDS::GPU& gpu);
|
2020-06-14 19:04:25 +00:00
|
|
|
~ARMv5();
|
2018-12-04 16:54:10 +00:00
|
|
|
|
2023-10-22 13:35:31 +00:00
|
|
|
void Reset() override;
|
2018-12-04 16:54:10 +00:00
|
|
|
|
2023-10-22 13:35:31 +00:00
|
|
|
void DoSavestate(Savestate* file) override;
|
2018-12-04 16:54:10 +00:00
|
|
|
|
2018-12-09 00:17:05 +00:00
|
|
|
void UpdateRegionTimings(u32 addrstart, u32 addrend);
|
2018-12-04 16:54:10 +00:00
|
|
|
|
2023-10-22 13:35:31 +00:00
|
|
|
void FillPipeline() override;
|
2019-08-17 14:50:48 +00:00
|
|
|
|
2023-10-22 13:35:31 +00:00
|
|
|
void JumpTo(u32 addr, bool restorecpsr = false) override;
|
2018-12-04 16:54:10 +00:00
|
|
|
|
2018-12-07 13:20:38 +00:00
|
|
|
void PrefetchAbort();
|
|
|
|
void DataAbort();
|
|
|
|
|
2023-10-22 13:35:31 +00:00
|
|
|
void Execute() override;
|
2019-07-14 17:24:00 +00:00
|
|
|
#ifdef JIT_ENABLED
|
2023-10-22 13:35:31 +00:00
|
|
|
void ExecuteJIT() override;
|
2019-07-14 17:24:00 +00:00
|
|
|
#endif
|
2018-12-04 16:54:10 +00:00
|
|
|
|
|
|
|
// all code accesses are forced nonseq 32bit
|
2019-01-04 20:47:06 +00:00
|
|
|
u32 CodeRead32(u32 addr, bool branch);
|
2018-12-04 16:54:10 +00:00
|
|
|
|
2023-10-22 13:35:31 +00:00
|
|
|
void DataRead8(u32 addr, u32* val) override;
|
|
|
|
void DataRead16(u32 addr, u32* val) override;
|
|
|
|
void DataRead32(u32 addr, u32* val) override;
|
|
|
|
void DataRead32S(u32 addr, u32* val) override;
|
|
|
|
void DataWrite8(u32 addr, u8 val) override;
|
|
|
|
void DataWrite16(u32 addr, u16 val) override;
|
|
|
|
void DataWrite32(u32 addr, u32 val) override;
|
|
|
|
void DataWrite32S(u32 addr, u32 val) override;
|
2018-12-04 16:54:10 +00:00
|
|
|
|
2023-10-22 13:35:31 +00:00
|
|
|
void AddCycles_C() override
|
2018-12-04 16:54:10 +00:00
|
|
|
{
|
|
|
|
// code only. always nonseq 32-bit for ARM9.
|
2018-12-09 00:17:05 +00:00
|
|
|
s32 numC = (R[15] & 0x2) ? 0 : CodeCycles;
|
2020-07-27 21:14:23 +00:00
|
|
|
Cycles += numC;
|
2016-12-03 00:31:33 +00:00
|
|
|
}
|
|
|
|
|
2023-10-22 13:35:31 +00:00
|
|
|
void AddCycles_CI(s32 numI) override
|
2016-11-03 00:38:58 +00:00
|
|
|
{
|
2018-12-04 16:54:10 +00:00
|
|
|
// code+internal
|
2018-12-09 00:17:05 +00:00
|
|
|
s32 numC = (R[15] & 0x2) ? 0 : CodeCycles;
|
2020-07-27 21:14:23 +00:00
|
|
|
Cycles += numC + numI;
|
2018-12-04 16:54:10 +00:00
|
|
|
}
|
2018-11-04 22:21:58 +00:00
|
|
|
|
2023-10-22 13:35:31 +00:00
|
|
|
void AddCycles_CDI() override
|
2018-12-04 16:54:10 +00:00
|
|
|
{
|
|
|
|
// LDR/LDM cycles. ARM9 seems to skip the internal cycle there.
|
|
|
|
// TODO: ITCM data fetches shouldn't be parallelized, they say
|
2018-12-09 00:17:05 +00:00
|
|
|
s32 numC = (R[15] & 0x2) ? 0 : CodeCycles;
|
2018-12-04 16:54:10 +00:00
|
|
|
s32 numD = DataCycles;
|
2018-11-04 22:21:58 +00:00
|
|
|
|
2018-12-09 00:17:05 +00:00
|
|
|
//if (DataRegion != CodeRegion)
|
2020-07-27 21:14:23 +00:00
|
|
|
Cycles += std::max(numC + numD - 6, std::max(numC, numD));
|
2018-12-09 00:17:05 +00:00
|
|
|
//else
|
2020-07-27 21:14:23 +00:00
|
|
|
// Cycles += numC + numD;
|
2018-12-04 16:54:10 +00:00
|
|
|
}
|
|
|
|
|
2023-10-22 13:35:31 +00:00
|
|
|
void AddCycles_CD() override
|
2018-12-04 16:54:10 +00:00
|
|
|
{
|
|
|
|
// TODO: ITCM data fetches shouldn't be parallelized, they say
|
2018-12-09 00:17:05 +00:00
|
|
|
s32 numC = (R[15] & 0x2) ? 0 : CodeCycles;
|
2018-12-04 16:54:10 +00:00
|
|
|
s32 numD = DataCycles;
|
|
|
|
|
2018-12-09 00:17:05 +00:00
|
|
|
//if (DataRegion != CodeRegion)
|
2020-07-27 21:14:23 +00:00
|
|
|
Cycles += std::max(numC + numD - 6, std::max(numC, numD));
|
2018-12-09 00:17:05 +00:00
|
|
|
//else
|
2020-07-27 21:14:23 +00:00
|
|
|
// Cycles += numC + numD;
|
2018-12-04 16:54:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void GetCodeMemRegion(u32 addr, NDS::MemRegion* region);
|
|
|
|
|
|
|
|
void CP15Reset();
|
|
|
|
void CP15DoSavestate(Savestate* file);
|
|
|
|
|
|
|
|
void UpdateDTCMSetting();
|
|
|
|
void UpdateITCMSetting();
|
|
|
|
|
2019-10-21 21:14:34 +00:00
|
|
|
void UpdatePURegion(u32 n);
|
|
|
|
void UpdatePURegions(bool update_all);
|
2018-12-07 13:20:38 +00:00
|
|
|
|
2019-01-04 20:47:06 +00:00
|
|
|
u32 RandomLineIndex();
|
|
|
|
|
|
|
|
void ICacheLookup(u32 addr);
|
|
|
|
void ICacheInvalidateByAddr(u32 addr);
|
|
|
|
void ICacheInvalidateAll();
|
|
|
|
|
2018-12-04 16:54:10 +00:00
|
|
|
void CP15Write(u32 id, u32 val);
|
|
|
|
u32 CP15Read(u32 id);
|
|
|
|
|
|
|
|
u32 CP15Control;
|
2017-01-30 17:36:11 +00:00
|
|
|
|
2019-01-04 20:47:06 +00:00
|
|
|
u32 RNGSeed;
|
|
|
|
|
2018-12-04 16:54:10 +00:00
|
|
|
u32 DTCMSetting, ITCMSetting;
|
|
|
|
|
2020-02-04 17:29:52 +00:00
|
|
|
// for aarch64 JIT they need to go up here
|
|
|
|
// to be addressable by a 12-bit immediate
|
2018-12-04 16:54:10 +00:00
|
|
|
u32 ITCMSize;
|
2021-10-28 23:35:22 +00:00
|
|
|
u32 DTCMBase, DTCMMask;
|
2020-02-04 17:29:52 +00:00
|
|
|
s32 RegionCodeCycles;
|
|
|
|
|
2020-06-14 19:04:25 +00:00
|
|
|
u8 ITCM[ITCMPhysicalSize];
|
|
|
|
u8* DTCM;
|
2018-12-04 17:32:19 +00:00
|
|
|
|
2019-01-04 20:47:06 +00:00
|
|
|
u8 ICache[0x2000];
|
|
|
|
u32 ICacheTags[64*4];
|
|
|
|
u8 ICacheCount[64];
|
|
|
|
|
2018-12-04 17:32:19 +00:00
|
|
|
u32 PU_CodeCacheable;
|
|
|
|
u32 PU_DataCacheable;
|
|
|
|
u32 PU_DataCacheWrite;
|
|
|
|
|
|
|
|
u32 PU_CodeRW;
|
|
|
|
u32 PU_DataRW;
|
|
|
|
|
|
|
|
u32 PU_Region[8];
|
2018-12-07 13:20:38 +00:00
|
|
|
|
|
|
|
// 0=dataR 1=dataW 2=codeR 4=datacache 5=datawrite 6=codecache
|
|
|
|
u8 PU_PrivMap[0x100000];
|
2018-12-09 00:17:05 +00:00
|
|
|
u8 PU_UserMap[0x100000];
|
|
|
|
|
|
|
|
// games operate under system mode, generally
|
2021-10-28 19:24:39 +00:00
|
|
|
//#define PU_Map PU_PrivMap
|
|
|
|
u8* PU_Map;
|
2018-12-07 13:20:38 +00:00
|
|
|
|
2018-12-09 00:17:05 +00:00
|
|
|
// code/16N/32N/32S
|
|
|
|
u8 MemTimings[0x100000][4];
|
2018-12-11 16:59:52 +00:00
|
|
|
|
2019-01-04 20:47:06 +00:00
|
|
|
u8* CurICacheLine;
|
2020-06-01 18:36:30 +00:00
|
|
|
|
|
|
|
bool (*GetMemRegion)(u32 addr, bool write, NDS::MemRegion* region);
|
2023-10-22 13:35:31 +00:00
|
|
|
|
|
|
|
#ifdef GDBSTUB_ENABLED
|
|
|
|
u32 ReadMem(u32 addr, int size) override;
|
|
|
|
void WriteMem(u32 addr, int size, u32 v) override;
|
|
|
|
#endif
|
2018-12-04 16:54:10 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
class ARMv4 : public ARM
|
|
|
|
{
|
|
|
|
public:
|
2023-11-25 17:32:09 +00:00
|
|
|
ARMv4(ARMJIT& jit, melonDS::GPU& gpu);
|
2018-12-04 16:54:10 +00:00
|
|
|
|
2023-10-22 13:35:31 +00:00
|
|
|
void Reset() override;
|
2020-06-01 18:36:30 +00:00
|
|
|
|
2023-10-22 13:35:31 +00:00
|
|
|
void FillPipeline() override;
|
2019-08-17 14:50:48 +00:00
|
|
|
|
2023-10-22 13:35:31 +00:00
|
|
|
void JumpTo(u32 addr, bool restorecpsr = false) override;
|
2018-12-04 16:54:10 +00:00
|
|
|
|
2023-10-22 13:35:31 +00:00
|
|
|
void Execute() override;
|
2019-07-14 17:24:00 +00:00
|
|
|
#ifdef JIT_ENABLED
|
2023-10-22 13:35:31 +00:00
|
|
|
void ExecuteJIT() override;
|
2019-07-14 17:24:00 +00:00
|
|
|
#endif
|
2018-12-04 16:54:10 +00:00
|
|
|
|
|
|
|
u16 CodeRead16(u32 addr)
|
|
|
|
{
|
2020-06-01 18:36:30 +00:00
|
|
|
return BusRead16(addr);
|
2017-01-30 17:36:11 +00:00
|
|
|
}
|
|
|
|
|
2018-12-04 16:54:10 +00:00
|
|
|
u32 CodeRead32(u32 addr)
|
|
|
|
{
|
2020-06-01 18:36:30 +00:00
|
|
|
return BusRead32(addr);
|
2018-12-04 16:54:10 +00:00
|
|
|
}
|
2017-01-30 17:36:11 +00:00
|
|
|
|
2023-10-22 13:35:31 +00:00
|
|
|
void DataRead8(u32 addr, u32* val) override
|
2017-01-30 17:36:11 +00:00
|
|
|
{
|
2020-06-01 18:36:30 +00:00
|
|
|
*val = BusRead8(addr);
|
2020-05-08 22:45:05 +00:00
|
|
|
DataRegion = addr;
|
2019-10-02 23:10:59 +00:00
|
|
|
DataCycles = NDS::ARM7MemTimings[addr >> 15][0];
|
2017-01-30 17:36:11 +00:00
|
|
|
}
|
|
|
|
|
2023-10-22 13:35:31 +00:00
|
|
|
void DataRead16(u32 addr, u32* val) override
|
2017-01-30 17:36:11 +00:00
|
|
|
{
|
2016-12-03 00:31:33 +00:00
|
|
|
addr &= ~1;
|
2018-12-04 16:54:10 +00:00
|
|
|
|
2020-06-01 18:36:30 +00:00
|
|
|
*val = BusRead16(addr);
|
2020-05-08 22:45:05 +00:00
|
|
|
DataRegion = addr;
|
2019-10-02 23:10:59 +00:00
|
|
|
DataCycles = NDS::ARM7MemTimings[addr >> 15][0];
|
2016-12-03 00:31:33 +00:00
|
|
|
}
|
|
|
|
|
2023-10-22 13:35:31 +00:00
|
|
|
void DataRead32(u32 addr, u32* val) override
|
2016-12-03 00:31:33 +00:00
|
|
|
{
|
|
|
|
addr &= ~3;
|
2018-12-04 16:54:10 +00:00
|
|
|
|
2020-06-01 18:36:30 +00:00
|
|
|
*val = BusRead32(addr);
|
2020-05-08 22:45:05 +00:00
|
|
|
DataRegion = addr;
|
2019-10-02 23:10:59 +00:00
|
|
|
DataCycles = NDS::ARM7MemTimings[addr >> 15][2];
|
2016-12-03 00:31:33 +00:00
|
|
|
}
|
|
|
|
|
2023-10-22 13:35:31 +00:00
|
|
|
void DataRead32S(u32 addr, u32* val) override
|
2016-12-03 00:31:33 +00:00
|
|
|
{
|
2018-12-09 00:17:05 +00:00
|
|
|
addr &= ~3;
|
2017-01-30 17:36:11 +00:00
|
|
|
|
2020-06-01 18:36:30 +00:00
|
|
|
*val = BusRead32(addr);
|
2019-10-02 23:10:59 +00:00
|
|
|
DataCycles += NDS::ARM7MemTimings[addr >> 15][3];
|
2016-12-03 00:31:33 +00:00
|
|
|
}
|
|
|
|
|
2023-10-22 13:35:31 +00:00
|
|
|
void DataWrite8(u32 addr, u8 val) override
|
2016-12-03 00:31:33 +00:00
|
|
|
{
|
2020-06-01 18:36:30 +00:00
|
|
|
BusWrite8(addr, val);
|
2020-05-08 22:45:05 +00:00
|
|
|
DataRegion = addr;
|
2019-10-02 23:10:59 +00:00
|
|
|
DataCycles = NDS::ARM7MemTimings[addr >> 15][0];
|
2018-12-09 00:17:05 +00:00
|
|
|
}
|
2018-12-04 16:54:10 +00:00
|
|
|
|
2023-10-22 13:35:31 +00:00
|
|
|
void DataWrite16(u32 addr, u16 val) override
|
2018-12-09 00:17:05 +00:00
|
|
|
{
|
|
|
|
addr &= ~1;
|
2017-01-30 17:36:11 +00:00
|
|
|
|
2020-06-01 18:36:30 +00:00
|
|
|
BusWrite16(addr, val);
|
2020-05-08 22:45:05 +00:00
|
|
|
DataRegion = addr;
|
2019-10-02 23:10:59 +00:00
|
|
|
DataCycles = NDS::ARM7MemTimings[addr >> 15][0];
|
2016-12-03 00:31:33 +00:00
|
|
|
}
|
|
|
|
|
2023-10-22 13:35:31 +00:00
|
|
|
void DataWrite32(u32 addr, u32 val) override
|
2016-12-03 00:31:33 +00:00
|
|
|
{
|
|
|
|
addr &= ~3;
|
2018-12-04 16:54:10 +00:00
|
|
|
|
2020-06-01 18:36:30 +00:00
|
|
|
BusWrite32(addr, val);
|
2020-05-08 22:45:05 +00:00
|
|
|
DataRegion = addr;
|
2019-10-02 23:10:59 +00:00
|
|
|
DataCycles = NDS::ARM7MemTimings[addr >> 15][2];
|
2018-12-09 00:17:05 +00:00
|
|
|
}
|
|
|
|
|
2023-10-22 13:35:31 +00:00
|
|
|
void DataWrite32S(u32 addr, u32 val) override
|
2018-12-09 00:17:05 +00:00
|
|
|
{
|
|
|
|
addr &= ~3;
|
2016-12-03 00:31:33 +00:00
|
|
|
|
2020-06-01 18:36:30 +00:00
|
|
|
BusWrite32(addr, val);
|
2019-10-02 23:10:59 +00:00
|
|
|
DataCycles += NDS::ARM7MemTimings[addr >> 15][3];
|
2016-11-03 00:38:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-10-22 13:35:31 +00:00
|
|
|
void AddCycles_C() override
|
2018-12-04 16:54:10 +00:00
|
|
|
{
|
|
|
|
// code only. this code fetch is sequential.
|
2020-07-27 21:14:23 +00:00
|
|
|
Cycles += NDS::ARM7MemTimings[CodeCycles][(CPSR&0x20)?1:3];
|
2018-12-04 16:54:10 +00:00
|
|
|
}
|
2017-01-30 17:36:11 +00:00
|
|
|
|
2023-10-22 13:35:31 +00:00
|
|
|
void AddCycles_CI(s32 num) override
|
2018-12-04 16:54:10 +00:00
|
|
|
{
|
|
|
|
// code+internal. results in a nonseq code fetch.
|
2020-07-27 21:14:23 +00:00
|
|
|
Cycles += NDS::ARM7MemTimings[CodeCycles][(CPSR&0x20)?0:2] + num;
|
2018-12-04 16:54:10 +00:00
|
|
|
}
|
2016-12-05 16:08:24 +00:00
|
|
|
|
2023-10-22 13:35:31 +00:00
|
|
|
void AddCycles_CDI() override
|
2018-12-04 16:54:10 +00:00
|
|
|
{
|
|
|
|
// LDR/LDM cycles.
|
2018-12-09 00:17:05 +00:00
|
|
|
s32 numC = NDS::ARM7MemTimings[CodeCycles][(CPSR&0x20)?0:2];
|
2018-12-04 16:54:10 +00:00
|
|
|
s32 numD = DataCycles;
|
2016-11-03 00:38:58 +00:00
|
|
|
|
2020-05-08 22:45:05 +00:00
|
|
|
if ((DataRegion >> 24) == 0x02) // mainRAM
|
2018-12-04 16:54:10 +00:00
|
|
|
{
|
2018-12-09 00:17:05 +00:00
|
|
|
if (CodeRegion == 0x02)
|
2020-07-27 21:14:23 +00:00
|
|
|
Cycles += numC + numD;
|
2018-12-04 16:54:10 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
numC++;
|
2020-07-27 21:14:23 +00:00
|
|
|
Cycles += std::max(numC + numD - 3, std::max(numC, numD));
|
2018-12-04 16:54:10 +00:00
|
|
|
}
|
|
|
|
}
|
2018-12-09 00:17:05 +00:00
|
|
|
else if (CodeRegion == 0x02)
|
2018-12-04 16:54:10 +00:00
|
|
|
{
|
|
|
|
numD++;
|
2020-07-27 21:14:23 +00:00
|
|
|
Cycles += std::max(numC + numD - 3, std::max(numC, numD));
|
2018-12-04 16:54:10 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-07-27 21:14:23 +00:00
|
|
|
Cycles += numC + numD + 1;
|
2018-12-04 16:54:10 +00:00
|
|
|
}
|
|
|
|
}
|
2016-11-24 23:08:53 +00:00
|
|
|
|
2023-10-22 13:35:31 +00:00
|
|
|
void AddCycles_CD() override
|
2018-12-04 16:54:10 +00:00
|
|
|
{
|
|
|
|
// TODO: max gain should be 5c when writing to mainRAM
|
2018-12-09 00:17:05 +00:00
|
|
|
s32 numC = NDS::ARM7MemTimings[CodeCycles][(CPSR&0x20)?0:2];
|
2018-12-04 16:54:10 +00:00
|
|
|
s32 numD = DataCycles;
|
2016-12-23 20:22:22 +00:00
|
|
|
|
2020-05-08 22:45:05 +00:00
|
|
|
if ((DataRegion >> 24) == 0x02)
|
2018-12-04 16:54:10 +00:00
|
|
|
{
|
2018-12-09 00:17:05 +00:00
|
|
|
if (CodeRegion == 0x02)
|
2020-07-27 21:14:23 +00:00
|
|
|
Cycles += numC + numD;
|
2018-12-04 16:54:10 +00:00
|
|
|
else
|
2020-07-27 21:14:23 +00:00
|
|
|
Cycles += std::max(numC + numD - 3, std::max(numC, numD));
|
2018-12-04 16:54:10 +00:00
|
|
|
}
|
2018-12-09 00:17:05 +00:00
|
|
|
else if (CodeRegion == 0x02)
|
2018-12-04 16:54:10 +00:00
|
|
|
{
|
2020-07-27 21:14:23 +00:00
|
|
|
Cycles += std::max(numC + numD - 3, std::max(numC, numD));
|
2018-12-04 16:54:10 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-07-27 21:14:23 +00:00
|
|
|
Cycles += numC + numD;
|
2018-12-04 16:54:10 +00:00
|
|
|
}
|
|
|
|
}
|
2016-11-03 00:38:58 +00:00
|
|
|
};
|
|
|
|
|
2017-06-13 09:17:22 +00:00
|
|
|
namespace ARMInterpreter
|
|
|
|
{
|
|
|
|
|
|
|
|
void A_UNK(ARM* cpu);
|
|
|
|
void T_UNK(ARM* cpu);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2020-05-08 22:45:05 +00:00
|
|
|
namespace NDS
|
|
|
|
{
|
|
|
|
|
|
|
|
extern ARMv5* ARM9;
|
|
|
|
extern ARMv4* ARM7;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2023-11-25 17:32:09 +00:00
|
|
|
}
|
2016-11-03 00:38:58 +00:00
|
|
|
#endif // ARM_H
|