180 lines
4.3 KiB
C++
180 lines
4.3 KiB
C++
#include <stdio.h>
|
|
#include "NDS.h"
|
|
#include "CP15.h"
|
|
#include "ARMInterpreter.h"
|
|
#include "ARMInterpreter_ALU.h"
|
|
#include "ARMInterpreter_Branch.h"
|
|
#include "ARMInterpreter_LoadStore.h"
|
|
|
|
|
|
namespace ARMInterpreter
|
|
{
|
|
|
|
|
|
s32 A_UNK(ARM* cpu)
|
|
{
|
|
printf("undefined ARM%d instruction %08X @ %08X\n", cpu->Num?7:9, cpu->CurInstr, cpu->R[15]-8);
|
|
for (int i = 0; i < 16; i++) printf("R%d: %08X\n", i, cpu->R[i]);
|
|
NDS::Halt();
|
|
return 0x7FFFFFFF;
|
|
}
|
|
|
|
s32 T_UNK(ARM* cpu)
|
|
{
|
|
printf("undefined THUMB%d instruction %04X @ %08X\n", cpu->Num?7:9, cpu->CurInstr, cpu->R[15]-4);
|
|
NDS::Halt();
|
|
return 0x7FFFFFFF;
|
|
}
|
|
|
|
|
|
|
|
s32 A_MSR_IMM(ARM* cpu)
|
|
{
|
|
u32* psr;
|
|
if (cpu->CurInstr & (1<<22))
|
|
{
|
|
switch (cpu->CPSR & 0x1F)
|
|
{
|
|
case 0x11: psr = &cpu->R_FIQ[7]; break;
|
|
case 0x12: psr = &cpu->R_IRQ[2]; break;
|
|
case 0x13: psr = &cpu->R_SVC[2]; break;
|
|
case 0x17: psr = &cpu->R_ABT[2]; break;
|
|
case 0x1B: psr = &cpu->R_UND[2]; break;
|
|
default: printf("bad CPU mode %08X\n", cpu->CPSR); return 1;
|
|
}
|
|
}
|
|
else
|
|
psr = &cpu->CPSR;
|
|
|
|
u32 oldpsr = *psr;
|
|
|
|
u32 mask = 0;
|
|
if (cpu->CurInstr & (1<<16)) mask |= 0x000000DF;
|
|
if (cpu->CurInstr & (1<<17)) mask |= 0x0000FF00;
|
|
if (cpu->CurInstr & (1<<18)) mask |= 0x00FF0000;
|
|
if (cpu->CurInstr & (1<<19)) mask |= 0xFF000000;
|
|
|
|
if ((cpu->CPSR & 0x1F) == 0x10) mask &= 0xFFFFFF00;
|
|
|
|
u32 val = ROR((cpu->CurInstr & 0xFF), ((cpu->CurInstr >> 7) & 0x1E));
|
|
|
|
*psr &= ~mask;
|
|
*psr |= (val & mask);
|
|
|
|
if (!(cpu->CurInstr & (1<<22)))
|
|
cpu->UpdateMode(oldpsr, cpu->CPSR);
|
|
|
|
return C_S(1);
|
|
}
|
|
|
|
s32 A_MSR_REG(ARM* cpu)
|
|
{
|
|
u32* psr;
|
|
if (cpu->CurInstr & (1<<22))
|
|
{
|
|
switch (cpu->CPSR & 0x1F)
|
|
{
|
|
case 0x11: psr = &cpu->R_FIQ[7]; break;
|
|
case 0x12: psr = &cpu->R_IRQ[2]; break;
|
|
case 0x13: psr = &cpu->R_SVC[2]; break;
|
|
case 0x17: psr = &cpu->R_ABT[2]; break;
|
|
case 0x1B: psr = &cpu->R_UND[2]; break;
|
|
default: printf("bad CPU mode %08X\n", cpu->CPSR); return 1;
|
|
}
|
|
}
|
|
else
|
|
psr = &cpu->CPSR;
|
|
|
|
u32 oldpsr = *psr;
|
|
|
|
u32 mask = 0;
|
|
if (cpu->CurInstr & (1<<16)) mask |= 0x000000DF;
|
|
if (cpu->CurInstr & (1<<17)) mask |= 0x0000FF00;
|
|
if (cpu->CurInstr & (1<<18)) mask |= 0x00FF0000;
|
|
if (cpu->CurInstr & (1<<19)) mask |= 0xFF000000;
|
|
|
|
if ((cpu->CPSR & 0x1F) == 0x10) mask &= 0xFFFFFF00;
|
|
|
|
u32 val = cpu->R[cpu->CurInstr & 0xF];
|
|
|
|
*psr &= ~mask;
|
|
*psr |= (val & mask);
|
|
|
|
if (!(cpu->CurInstr & (1<<22)))
|
|
cpu->UpdateMode(oldpsr, cpu->CPSR);
|
|
|
|
return C_S(1);
|
|
}
|
|
|
|
s32 A_MRS(ARM* cpu)
|
|
{
|
|
u32 psr;
|
|
if (cpu->CurInstr & (1<<22))
|
|
{
|
|
switch (cpu->CPSR & 0x1F)
|
|
{
|
|
case 0x11: psr = cpu->R_FIQ[7]; break;
|
|
case 0x12: psr = cpu->R_IRQ[2]; break;
|
|
case 0x13: psr = cpu->R_SVC[2]; break;
|
|
case 0x17: psr = cpu->R_ABT[2]; break;
|
|
case 0x1B: psr = cpu->R_UND[2]; break;
|
|
default: printf("bad CPU mode %08X\n", cpu->CPSR); return 1;
|
|
}
|
|
}
|
|
else
|
|
psr = cpu->CPSR;
|
|
|
|
cpu->R[(cpu->CurInstr>>12) & 0xF] = psr;
|
|
|
|
return C_S(1);
|
|
}
|
|
|
|
|
|
s32 A_MCR(ARM* cpu)
|
|
{
|
|
u32 cp = (cpu->CurInstr >> 8) & 0xF;
|
|
//u32 op = (cpu->CurInstr >> 21) & 0x7;
|
|
u32 cn = (cpu->CurInstr >> 16) & 0xF;
|
|
u32 cm = cpu->CurInstr & 0xF;
|
|
u32 cpinfo = (cpu->CurInstr >> 5) & 0x7;
|
|
|
|
if (cpu->Num==0 && cp==15)
|
|
{
|
|
CP15::Write((cn<<8)|(cm<<4)|cpinfo, cpu->R[(cpu->CurInstr>>12)&0xF]);
|
|
}
|
|
else
|
|
{
|
|
printf("bad MCR opcode p%d,%d,%d,%d on ARM%d\n", cp, cn, cm, cpinfo, cpu->Num?7:9);
|
|
}
|
|
|
|
return C_S(1) + 1; // TODO: checkme
|
|
}
|
|
|
|
s32 A_MRC(ARM* cpu)
|
|
{
|
|
u32 cp = (cpu->CurInstr >> 8) & 0xF;
|
|
//u32 op = (cpu->CurInstr >> 21) & 0x7;
|
|
u32 cn = (cpu->CurInstr >> 16) & 0xF;
|
|
u32 cm = cpu->CurInstr & 0xF;
|
|
u32 cpinfo = (cpu->CurInstr >> 5) & 0x7;
|
|
|
|
if (cpu->Num==0 && cp==15)
|
|
{
|
|
cpu->R[(cpu->CurInstr>>12)&0xF] = CP15::Read((cn<<8)|(cm<<4)|cpinfo);
|
|
}
|
|
else
|
|
{
|
|
printf("bad MRC opcode p%d,%d,%d,%d on ARM%d\n", cp, cn, cm, cpinfo, cpu->Num?7:9);
|
|
}
|
|
|
|
return C_S(1) + 1 + C_I(1); // TODO: checkme
|
|
}
|
|
|
|
|
|
|
|
#define INSTRFUNC_PROTO(x) s32 (*x)(ARM* cpu)
|
|
#include "ARM_InstrTable.h"
|
|
#undef INSTRFUNC_PROTO
|
|
|
|
}
|