Inline CPU stepping

This commit is contained in:
Jeffrey Pfau 2014-01-21 21:42:21 -08:00
parent f8de62ba71
commit 52808da265
5 changed files with 126 additions and 130 deletions

View File

@ -155,6 +155,125 @@ void ARMRaiseSWI(struct ARMCore* cpu) {
cpu->cpsr.i = 1; cpu->cpsr.i = 1;
} }
static inline ARMInstruction _ARMLoadInstructionARM(struct ARMMemory* memory, uint32_t address, uint32_t* opcodeOut) {
uint32_t opcode;
LOAD_32(opcode, address & memory->activeMask, memory->activeRegion);
*opcodeOut = opcode;
return _armTable[((opcode >> 16) & 0xFF0) | ((opcode >> 4) & 0x00F)];
}
static inline void ARMStep(struct ARMCore* cpu) {
uint32_t opcode;
cpu->currentPC = cpu->gprs[ARM_PC] - WORD_SIZE_ARM;
ARMInstruction instruction = _ARMLoadInstructionARM(cpu->memory, cpu->currentPC, &opcode);
cpu->gprs[ARM_PC] += WORD_SIZE_ARM;
int condition = opcode >> 28;
if (condition == 0xE) {
instruction(cpu, opcode);
return;
} else {
switch (condition) {
case 0x0:
if (!ARM_COND_EQ) {
cpu->cycles += ARM_PREFETCH_CYCLES;
return;
}
break;
case 0x1:
if (!ARM_COND_NE) {
cpu->cycles += ARM_PREFETCH_CYCLES;
return;
}
break;
case 0x2:
if (!ARM_COND_CS) {
cpu->cycles += ARM_PREFETCH_CYCLES;
return;
}
break;
case 0x3:
if (!ARM_COND_CC) {
cpu->cycles += ARM_PREFETCH_CYCLES;
return;
}
break;
case 0x4:
if (!ARM_COND_MI) {
cpu->cycles += ARM_PREFETCH_CYCLES;
return;
}
break;
case 0x5:
if (!ARM_COND_PL) {
cpu->cycles += ARM_PREFETCH_CYCLES;
return;
}
break;
case 0x6:
if (!ARM_COND_VS) {
cpu->cycles += ARM_PREFETCH_CYCLES;
return;
}
break;
case 0x7:
if (!ARM_COND_VC) {
cpu->cycles += ARM_PREFETCH_CYCLES;
return;
}
break;
case 0x8:
if (!ARM_COND_HI) {
cpu->cycles += ARM_PREFETCH_CYCLES;
return;
}
break;
case 0x9:
if (!ARM_COND_LS) {
cpu->cycles += ARM_PREFETCH_CYCLES;
return;
}
break;
case 0xA:
if (!ARM_COND_GE) {
cpu->cycles += ARM_PREFETCH_CYCLES;
return;
}
break;
case 0xB:
if (!ARM_COND_LT) {
cpu->cycles += ARM_PREFETCH_CYCLES;
return;
}
break;
case 0xC:
if (!ARM_COND_GT) {
cpu->cycles += ARM_PREFETCH_CYCLES;
return;
}
break;
case 0xD:
if (!ARM_COND_LE) {
cpu->cycles += ARM_PREFETCH_CYCLES;
return;
}
break;
default:
break;
}
}
instruction(cpu, opcode);
}
static inline void ThumbStep(struct ARMCore* cpu) {
cpu->currentPC = cpu->gprs[ARM_PC] - WORD_SIZE_THUMB;
cpu->gprs[ARM_PC] += WORD_SIZE_THUMB;
uint16_t opcode;
LOAD_16(opcode, cpu->currentPC & cpu->memory->activeMask, cpu->memory->activeRegion);
ThumbInstruction instruction = _thumbTable[opcode >> 6];
instruction(cpu, opcode);
}
void ARMRun(struct ARMCore* cpu) { void ARMRun(struct ARMCore* cpu) {
if (cpu->executionMode == MODE_THUMB) { if (cpu->executionMode == MODE_THUMB) {
ThumbStep(cpu); ThumbStep(cpu);

View File

@ -9,8 +9,6 @@ enum {
PSR_STATE_MASK = 0x00000020 PSR_STATE_MASK = 0x00000020
}; };
#define ARM_PREFETCH_CYCLES (1 + cpu->memory->activePrefetchCycles32)
// Addressing mode 1 // Addressing mode 1
static inline void _shiftLSL(struct ARMCore* cpu, uint32_t opcode) { static inline void _shiftLSL(struct ARMCore* cpu, uint32_t opcode) {
int rm = opcode & 0x0000000F; int rm = opcode & 0x0000000F;
@ -183,119 +181,6 @@ static inline void _immediate(struct ARMCore* cpu, uint32_t opcode) {
} }
} }
static const ARMInstruction _armTable[0x1000];
static ARMInstruction _ARMLoadInstructionARM(struct ARMMemory* memory, uint32_t address, uint32_t* opcodeOut) {
uint32_t opcode;
LOAD_32(opcode, address & memory->activeMask, memory->activeRegion);
*opcodeOut = opcode;
return _armTable[((opcode >> 16) & 0xFF0) | ((opcode >> 4) & 0x00F)];
}
void ARMStep(struct ARMCore* cpu) {
// TODO
uint32_t opcode;
cpu->currentPC = cpu->gprs[ARM_PC] - WORD_SIZE_ARM;
ARMInstruction instruction = _ARMLoadInstructionARM(cpu->memory, cpu->currentPC, &opcode);
cpu->gprs[ARM_PC] += WORD_SIZE_ARM;
int condition = opcode >> 28;
if (condition == 0xE) {
instruction(cpu, opcode);
return;
} else {
switch (condition) {
case 0x0:
if (!ARM_COND_EQ) {
cpu->cycles += ARM_PREFETCH_CYCLES;
return;
}
break;
case 0x1:
if (!ARM_COND_NE) {
cpu->cycles += ARM_PREFETCH_CYCLES;
return;
}
break;
case 0x2:
if (!ARM_COND_CS) {
cpu->cycles += ARM_PREFETCH_CYCLES;
return;
}
break;
case 0x3:
if (!ARM_COND_CC) {
cpu->cycles += ARM_PREFETCH_CYCLES;
return;
}
break;
case 0x4:
if (!ARM_COND_MI) {
cpu->cycles += ARM_PREFETCH_CYCLES;
return;
}
break;
case 0x5:
if (!ARM_COND_PL) {
cpu->cycles += ARM_PREFETCH_CYCLES;
return;
}
break;
case 0x6:
if (!ARM_COND_VS) {
cpu->cycles += ARM_PREFETCH_CYCLES;
return;
}
break;
case 0x7:
if (!ARM_COND_VC) {
cpu->cycles += ARM_PREFETCH_CYCLES;
return;
}
break;
case 0x8:
if (!ARM_COND_HI) {
cpu->cycles += ARM_PREFETCH_CYCLES;
return;
}
break;
case 0x9:
if (!ARM_COND_LS) {
cpu->cycles += ARM_PREFETCH_CYCLES;
return;
}
break;
case 0xA:
if (!ARM_COND_GE) {
cpu->cycles += ARM_PREFETCH_CYCLES;
return;
}
break;
case 0xB:
if (!ARM_COND_LT) {
cpu->cycles += ARM_PREFETCH_CYCLES;
return;
}
break;
case 0xC:
if (!ARM_COND_GT) {
cpu->cycles += ARM_PREFETCH_CYCLES;
return;
}
break;
case 0xD:
if (!ARM_COND_LE) {
cpu->cycles += ARM_PREFETCH_CYCLES;
return;
}
break;
default:
break;
}
}
instruction(cpu, opcode);
}
// Instruction definitions // Instruction definitions
// Beware pre-processor antics // Beware pre-processor antics
@ -1141,6 +1026,6 @@ DEFINE_INSTRUCTION_ARM(SWI, cpu->board->swi32(cpu->board, opcode & 0xFFFFFF))
DECLARE_ARM_COPROCESSOR_BLOCK(EMITTER, CDP, MRC), \ DECLARE_ARM_COPROCESSOR_BLOCK(EMITTER, CDP, MRC), \
DECLARE_ARM_SWI_BLOCK(EMITTER) DECLARE_ARM_SWI_BLOCK(EMITTER)
static const ARMInstruction _armTable[0x1000] = { const ARMInstruction _armTable[0x1000] = {
DECLARE_ARM_EMITTER_BLOCK(_ARMInstruction) DECLARE_ARM_EMITTER_BLOCK(_ARMInstruction)
}; };

View File

@ -3,9 +3,12 @@
#include <stdint.h> #include <stdint.h>
#define ARM_PREFETCH_CYCLES (1 + cpu->memory->activePrefetchCycles32)
struct ARMCore; struct ARMCore;
void ARMStep(struct ARMCore* cpu);
typedef void (*ARMInstruction)(struct ARMCore*, uint32_t opcode); typedef void (*ARMInstruction)(struct ARMCore*, uint32_t opcode);
const ARMInstruction _armTable[0x1000];
#endif #endif

View File

@ -2,17 +2,6 @@
#include "isa-inlines.h" #include "isa-inlines.h"
static const ThumbInstruction _thumbTable[0x400];
void ThumbStep(struct ARMCore* cpu) {
cpu->currentPC = cpu->gprs[ARM_PC] - WORD_SIZE_THUMB;
cpu->gprs[ARM_PC] += WORD_SIZE_THUMB;
uint16_t opcode;
LOAD_16(opcode, cpu->currentPC & cpu->memory->activeMask, cpu->memory->activeRegion);
ThumbInstruction instruction = _thumbTable[opcode >> 6];
instruction(cpu, opcode);
}
// Instruction definitions // Instruction definitions
// Beware pre-processor insanity // Beware pre-processor insanity
@ -591,6 +580,6 @@ DEFINE_INSTRUCTION_THUMB(SWI, cpu->board->swi16(cpu->board, opcode & 0xFF))
DO_8(DO_4(DECLARE_INSTRUCTION_THUMB(EMITTER, BL1))), \ DO_8(DO_4(DECLARE_INSTRUCTION_THUMB(EMITTER, BL1))), \
DO_8(DO_4(DECLARE_INSTRUCTION_THUMB(EMITTER, BL2))) DO_8(DO_4(DECLARE_INSTRUCTION_THUMB(EMITTER, BL2)))
static const ThumbInstruction _thumbTable[0x400] = { const ThumbInstruction _thumbTable[0x400] = {
DECLARE_THUMB_EMITTER_BLOCK(_ThumbInstruction) DECLARE_THUMB_EMITTER_BLOCK(_ThumbInstruction)
}; };

View File

@ -5,7 +5,7 @@
struct ARMCore; struct ARMCore;
void ThumbStep(struct ARMCore* cpu);
typedef void (*ThumbInstruction)(struct ARMCore*, uint16_t opcode); typedef void (*ThumbInstruction)(struct ARMCore*, uint16_t opcode);
const ThumbInstruction _thumbTable[0x400];
#endif #endif