mirror of
https://github.com/TomHarte/CLK.git
synced 2025-04-05 04:37:41 +00:00
Add a mere calculate effective address pathway.
Plus a lot of waffle to try to justify the further code duplication.
This commit is contained in:
parent
c7c12f9638
commit
9d79e64f5c
@ -38,6 +38,30 @@ enum ExecutionState: int {
|
||||
BusOrAddressErrorException,
|
||||
|
||||
// Specific addressing mode fetches.
|
||||
//
|
||||
// Additional context here is that I'm very much on the fence but
|
||||
// for now am telling myself:
|
||||
//
|
||||
// (1) the overwhelming majority of instructions that need an
|
||||
// effective address calculation use it for an operand read
|
||||
// immediately afterwards, so keeping those things bound
|
||||
// avoids a large number of conditional branches; and
|
||||
// (2) making a decision between byte/word and long-word once at
|
||||
// the outset also saves a conditional for any two-operand
|
||||
// instructions (which is also the majority); but
|
||||
// (3) some instructions do just need the address calculation —
|
||||
// LEA and PEA are obvious examples, but are not the
|
||||
// exhaustive list — so a third route just to do the
|
||||
// calculation is necessary.
|
||||
//
|
||||
// My internal dialogue then argues that each of these is actually
|
||||
// a small amount of code, so the need manually to duplicate (per
|
||||
// the control-flow constraints of using a switch as a coroutine)
|
||||
// isn't too ugly. Possibly even less ugly than pulling things out
|
||||
// with a macro, especially for debugging.
|
||||
//
|
||||
// Further consideration may be necessary. Especially once this is
|
||||
// up on its feet and profiling becomes an option.
|
||||
|
||||
FetchAddressRegisterIndirect_bw,
|
||||
FetchAddressRegisterIndirectWithPostincrement_bw,
|
||||
@ -61,6 +85,17 @@ enum ExecutionState: int {
|
||||
FetchAbsoluteLong_l,
|
||||
FetchImmediateData_l,
|
||||
|
||||
CalcEffectiveAddress,
|
||||
CalcAddressRegisterIndirect,
|
||||
CalcAddressRegisterIndirectWithPostincrement,
|
||||
CalcAddressRegisterIndirectWithPredecrement,
|
||||
CalcAddressRegisterIndirectWithDisplacement,
|
||||
CalcAddressRegisterIndirectWithIndex8bitDisplacement,
|
||||
CalcProgramCounterIndirectWithDisplacement,
|
||||
CalcProgramCounterIndirectWithIndex8bitDisplacement,
|
||||
CalcAbsoluteShort,
|
||||
CalcAbsoluteLong,
|
||||
|
||||
// Various forms of perform; each of these will
|
||||
// perform the current instruction, then do the
|
||||
// indicated bus cycle.
|
||||
@ -719,6 +754,32 @@ void Processor<BusHandler, dtack_is_implicit, permit_overrun, signal_will_perfor
|
||||
}
|
||||
break;
|
||||
|
||||
BeginState(CalcEffectiveAddress):
|
||||
switch(instruction_.mode(next_operand_)) {
|
||||
default:
|
||||
state_ = post_ea_state_;
|
||||
break;
|
||||
|
||||
case Mode::AddressRegisterIndirect:
|
||||
MoveToState(CalcAddressRegisterIndirect);
|
||||
case Mode::AddressRegisterIndirectWithPostincrement:
|
||||
MoveToState(CalcAddressRegisterIndirectWithPostincrement);
|
||||
case Mode::AddressRegisterIndirectWithPredecrement:
|
||||
MoveToState(CalcAddressRegisterIndirectWithPredecrement);
|
||||
case Mode::AddressRegisterIndirectWithDisplacement:
|
||||
MoveToState(CalcAddressRegisterIndirectWithDisplacement);
|
||||
case Mode::AddressRegisterIndirectWithIndex8bitDisplacement:
|
||||
MoveToState(CalcAddressRegisterIndirectWithIndex8bitDisplacement);
|
||||
case Mode::ProgramCounterIndirectWithDisplacement:
|
||||
MoveToState(CalcProgramCounterIndirectWithDisplacement);
|
||||
case Mode::ProgramCounterIndirectWithIndex8bitDisplacement:
|
||||
MoveToState(CalcProgramCounterIndirectWithIndex8bitDisplacement);
|
||||
case Mode::AbsoluteShort:
|
||||
MoveToState(CalcAbsoluteShort);
|
||||
case Mode::AbsoluteLong:
|
||||
MoveToState(CalcAbsoluteLong);
|
||||
}
|
||||
|
||||
// MARK: - Fetch, addressing modes.
|
||||
|
||||
//
|
||||
@ -741,13 +802,18 @@ void Processor<BusHandler, dtack_is_implicit, permit_overrun, signal_will_perfor
|
||||
Access(operand_[next_operand_].low); // nr
|
||||
MoveToNextOperand(FetchOperand_l);
|
||||
|
||||
BeginState(CalcAddressRegisterIndirect):
|
||||
effective_address_[next_operand_] = registers_[8 + instruction_.reg(next_operand_)].l;
|
||||
state_ = post_ea_state_;
|
||||
break;
|
||||
|
||||
//
|
||||
// AddressRegisterIndirectWithPostincrement
|
||||
//
|
||||
BeginState(FetchAddressRegisterIndirectWithPostincrement_bw):
|
||||
effective_address_[next_operand_] = registers_[8 + instruction_.reg(next_operand_)].l;
|
||||
registers_[8 + instruction_.reg(next_operand_)].l +=
|
||||
byte_word_increments[int(instruction_.operand_size())][instruction_.reg(next_operand_)];
|
||||
address_increments[int(instruction_.operand_size())][instruction_.reg(next_operand_)];
|
||||
|
||||
SetDataAddress(effective_address_[next_operand_]);
|
||||
Access(operand_[next_operand_].low); // nr
|
||||
@ -763,12 +829,19 @@ void Processor<BusHandler, dtack_is_implicit, permit_overrun, signal_will_perfor
|
||||
Access(operand_[next_operand_].low); // nr
|
||||
MoveToNextOperand(FetchOperand_l);
|
||||
|
||||
BeginState(CalcAddressRegisterIndirectWithPostincrement):
|
||||
effective_address_[next_operand_] = registers_[8 + instruction_.reg(next_operand_)].l;
|
||||
registers_[8 + instruction_.reg(next_operand_)].l +=
|
||||
address_increments[int(instruction_.operand_size())][instruction_.reg(next_operand_)];
|
||||
state_ = post_ea_state_;
|
||||
break;
|
||||
|
||||
//
|
||||
// AddressRegisterIndirectWithPredecrement
|
||||
//
|
||||
BeginState(FetchAddressRegisterIndirectWithPredecrement_bw):
|
||||
registers_[8 + instruction_.reg(next_operand_)].l -=
|
||||
byte_word_increments[int(instruction_.operand_size())][instruction_.reg(next_operand_)];
|
||||
address_increments[int(instruction_.operand_size())][instruction_.reg(next_operand_)];
|
||||
effective_address_[next_operand_] = registers_[8 + instruction_.reg(next_operand_)].l;
|
||||
SetDataAddress(effective_address_[next_operand_]);
|
||||
|
||||
@ -787,6 +860,12 @@ void Processor<BusHandler, dtack_is_implicit, permit_overrun, signal_will_perfor
|
||||
Access(operand_[next_operand_].low); // nr
|
||||
MoveToNextOperand(FetchOperand_l);
|
||||
|
||||
BeginState(CalcAddressRegisterIndirectWithPredecrement):
|
||||
registers_[8 + instruction_.reg(next_operand_)].l -= address_increments[int(instruction_.operand_size())][instruction_.reg(next_operand_)];
|
||||
effective_address_[next_operand_] = registers_[8 + instruction_.reg(next_operand_)].l;
|
||||
state_ = post_ea_state_;
|
||||
break;
|
||||
|
||||
//
|
||||
// AddressRegisterIndirectWithDisplacement
|
||||
//
|
||||
@ -812,6 +891,15 @@ void Processor<BusHandler, dtack_is_implicit, permit_overrun, signal_will_perfor
|
||||
Access(operand_[next_operand_].low); // nr
|
||||
MoveToNextOperand(FetchOperand_l);
|
||||
|
||||
BeginState(CalcAddressRegisterIndirectWithDisplacement):
|
||||
effective_address_[next_operand_] =
|
||||
registers_[8 + instruction_.reg(next_operand_)].l +
|
||||
int16_t(prefetch_.w);
|
||||
Prefetch(); // np
|
||||
|
||||
state_ = post_ea_state_;
|
||||
break;
|
||||
|
||||
//
|
||||
// ProgramCounterIndirectWithDisplacement
|
||||
//
|
||||
@ -837,6 +925,18 @@ void Processor<BusHandler, dtack_is_implicit, permit_overrun, signal_will_perfor
|
||||
Access(operand_[next_operand_].low); // nr
|
||||
MoveToNextOperand(FetchOperand_l);
|
||||
|
||||
BeginState(CalcProgramCounterIndirectWithDisplacement):
|
||||
effective_address_[next_operand_] =
|
||||
program_counter_.l - 2 +
|
||||
int16_t(prefetch_.w);
|
||||
Prefetch(); // np
|
||||
|
||||
state_ = post_ea_state_;
|
||||
break;
|
||||
|
||||
//
|
||||
// AddressRegisterIndirectWithIndex8bitDisplacement
|
||||
//
|
||||
#define d8Xn(base) \
|
||||
base + \
|
||||
((prefetch_.w & 0x800) ? \
|
||||
@ -844,9 +944,6 @@ void Processor<BusHandler, dtack_is_implicit, permit_overrun, signal_will_perfor
|
||||
int16_t(registers_[prefetch_.w >> 12].w)) + \
|
||||
int8_t(prefetch_.b);
|
||||
|
||||
//
|
||||
// AddressRegisterIndirectWithIndex8bitDisplacement
|
||||
//
|
||||
BeginState(FetchAddressRegisterIndirectWithIndex8bitDisplacement_bw):
|
||||
effective_address_[next_operand_] = d8Xn(registers_[8 + instruction_.reg(next_operand_)].l);
|
||||
SetDataAddress(effective_address_[next_operand_]);
|
||||
@ -867,6 +964,12 @@ void Processor<BusHandler, dtack_is_implicit, permit_overrun, signal_will_perfor
|
||||
Access(operand_[next_operand_].low); // nr
|
||||
MoveToNextOperand(FetchOperand_l);
|
||||
|
||||
BeginState(CalcAddressRegisterIndirectWithIndex8bitDisplacement):
|
||||
effective_address_[next_operand_] = d8Xn(registers_[8 + instruction_.reg(next_operand_)].l);
|
||||
IdleBus(1); // n
|
||||
state_ = post_ea_state_;
|
||||
break;
|
||||
|
||||
//
|
||||
// ProgramCounterIndirectWithIndex8bitDisplacement
|
||||
//
|
||||
@ -880,7 +983,7 @@ void Processor<BusHandler, dtack_is_implicit, permit_overrun, signal_will_perfor
|
||||
MoveToNextOperand(FetchOperand_bw);
|
||||
|
||||
BeginState(FetchProgramCounterIndirectWithIndex8bitDisplacement_l):
|
||||
effective_address_[next_operand_] = d8Xn(program_counter_.l - 2);;
|
||||
effective_address_[next_operand_] = d8Xn(program_counter_.l - 2);
|
||||
SetDataAddress(effective_address_[next_operand_]);
|
||||
|
||||
IdleBus(1); // n
|
||||
@ -890,6 +993,12 @@ void Processor<BusHandler, dtack_is_implicit, permit_overrun, signal_will_perfor
|
||||
Access(operand_[next_operand_].low); // nr
|
||||
MoveToNextOperand(FetchOperand_l);
|
||||
|
||||
BeginState(CalcProgramCounterIndirectWithIndex8bitDisplacement):
|
||||
effective_address_[next_operand_] = d8Xn(program_counter_.l - 2);
|
||||
IdleBus(1); // n
|
||||
state_ = post_ea_state_;
|
||||
break;
|
||||
|
||||
#undef d8Xn
|
||||
|
||||
//
|
||||
@ -913,6 +1022,12 @@ void Processor<BusHandler, dtack_is_implicit, permit_overrun, signal_will_perfor
|
||||
Access(operand_[next_operand_].low); // nr
|
||||
MoveToNextOperand(FetchOperand_l);
|
||||
|
||||
BeginState(CalcAbsoluteShort):
|
||||
effective_address_[next_operand_] = int16_t(prefetch_.w);
|
||||
Prefetch(); // np
|
||||
state_ = post_ea_state_;
|
||||
break;
|
||||
|
||||
//
|
||||
// AbsoluteLong
|
||||
//
|
||||
@ -938,6 +1053,13 @@ void Processor<BusHandler, dtack_is_implicit, permit_overrun, signal_will_perfor
|
||||
Access(operand_[next_operand_].low); // nr
|
||||
MoveToNextOperand(FetchOperand_l);
|
||||
|
||||
BeginState(CalcAbsoluteLong):
|
||||
Prefetch(); // np
|
||||
effective_address_[next_operand_] = prefetch_.l;
|
||||
Prefetch(); // np
|
||||
state_ = post_ea_state_;
|
||||
break;
|
||||
|
||||
//
|
||||
// ImmediateData
|
||||
//
|
||||
@ -1079,11 +1201,11 @@ void Processor<BusHandler, dtack_is_implicit, permit_overrun, signal_will_perfor
|
||||
SetupDataAccess(Microcycle::Read, select_flag_);
|
||||
|
||||
SetDataAddress(registers_[8 + instruction_.reg(0)].l);
|
||||
registers_[8 + instruction_.reg(0)].l -= byte_word_increments[int(instruction_.operand_size())][instruction_.reg(0)];
|
||||
registers_[8 + instruction_.reg(0)].l -= address_increments[int(instruction_.operand_size())][instruction_.reg(0)];
|
||||
Access(operand_[0].low); // nr
|
||||
|
||||
SetDataAddress(registers_[8 + instruction_.reg(1)].l);
|
||||
registers_[8 + instruction_.reg(1)].l -= byte_word_increments[int(instruction_.operand_size())][instruction_.reg(1)];
|
||||
registers_[8 + instruction_.reg(1)].l -= address_increments[int(instruction_.operand_size())][instruction_.reg(1)];
|
||||
Access(operand_[1].low); // nr
|
||||
|
||||
Prefetch(); // np
|
||||
|
@ -68,6 +68,10 @@ struct ProcessorBase: public InstructionSet::M68k::NullFlowController {
|
||||
/// into the proper exception.
|
||||
int post_dtack_state_ = 0;
|
||||
|
||||
/// If using CalcEffectiveAddress, this is the state to adopt after the
|
||||
/// effective address for next_operand_ has been calculated.
|
||||
int post_ea_state_ = 0;
|
||||
|
||||
/// The perform state for this operation.
|
||||
int perform_state_ = 0;
|
||||
|
||||
@ -95,47 +99,56 @@ struct ProcessorBase: public InstructionSet::M68k::NullFlowController {
|
||||
/// determine total operation cost.
|
||||
bool did_bit_op_high_ = false;
|
||||
|
||||
// Flow controller... all TODO.
|
||||
using Preinstruction = InstructionSet::M68k::Preinstruction;
|
||||
|
||||
static constexpr uint32_t byte_word_increments[2][8] = {
|
||||
// A lookup table that aids with effective address calculation in
|
||||
// predecrement and postincrement modes; index as [size][register]
|
||||
// and note that [0][7] is 2 rather than 1.
|
||||
static constexpr uint32_t address_increments[3][8] = {
|
||||
{ 1, 1, 1, 1, 1, 1, 1, 2, },
|
||||
{ 2, 2, 2, 2, 2, 2, 2, 2, }
|
||||
{ 2, 2, 2, 2, 2, 2, 2, 2, },
|
||||
{ 4, 4, 4, 4, 4, 4, 4, 4, },
|
||||
};
|
||||
static_assert(int(InstructionSet::M68k::DataSize::Byte) == 0);
|
||||
static_assert(int(InstructionSet::M68k::DataSize::Word) == 1);
|
||||
static_assert(int(InstructionSet::M68k::DataSize::LongWord) == 2);
|
||||
|
||||
/// Used by some dedicated read-modify-write perform patterns to
|
||||
/// determine the size of the bus operation.
|
||||
Microcycle::OperationT select_flag_ = 0;
|
||||
|
||||
template <typename IntT> void did_mulu(IntT) {}
|
||||
template <typename IntT> void did_muls(IntT) {}
|
||||
// Flow controller... many TODO.
|
||||
using Preinstruction = InstructionSet::M68k::Preinstruction;
|
||||
template <typename IntT> void did_mulu(IntT) {} //
|
||||
template <typename IntT> void did_muls(IntT) {} //
|
||||
inline void did_chk(bool, bool);
|
||||
inline void did_scc(bool);
|
||||
inline void did_shift(int) {}
|
||||
template <bool did_overflow> void did_divu(uint32_t, uint32_t) {}
|
||||
template <bool did_overflow> void did_divs(int32_t, int32_t) {}
|
||||
inline void did_shift(int) {} //
|
||||
template <bool did_overflow> void did_divu(uint32_t, uint32_t) {} //
|
||||
template <bool did_overflow> void did_divs(int32_t, int32_t) {} //
|
||||
inline void did_bit_op(int);
|
||||
inline void did_update_status();
|
||||
template <typename IntT> void complete_bcc(bool, IntT);
|
||||
inline void complete_dbcc(bool, bool, int16_t);
|
||||
inline void bsr(uint32_t);
|
||||
inline void jsr(uint32_t) {}
|
||||
inline void jmp(uint32_t) {}
|
||||
inline void rtr() {}
|
||||
inline void rte() {}
|
||||
inline void rts() {}
|
||||
inline void stop() {}
|
||||
inline void reset() {}
|
||||
inline void link(Preinstruction, uint32_t) {}
|
||||
inline void unlink(uint32_t &) {}
|
||||
inline void pea(uint32_t) {}
|
||||
inline void move_to_usp(uint32_t) {}
|
||||
inline void move_from_usp(uint32_t &) {}
|
||||
inline void tas(Preinstruction, uint32_t) {}
|
||||
inline void jsr(uint32_t) {} //
|
||||
inline void jmp(uint32_t) {} //
|
||||
inline void rtr() {} //
|
||||
inline void rte() {} //
|
||||
inline void rts() {} //
|
||||
inline void stop() {} //
|
||||
inline void reset() {} //
|
||||
inline void link(Preinstruction, uint32_t) {} //
|
||||
inline void unlink(uint32_t &) {} //
|
||||
inline void pea(uint32_t) {} //
|
||||
inline void move_to_usp(uint32_t) {} //
|
||||
inline void move_from_usp(uint32_t &) {} //
|
||||
inline void tas(Preinstruction, uint32_t) {} //
|
||||
template <bool use_current_instruction_pc = true> void raise_exception(int) {}
|
||||
|
||||
// These aren't implemented because the specific details of the implementation
|
||||
// mean that the performer call-out isn't necessary.
|
||||
template <typename IntT> void movep(Preinstruction, uint32_t, uint32_t) {}
|
||||
template <typename IntT> void movem_toM(Preinstruction, uint32_t, uint32_t) {}
|
||||
template <typename IntT> void movem_toR(Preinstruction, uint32_t, uint32_t) {}
|
||||
template <bool use_current_instruction_pc = true> void raise_exception(int) {}
|
||||
|
||||
// Some microcycles that will be modified as required and used in the main loop;
|
||||
// the semantics of a switch statement make in-place declarations awkward and
|
||||
|
Loading…
x
Reference in New Issue
Block a user