From 9d79e64f5ce2926b0abdbdb7c735682b0ea0fa7e Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Fri, 20 May 2022 16:23:52 -0400 Subject: [PATCH] Add a mere calculate effective address pathway. Plus a lot of waffle to try to justify the further code duplication. --- .../Implementation/68000Mk2Implementation.hpp | 138 +++++++++++++++++- .../Implementation/68000Mk2Storage.hpp | 61 +++++--- 2 files changed, 167 insertions(+), 32 deletions(-) diff --git a/Processors/68000Mk2/Implementation/68000Mk2Implementation.hpp b/Processors/68000Mk2/Implementation/68000Mk2Implementation.hpp index 294f9dce8..d892e647a 100644 --- a/Processors/68000Mk2/Implementation/68000Mk2Implementation.hpp +++ b/Processors/68000Mk2/Implementation/68000Mk2Implementation.hpp @@ -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> 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 void did_mulu(IntT) {} - template void did_muls(IntT) {} + // Flow controller... many TODO. + using Preinstruction = InstructionSet::M68k::Preinstruction; + template void did_mulu(IntT) {} // + template void did_muls(IntT) {} // inline void did_chk(bool, bool); inline void did_scc(bool); - inline void did_shift(int) {} - template void did_divu(uint32_t, uint32_t) {} - template void did_divs(int32_t, int32_t) {} + inline void did_shift(int) {} // + template void did_divu(uint32_t, uint32_t) {} // + template void did_divs(int32_t, int32_t) {} // inline void did_bit_op(int); inline void did_update_status(); template 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 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 void movep(Preinstruction, uint32_t, uint32_t) {} template void movem_toM(Preinstruction, uint32_t, uint32_t) {} template void movem_toR(Preinstruction, uint32_t, uint32_t) {} - template 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