From 59e1a5e5f677acef1a409e2d99cf58ccd0a2173d Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Fri, 26 Jul 2024 21:43:54 -0400 Subject: [PATCH 1/2] Correct 6502 JAM behaviour. --- .../6502/Implementation/6502Implementation.hpp | 18 +++++++++++------- Processors/6502/Implementation/6502Storage.cpp | 2 +- Processors/6502/Implementation/6502Storage.hpp | 9 ++++++--- 3 files changed, 18 insertions(+), 11 deletions(-) diff --git a/Processors/6502/Implementation/6502Implementation.hpp b/Processors/6502/Implementation/6502Implementation.hpp index 223701bb8..9af00aea8 100644 --- a/Processors/6502/Implementation/6502Implementation.hpp +++ b/Processors/6502/Implementation/6502Implementation.hpp @@ -59,10 +59,7 @@ void Processor::run_for(const Cycles cycles) { }; const auto throwaway_read = [&](uint16_t address) { - next_bus_operation_ = BusOperation::Read; - bus_address_ = address; - bus_value_ = &bus_throwaway_; - bus_throwaway_ = 0xff; + read_mem(bus_throwaway_, address); }; const auto write_mem = [&](uint8_t &val, uint16_t address) { @@ -232,10 +229,13 @@ void Processor::run_for(const Cycles cycles) { // MARK: - JAM, WAI, STP - case OperationScheduleJam: { + case OperationSetJAMmed: is_jammed_ = true; - scheduled_program_counter_ = operations_[CPU::MOS6502::JamOpcode]; - } continue; + scheduled_program_counter_ -= 2; + continue; + + case CycleFetchFFFE: read_mem(bus_throwaway_, 0xfffe); break; + case CycleFetchFFFF: read_mem(bus_throwaway_, 0xffff); break; case OperationScheduleStop: stop_is_active_ = true; @@ -804,6 +804,9 @@ template void Proces void ProcessorBase::set_reset_line(bool active) { interrupt_requests_ = (interrupt_requests_ & ~InterruptRequestFlags::Reset) | (active ? InterruptRequestFlags::Reset : 0); + if(is_jammed_) { + restart_operation_fetch(); + } } bool ProcessorBase::get_is_resetting() const { @@ -870,6 +873,7 @@ void ProcessorBase::set_value_of(Register r, uint16_t value) { } void ProcessorBase::restart_operation_fetch() { + is_jammed_ = false; scheduled_program_counter_ = nullptr; next_bus_operation_ = BusOperation::None; } diff --git a/Processors/6502/Implementation/6502Storage.cpp b/Processors/6502/Implementation/6502Storage.cpp index a02178842..f1a9e8835 100644 --- a/Processors/6502/Implementation/6502Storage.cpp +++ b/Processors/6502/Implementation/6502Storage.cpp @@ -73,7 +73,7 @@ using namespace CPU::MOS6502; #define ImpliedNop() {OperationMoveToNextProgram} #define ImmediateNop() Program(OperationIncrementPC) -#define JAM {CycleFetchOperand, OperationScheduleJam} +#define JAM {CycleFetchFFFF, CycleFetchFFFE, CycleFetchFFFE, CycleFetchFFFF, OperationSetJAMmed} ProcessorStorage::ProcessorStorage(Personality personality) { const InstructionList operations_6502[] = { diff --git a/Processors/6502/Implementation/6502Storage.hpp b/Processors/6502/Implementation/6502Storage.hpp index fd3c9a78d..6d7a163fb 100644 --- a/Processors/6502/Implementation/6502Storage.hpp +++ b/Processors/6502/Implementation/6502Storage.hpp @@ -198,9 +198,12 @@ class ProcessorStorage { OperationSetFlagsFromX, // sets the zero and negative flags based on the value of x OperationSetFlagsFromY, // sets the zero and negative flags based on the value of y - OperationScheduleJam, // schedules the program for operation F2 - OperationScheduleWait, // puts the processor into WAI mode (i.e. it'll do nothing until an interrupt is received) - OperationScheduleStop, // puts the processor into STP mode (i.e. it'll do nothing until a reset is received) + OperationScheduleWait, // puts the processor into WAI mode (i.e. it'll do nothing until an interrupt is received) + OperationScheduleStop, // puts the processor into STP mode (i.e. it'll do nothing until a reset is received) + + CycleFetchFFFE, // perform a throwaway read from $FFFE + CycleFetchFFFF, // perform a throwaway read from $FFFF + OperationSetJAMmed, // decrements the micro-operation program counter back to the operation before this one, marking the CPU as jammed }; using InstructionList = MicroOp[12]; From 76ca60702136818b807c592b84420cca40e4279f Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Fri, 26 Jul 2024 21:45:03 -0400 Subject: [PATCH 2/2] Add a graceful end for JAM. --- .../Mac/Clock SignalTests/65816ComparativeTests.mm | 11 +++++++++++ 1 file changed, 11 insertions(+) diff --git a/OSBindings/Mac/Clock SignalTests/65816ComparativeTests.mm b/OSBindings/Mac/Clock SignalTests/65816ComparativeTests.mm index fb35314b5..6f9d702bd 100644 --- a/OSBindings/Mac/Clock SignalTests/65816ComparativeTests.mm +++ b/OSBindings/Mac/Clock SignalTests/65816ComparativeTests.mm @@ -29,6 +29,16 @@ struct BusHandler: public CPU::MOS6502Esque::BusHandlerT { std::unordered_map inventions; Cycles perform_bus_operation(CPU::MOS6502Esque::BusOperation operation, AddressType address, uint8_t *value) { + // Check for a JAM; if one is found then record just five more bus cycles, arbitrarily. + if(jam_count) { + --jam_count; + if(!jam_count) { + throw StopException(); + } + } else if(processor.is_jammed()) { + jam_count = 5; + } + // Record the basics of the operation. auto &cycle = cycles.emplace_back(); cycle.operation = operation; @@ -110,6 +120,7 @@ struct BusHandler: public CPU::MOS6502Esque::BusHandlerT { int pc_overshoot = 0; std::optional initial_pc; bool allow_pc_repetition = false; + int jam_count = 0; struct Cycle { CPU::MOS6502Esque::BusOperation operation;