From 8789ffda152edde20b151c2cca0157b955a5268e Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 24 Jan 2021 22:30:42 -0500 Subject: [PATCH] Corrects performer storage, RMW/W confusion, implicit casts, port readback. --- InstructionSets/CachingExecutor.hpp | 6 ++++-- InstructionSets/M50740/Executor.cpp | 30 +++++++++++++++----------- InstructionSets/M50740/Executor.hpp | 5 +++-- InstructionSets/M50740/Instruction.hpp | 4 ++-- Machines/Apple/AppleIIgs/ADB.cpp | 5 +++-- Machines/Apple/AppleIIgs/ADB.hpp | 1 + 6 files changed, 30 insertions(+), 21 deletions(-) diff --git a/InstructionSets/CachingExecutor.hpp b/InstructionSets/CachingExecutor.hpp index 33629f0ff..a32f93eaa 100644 --- a/InstructionSets/CachingExecutor.hpp +++ b/InstructionSets/CachingExecutor.hpp @@ -93,7 +93,7 @@ template < // Storage for the statically-allocated list of performers. It's a bit more // work for executors to fill this array, but subsequently performers can be // indexed by array position, which is a lot more compact than a generic pointer. - std::array performers_; + std::array performers_; ProgramCounterType program_counter_; /*! @@ -149,10 +149,12 @@ template < while(remaining_duration_ > 0) { has_branched_ = false; + Executor *const executor = static_cast(this); while(remaining_duration_ > 0 && !has_branched_) { const auto performer = performers_[program_[program_index_]]; ++program_index_; - (static_cast(this)->*performer)(); + + (executor->*performer)(); } } } diff --git a/InstructionSets/M50740/Executor.cpp b/InstructionSets/M50740/Executor.cpp index 79c3edbb6..3e9339bda 100644 --- a/InstructionSets/M50740/Executor.cpp +++ b/InstructionSets/M50740/Executor.cpp @@ -305,9 +305,9 @@ template void Executor::pe // sequence below that wraps the address and checks whether // a write is valid [if required]. - int address; + unsigned int address; #define next8() memory_[(program_counter_ + 1) & 0x1fff] -#define next16() (memory_[(program_counter_ + 1) & 0x1fff] | (memory_[(program_counter_ + 2) & 0x1fff] << 8)) +#define next16() uint16_t(memory_[(program_counter_ + 1) & 0x1fff] | (memory_[(program_counter_ + 2) & 0x1fff] << 8)) // Underlying assumption below: the instruction stream will never // overlap with IO ports. @@ -333,7 +333,7 @@ template void Executor::pe // Special-purpose addressing modes. case AddressingMode::Relative: - address = program_counter_ + 1 + size(addressing_mode) + int8_t(next8()); + address = unsigned(program_counter_ + 1 + size(addressing_mode) + int8_t(next8())); break; case AddressingMode::SpecialPage: address = 0x1f00 | next8(); break; @@ -350,16 +350,16 @@ template void Executor::pe uint8_t value; if constexpr (addressing_mode == AddressingMode::AccumulatorRelative) { value = a_; - address = program_counter_ + 1 + size(addressing_mode) + int8_t(next8()); + address = unsigned(program_counter_ + 1 + size(addressing_mode) + int8_t(next8())); } else { value = read(next8()); - address = program_counter_ + 1 + size(addressing_mode) + int8_t(memory_[(program_counter_+2)&0x1fff]); + address = unsigned(program_counter_ + 1 + size(addressing_mode) + int8_t(memory_[(program_counter_+2)&0x1fff])); } program_counter_ += 1 + size(addressing_mode); switch(operation) { case Operation::BBS0: case Operation::BBS1: case Operation::BBS2: case Operation::BBS3: case Operation::BBS4: case Operation::BBS5: case Operation::BBS6: case Operation::BBS7: - if constexpr (operation >= Operation::BBS0 && operation < Operation::BBS7) { + if constexpr (operation >= Operation::BBS0 && operation <= Operation::BBS7) { if(value & (1 << (int(operation) - int(Operation::BBS0)))) { set_program_counter(uint16_t(address)); subtract_duration(2); @@ -368,7 +368,7 @@ template void Executor::pe return; case Operation::BBC0: case Operation::BBC1: case Operation::BBC2: case Operation::BBC3: case Operation::BBC4: case Operation::BBC5: case Operation::BBC6: case Operation::BBC7: - if constexpr (operation >= Operation::BBC0 && operation < Operation::BBS7) { + if constexpr (operation >= Operation::BBC0 && operation <= Operation::BBS7) { if(value & (1 << (int(operation) - int(Operation::BBC0)))) { set_program_counter(uint16_t(address)); subtract_duration(2); @@ -390,21 +390,21 @@ template void Executor::pe case AddressingMode::ZeroPageIndirect: address = next8(); - address = memory_[address] | (memory_[(address + 1) & 0xff] << 8); + address = unsigned(memory_[address] | (memory_[(address + 1) & 0xff] << 8)); break; case AddressingMode::XIndirect: address = (next8() + x_) & 0xff; - address = memory_[address] | (memory_[(address + 1)&0xff] << 8); + address = unsigned(memory_[address] | (memory_[(address + 1)&0xff] << 8)); break; case AddressingMode::IndirectY: - address = (memory_[next8()] | (memory_[(next8()+1)&0xff] << 8)) + y_; + address = unsigned((memory_[next8()] | (memory_[(next8()+1)&0xff] << 8)) + y_); break; case AddressingMode::AbsoluteIndirect: address = next16(); - address = memory_[address] | (memory_[(address + 1) & 0x1fff] << 8); + address = unsigned(memory_[address] | (memory_[(address + 1) & 0x1fff] << 8)); break; default: @@ -488,11 +488,15 @@ template void Executor::perform(uint8_t *operand [[maybe_u case Operation::SEB0: case Operation::SEB1: case Operation::SEB2: case Operation::SEB3: case Operation::SEB4: case Operation::SEB5: case Operation::SEB6: case Operation::SEB7: - *operand |= 1 << (int(operation) - int(Operation::SEB0)); + if constexpr(operation >= Operation::SEB0 && operation <= Operation::SEB7) { + *operand |= 1 << (int(operation) - int(Operation::SEB0)); + } break; case Operation::CLB0: case Operation::CLB1: case Operation::CLB2: case Operation::CLB3: case Operation::CLB4: case Operation::CLB5: case Operation::CLB6: case Operation::CLB7: - *operand &= ~(1 << (int(operation) - int(Operation::CLB0))); + if constexpr(operation >= Operation::CLB0 && operation <= Operation::CLB7) { + *operand &= ~(1 << (int(operation) - int(Operation::CLB0))); + } break; case Operation::CLI: interrupt_disable_ = 0x00; break; diff --git a/InstructionSets/M50740/Executor.hpp b/InstructionSets/M50740/Executor.hpp index 106fe55c9..a15fa73de 100644 --- a/InstructionSets/M50740/Executor.hpp +++ b/InstructionSets/M50740/Executor.hpp @@ -14,6 +14,7 @@ #include "../CachingExecutor.hpp" #include "../../ClockReceiver/ClockReceiver.hpp" +#include #include #include @@ -63,7 +64,7 @@ class Executor: public CachingExecutor { */ inline void parse(uint16_t start, uint16_t closing_bound) { Parser parser; - parser.parse(*this, memory_, start & 0x1fff, closing_bound); + parser.parse(*this, &memory_[0], start & 0x1fff, closing_bound); } private: @@ -121,7 +122,7 @@ class Executor: public CachingExecutor { // MARK: - Instruction set state. // Memory. - uint8_t memory_[0x2000]; + std::array memory_; // Registers. uint8_t a_, x_, y_, s_; diff --git a/InstructionSets/M50740/Instruction.hpp b/InstructionSets/M50740/Instruction.hpp index f7a6ecd39..f38ff4f10 100644 --- a/InstructionSets/M50740/Instruction.hpp +++ b/InstructionSets/M50740/Instruction.hpp @@ -94,8 +94,8 @@ static constexpr auto MinOperation = int(Operation::BBC0); constexpr AccessType access_type(Operation operation) { if(operation < Operation::ADC) return AccessType::None; if(operation < Operation::ASL) return AccessType::Read; - if(operation < Operation::LDM) return AccessType::Write; - return AccessType::ReadModifyWrite; + if(operation < Operation::LDM) return AccessType::ReadModifyWrite; + return AccessType::Write; } constexpr bool uses_index_mode(Operation operation) { diff --git a/Machines/Apple/AppleIIgs/ADB.cpp b/Machines/Apple/AppleIIgs/ADB.cpp index a81798675..0247211ba 100644 --- a/Machines/Apple/AppleIIgs/ADB.cpp +++ b/Machines/Apple/AppleIIgs/ADB.cpp @@ -226,6 +226,7 @@ void GLU::run_for(Cycles cycles) { // MARK: - M50470 port handler void GLU::set_port_output(int port, uint8_t value) { + ports_[port] = value; switch(port) { case 0: printf("Set R%d: %02x\n", register_address_, value); @@ -261,10 +262,10 @@ uint8_t GLU::get_port_input(int port) { return 0x06; case 2: printf("ADB data line input, etc\n"); - return 0xff; + return ports_[2]; case 3: // printf("ADB data line output, etc\n"); - break; + return ports_[3]; default: assert(false); } diff --git a/Machines/Apple/AppleIIgs/ADB.hpp b/Machines/Apple/AppleIIgs/ADB.hpp index 6d255513c..4239bdb75 100644 --- a/Machines/Apple/AppleIIgs/ADB.hpp +++ b/Machines/Apple/AppleIIgs/ADB.hpp @@ -67,6 +67,7 @@ class GLU: public InstructionSet::M50740::PortHandler { uint8_t registers_[16]; uint8_t register_address_; + uint8_t ports_[4]; }; }