147 lines
3.8 KiB
C++
147 lines
3.8 KiB
C++
#include "cpu-impl.hh"
|
|
#include "util/bits.hh"
|
|
#include "util/log.hh"
|
|
#include "utility.hh"
|
|
#include <algorithm>
|
|
#include <cstdio>
|
|
|
|
using namespace logger;
|
|
|
|
namespace matar {
|
|
CpuImpl::CpuImpl(const Bus& bus) noexcept
|
|
: bus(std::make_shared<Bus>(bus))
|
|
, gpr({ 0 })
|
|
, cpsr(0)
|
|
, spsr(0)
|
|
, is_flushed(false)
|
|
, gpr_banked({ { 0 }, { 0 }, { 0 }, { 0 }, { 0 }, { 0 } })
|
|
, spsr_banked({ 0, 0, 0, 0, 0 }) {
|
|
cpsr.set_mode(Mode::Supervisor);
|
|
cpsr.set_irq_disabled(true);
|
|
cpsr.set_fiq_disabled(true);
|
|
cpsr.set_state(State::Arm);
|
|
log_info("CPU successfully initialised");
|
|
|
|
// PC always points to two instructions ahead
|
|
// PC - 2 is the instruction being executed
|
|
pc += 2 * arm::INSTRUCTION_SIZE;
|
|
}
|
|
|
|
/* change modes */
|
|
void
|
|
CpuImpl::chg_mode(const Mode to) {
|
|
Mode from = cpsr.mode();
|
|
|
|
if (from == to)
|
|
return;
|
|
|
|
/* TODO: replace visible registers with view once I understand how to
|
|
* concatenate views */
|
|
#define STORE_BANKED(mode, MODE) \
|
|
std::copy(gpr.begin() + GPR_##MODE##_FIRST, \
|
|
gpr.begin() + gpr.size() - 1, \
|
|
gpr_banked.mode.begin())
|
|
|
|
switch (from) {
|
|
case Mode::Fiq:
|
|
STORE_BANKED(fiq, FIQ);
|
|
spsr_banked.fiq = spsr;
|
|
break;
|
|
|
|
case Mode::Supervisor:
|
|
STORE_BANKED(svc, SVC);
|
|
spsr_banked.svc = spsr;
|
|
break;
|
|
|
|
case Mode::Abort:
|
|
STORE_BANKED(abt, ABT);
|
|
spsr_banked.abt = spsr;
|
|
break;
|
|
|
|
case Mode::Irq:
|
|
STORE_BANKED(irq, IRQ);
|
|
spsr_banked.irq = spsr;
|
|
break;
|
|
|
|
case Mode::Undefined:
|
|
STORE_BANKED(und, UND);
|
|
spsr_banked.und = spsr;
|
|
break;
|
|
|
|
case Mode::User:
|
|
case Mode::System:
|
|
STORE_BANKED(old, SYS_USR);
|
|
break;
|
|
}
|
|
|
|
#define RESTORE_BANKED(mode, MODE) \
|
|
std::copy(gpr_banked.mode.begin(), \
|
|
gpr_banked.mode.end(), \
|
|
gpr.begin() + GPR_##MODE##_FIRST)
|
|
|
|
switch (to) {
|
|
case Mode::Fiq:
|
|
RESTORE_BANKED(fiq, FIQ);
|
|
spsr = spsr_banked.fiq;
|
|
break;
|
|
|
|
case Mode::Supervisor:
|
|
RESTORE_BANKED(svc, SVC);
|
|
spsr = spsr_banked.svc;
|
|
break;
|
|
|
|
case Mode::Abort:
|
|
RESTORE_BANKED(abt, ABT);
|
|
spsr = spsr_banked.abt;
|
|
break;
|
|
|
|
case Mode::Irq:
|
|
RESTORE_BANKED(irq, IRQ);
|
|
spsr = spsr_banked.irq;
|
|
break;
|
|
|
|
case Mode::Undefined:
|
|
RESTORE_BANKED(und, UND);
|
|
spsr = spsr_banked.und;
|
|
break;
|
|
|
|
case Mode::User:
|
|
case Mode::System:
|
|
STORE_BANKED(old, SYS_USR);
|
|
break;
|
|
}
|
|
|
|
#undef RESTORE_BANKED
|
|
|
|
cpsr.set_mode(to);
|
|
}
|
|
|
|
void
|
|
CpuImpl::step() {
|
|
// Current instruction is two instructions behind PC
|
|
uint32_t cur_pc = pc - 2 * arm::INSTRUCTION_SIZE;
|
|
|
|
if (cpsr.state() == State::Arm) {
|
|
debug(cur_pc);
|
|
uint32_t x = bus->read_word(cur_pc);
|
|
arm::Instruction instruction(x);
|
|
log_info("{:#034b}", x);
|
|
|
|
exec_arm(instruction);
|
|
|
|
log_info("0x{:08X} : {}", cur_pc, instruction.disassemble());
|
|
|
|
if (is_flushed) {
|
|
// if flushed, do not increment the PC, instead set it to two
|
|
// instructions ahead to account for flushed "fetch" and "decode"
|
|
// instructions
|
|
pc += 2 * arm::INSTRUCTION_SIZE;
|
|
is_flushed = false;
|
|
} else {
|
|
// if not flushed continue like normal
|
|
pc += arm::INSTRUCTION_SIZE;
|
|
}
|
|
}
|
|
}
|
|
}
|