diff --git a/Machines/ZX8081/ZX8081.cpp b/Machines/ZX8081/ZX8081.cpp index 4142ba511..346030674 100644 --- a/Machines/ZX8081/ZX8081.cpp +++ b/Machines/ZX8081/ZX8081.cpp @@ -59,6 +59,7 @@ int Machine::perform_machine_cycle(const CPU::Z80::MachineCycle &cycle) { uint16_t refresh = 0; uint16_t address = cycle.address ? *cycle.address : 0; + bool is_opcode_read = false; switch(cycle.operation) { case CPU::Z80::MachineCycle::Operation::Output: set_vsync(false); @@ -90,7 +91,8 @@ int Machine::perform_machine_cycle(const CPU::Z80::MachineCycle &cycle) { horizontal_counter_ = 0; break; - case CPU::Z80::MachineCycle::Operation::ReadOpcode: + case CPU::Z80::MachineCycle::Operation::ReadOpcodeStart: + case CPU::Z80::MachineCycle::Operation::ReadOpcodeWait: // The ZX80 and 81 signal an interrupt while refresh is active and bit 6 of the refresh // address is low. The Z80 signals a refresh, providing the refresh address during the // final two cycles of an opcode fetch. Therefore communicate a transient signalling @@ -110,6 +112,7 @@ int Machine::perform_machine_cycle(const CPU::Z80::MachineCycle &cycle) { return 0; } } + is_opcode_read = true; case CPU::Z80::MachineCycle::Operation::Read: if(address < ram_base_) { @@ -120,7 +123,7 @@ int Machine::perform_machine_cycle(const CPU::Z80::MachineCycle &cycle) { // If this is an M1 cycle reading from above the 32kb mark and HALT is not // currently active, perform a video output and return a NOP. Otherwise, // just return the value as read. - if(cycle.operation == CPU::Z80::MachineCycle::Operation::ReadOpcode && address&0x8000 && !(value & 0x40) && !get_halt_line()) { + if(is_opcode_read && address&0x8000 && !(value & 0x40) && !get_halt_line()) { size_t char_address = (size_t)((refresh & 0xff00) | ((value & 0x3f) << 3) | line_counter_); if(char_address < ram_base_) { uint8_t mask = (value & 0x80) ? 0x00 : 0xff; diff --git a/OSBindings/Mac/Clock SignalTests/Bridges/TestMachineZ80.h b/OSBindings/Mac/Clock SignalTests/Bridges/TestMachineZ80.h index 55f6fdcd7..f1d0ec6a3 100644 --- a/OSBindings/Mac/Clock SignalTests/Bridges/TestMachineZ80.h +++ b/OSBindings/Mac/Clock SignalTests/Bridges/TestMachineZ80.h @@ -52,7 +52,6 @@ typedef NS_ENUM(NSInteger, CSTestMachineZ80Register) { - (uint8_t)valueAtAddress:(uint16_t)address; - (void)runForNumberOfCycles:(int)cycles; -- (void)runToNextInstruction; - (void)setValue:(uint16_t)value forRegister:(CSTestMachineZ80Register)reg; - (uint16_t)valueForRegister:(CSTestMachineZ80Register)reg; diff --git a/OSBindings/Mac/Clock SignalTests/Bridges/TestMachineZ80.mm b/OSBindings/Mac/Clock SignalTests/Bridges/TestMachineZ80.mm index 4137bc8a3..fa05a7e38 100644 --- a/OSBindings/Mac/Clock SignalTests/Bridges/TestMachineZ80.mm +++ b/OSBindings/Mac/Clock SignalTests/Bridges/TestMachineZ80.mm @@ -12,7 +12,6 @@ @interface CSTestMachineZ80 () - (void)testMachineDidPerformBusOperation:(CPU::Z80::MachineCycle::Operation)operation - phase:(CPU::Z80::MachineCycle::Phase)phase address:(uint16_t)address value:(uint8_t)value timeStamp:(int)time_stamp; @@ -24,8 +23,8 @@ class BusOperationHandler: public CPU::Z80::AllRAMProcessor::MemoryAccessDelegat public: BusOperationHandler(CSTestMachineZ80 *targetMachine) : target_(targetMachine) {} - void z80_all_ram_processor_did_perform_bus_operation(CPU::Z80::AllRAMProcessor &processor, CPU::Z80::MachineCycle::Operation operation, CPU::Z80::MachineCycle::Phase phase, uint16_t address, uint8_t value, int time_stamp) { - [target_ testMachineDidPerformBusOperation:operation phase:phase address:address value:value timeStamp:time_stamp]; + void z80_all_ram_processor_did_perform_bus_operation(CPU::Z80::AllRAMProcessor &processor, CPU::Z80::MachineCycle::Operation operation, uint16_t address, uint8_t value, int time_stamp) { + [target_ testMachineDidPerformBusOperation:operation address:address value:value timeStamp:time_stamp]; } private: @@ -102,7 +101,6 @@ static CPU::Z80::Register registerForRegister(CSTestMachineZ80Register reg) { BusOperationHandler *_busOperationHandler; NSMutableArray *_busOperationCaptures; - BOOL _isAtReadOpcode; int _timeSeekingReadOpcode; int _lastOpcodeTime; } @@ -173,17 +171,6 @@ static CPU::Z80::Register registerForRegister(CSTestMachineZ80Register reg) { return _processor; } -#pragma mark - Z80-specific Runner - -- (void)runToNextInstruction { - _isAtReadOpcode = NO; - _timeSeekingReadOpcode = 0; - while(!_isAtReadOpcode) { - _timeSeekingReadOpcode++; - _processor->run_for_cycles(1); - } -} - #pragma mark - Bus operation accumulation - (void)setCaptureBusActivity:(BOOL)captureBusActivity { @@ -191,50 +178,44 @@ static CPU::Z80::Register registerForRegister(CSTestMachineZ80Register reg) { _processor->set_memory_access_delegate(captureBusActivity ? _busOperationHandler : nullptr); } -- (void)testMachineDidPerformBusOperation:(CPU::Z80::MachineCycle::Operation)operation phase:(CPU::Z80::MachineCycle::Phase)phase address:(uint16_t)address value:(uint8_t)value timeStamp:(int)timeStamp { +- (void)testMachineDidPerformBusOperation:(CPU::Z80::MachineCycle::Operation)operation address:(uint16_t)address value:(uint8_t)value timeStamp:(int)timeStamp { int length = timeStamp - _lastOpcodeTime; _lastOpcodeTime = timeStamp; - if(operation == CPU::Z80::MachineCycle::Operation::ReadOpcode && length < _timeSeekingReadOpcode) - _isAtReadOpcode = YES; if(self.captureBusActivity) { CSTestMachineZ80BusOperationCapture *capture = [[CSTestMachineZ80BusOperationCapture alloc] init]; - if(phase == CPU::Z80::MachineCycle::Phase::End) { - switch(operation) { - case CPU::Z80::MachineCycle::Operation::Write: - capture.operation = CSTestMachineZ80BusOperationCaptureOperationWrite; - break; + switch(operation) { + case CPU::Z80::MachineCycle::Operation::Write: + capture.operation = CSTestMachineZ80BusOperationCaptureOperationWrite; + break; - case CPU::Z80::MachineCycle::Operation::Read: - capture.operation = CSTestMachineZ80BusOperationCaptureOperationRead; - break; + case CPU::Z80::MachineCycle::Operation::Read: + capture.operation = CSTestMachineZ80BusOperationCaptureOperationRead; + break; - case CPU::Z80::MachineCycle::Operation::ReadOpcode: - case CPU::Z80::MachineCycle::Operation::Refresh: - capture.operation = CSTestMachineZ80BusOperationCaptureOperationReadOpcode; - break; + case CPU::Z80::MachineCycle::Operation::Refresh: + capture.operation = CSTestMachineZ80BusOperationCaptureOperationReadOpcode; + break; - case CPU::Z80::MachineCycle::Operation::Input: - capture.operation = CSTestMachineZ80BusOperationCaptureOperationPortRead; - break; + case CPU::Z80::MachineCycle::Operation::Input: + capture.operation = CSTestMachineZ80BusOperationCaptureOperationPortRead; + break; - case CPU::Z80::MachineCycle::Operation::Output: - capture.operation = CSTestMachineZ80BusOperationCaptureOperationPortWrite; - break; + case CPU::Z80::MachineCycle::Operation::Output: + capture.operation = CSTestMachineZ80BusOperationCaptureOperationPortWrite; + break; - case CPU::Z80::MachineCycle::Operation::Internal: - capture.operation = CSTestMachineZ80BusOperationCaptureOperationInternalOperation; - break; + case CPU::Z80::MachineCycle::Operation::Internal: + capture.operation = CSTestMachineZ80BusOperationCaptureOperationInternalOperation; + break; - default: - return; - } - capture.address = address; - capture.value = value; - capture.timeStamp = timeStamp; - - [_busOperationCaptures addObject:capture]; + default: return; } + capture.address = address; + capture.value = value; + capture.timeStamp = timeStamp; + + [_busOperationCaptures addObject:capture]; } } diff --git a/Processors/Z80/Z80.hpp b/Processors/Z80/Z80.hpp index 0b4a6e776..f7b993ba4 100644 --- a/Processors/Z80/Z80.hpp +++ b/Processors/Z80/Z80.hpp @@ -64,47 +64,62 @@ enum Flag: uint8_t { */ struct MachineCycle { enum Operation { - ReadOpcode = 0, - Refresh, - Read, Write, - Input, Output, + ReadOpcodeStart = 0, + ReadOpcodeWait, + Read, + Write, + Input, + Output, Interrupt, + + Refresh, + Internal, BusAcknowledge, - Internal + + ReadStart, + ReadWait, + WriteStart, + WriteWait, + InputStart, + InputWait, + OutputStart, + OutputWait } operation; - enum Phase { - Start, - Wait, - End - } phase; int length; uint16_t *address; uint8_t *value; bool was_requested; + + inline bool expects_action() const { + return operation <= Operation::Interrupt; + } + inline bool is_terminal() const { + return operation <= Operation::BusAcknowledge; + } }; // Elemental bus operations -#define ReadOpcodeStart() {MachineCycle::ReadOpcode, MachineCycle::Phase::Start, 2, &pc_.full, &operation_, false} -#define ReadOpcodeWait(length, f) {MachineCycle::ReadOpcode, MachineCycle::Phase::Wait, length, &pc_.full, &operation_, f} -#define Refresh(len) {MachineCycle::Refresh, MachineCycle::Phase::End, len, &ir_.full, nullptr, false} +#define ReadOpcodeStart() {MachineCycle::ReadOpcodeStart, 2, &pc_.full, &operation_, false} +#define ReadOpcodeWait(length, f) {MachineCycle::ReadOpcodeWait, length, &pc_.full, &operation_, f} +#define Refresh(len) {MachineCycle::Refresh, len, &ir_.full, nullptr, false} -#define ReadStart(addr, val) {MachineCycle::Read, MachineCycle::Phase::Start, 2, &addr.full, &val, false} -#define ReadWait(l, addr, val, f) {MachineCycle::Read, MachineCycle::Phase::Wait, l, &addr.full, &val, f} -#define ReadEnd(addr, val) {MachineCycle::Read, MachineCycle::Phase::End, 1, &addr.full, &val, false} +#define ReadStart(addr, val) {MachineCycle::ReadStart, 2, &addr.full, &val, false} +#define ReadWait(l, addr, val, f) {MachineCycle::ReadWait, l, &addr.full, &val, f} +#define ReadEnd(addr, val) {MachineCycle::Read, 1, &addr.full, &val, false} -#define WriteStart(addr, val) {MachineCycle::Write, MachineCycle::Phase::Start, 2, &addr.full, &val, false} -#define WriteWait(l, addr, val, f) {MachineCycle::Write, MachineCycle::Phase::Wait, l, &addr.full, &val, f} -#define WriteEnd(addr, val) {MachineCycle::Write, MachineCycle::Phase::End, 1, &addr.full, &val, false} +#define WriteStart(addr, val) {MachineCycle::WriteStart, 2, &addr.full, &val, false} +#define WriteWait(l, addr, val, f) {MachineCycle::WriteWait, l, &addr.full, &val, f} +#define WriteEnd(addr, val) {MachineCycle::Write, 1, &addr.full, &val, false} -#define InputStart(addr, val) {MachineCycle::Input, MachineCycle::Phase::Start, 2, &addr.full, &val, false} -#define InputWait(addr, val, f) {MachineCycle::Input, MachineCycle::Phase::Wait, 1, &addr.full, &val, f} -#define InputEnd(addr, val) {MachineCycle::Input, MachineCycle::Phase::End, 1, &addr.full, &val, false} +#define InputStart(addr, val) {MachineCycle::InputStart, 2, &addr.full, &val, false} +#define InputWait(addr, val, f) {MachineCycle::InputWait, 1, &addr.full, &val, f} +#define InputEnd(addr, val) {MachineCycle::Input, 1, &addr.full, &val, false} -#define OutputStart(addr, val) {MachineCycle::Output, MachineCycle::Phase::Start, 2, &addr.full, &val} -#define OutputWait(addr, val, f) {MachineCycle::Output, MachineCycle::Phase::Wait, 1, &addr.full, &val, f} -#define OutputEnd(addr, val) {MachineCycle::Output, MachineCycle::Phase::End, 1, &addr.full, &val} +#define OutputStart(addr, val) {MachineCycle::OutputStart, 2, &addr.full, &val} +#define OutputWait(addr, val, f) {MachineCycle::OutputWait, 1, &addr.full, &val, f} +#define OutputEnd(addr, val) {MachineCycle::Output, 1, &addr.full, &val} -#define IntAck(length, val) {MachineCycle::Operation::Interrupt, MachineCycle::Phase::End, length, nullptr, &val} +#define IntAck(length, val) {MachineCycle::Interrupt, length, nullptr, &val} // A wrapper to express a bus operation as a micro-op #define BusOp(op) {MicroOp::BusOperation, nullptr, nullptr, op} @@ -119,7 +134,7 @@ struct MachineCycle { #define Input(addr, val) BusOp(InputStart(addr, val)), BusOp(InputWait(addr, val, false)), BusOp(InputWait(addr, val, true)), BusOp(InputEnd(addr, val)) #define Output(addr, val) BusOp(OutputStart(addr, val)), BusOp(OutputWait(addr, val, false)), BusOp(OutputWait(addr, val, true)), BusOp(OutputEnd(addr, val)) -#define InternalOperation(len) {MicroOp::BusOperation, nullptr, nullptr, {MachineCycle::Internal, MachineCycle::Phase::End, len}} +#define InternalOperation(len) {MicroOp::BusOperation, nullptr, nullptr, {MachineCycle::Internal, len}} /// A sequence is a series of micro-ops that ends in a move-to-next-program operation. #define Sequence(...) { __VA_ARGS__, {MicroOp::MoveToNextProgram} } @@ -173,6 +188,7 @@ template class Processor { uint8_t last_request_status_; bool irq_line_; bool bus_request_line_; + bool wait_line_; uint8_t operation_; RegisterPair temp16_, memptr_; @@ -745,6 +761,7 @@ template class Processor { halt_mask_(0xff), number_of_cycles_(0), interrupt_mode_(0), + wait_line_(false), request_status_(Interrupt::PowerOn), last_request_status_(Interrupt::PowerOn), irq_line_(false), @@ -855,7 +872,7 @@ template class Processor { while(1) { while(bus_request_line_) { - static MachineCycle bus_acknowledge_cycle = {MachineCycle::Operation::BusAcknowledge, MachineCycle::Phase::End, 1}; + static MachineCycle bus_acknowledge_cycle = {MachineCycle::Operation::BusAcknowledge, 1}; number_of_cycles_ -= static_cast(this)->perform_machine_cycle(bus_acknowledge_cycle) + 1; if(!number_of_cycles_) { static_cast(this)->flush(); @@ -875,14 +892,18 @@ template class Processor { switch(operation->type) { case MicroOp::BusOperation: - if(operation->machine_cycle.was_requested) { // TODO: && !wait_line_ - continue; - } if(number_of_cycles_ < operation->machine_cycle.length) { scheduled_program_counter_--; static_cast(this)->flush(); return; } + if(operation->machine_cycle.was_requested) { + if(wait_line_) { + scheduled_program_counter_--; + } else { + continue; + } + } number_of_cycles_ -= operation->machine_cycle.length; last_request_status_ = request_status_; number_of_cycles_ -= static_cast(this)->perform_machine_cycle(operation->machine_cycle); @@ -1892,6 +1913,13 @@ template class Processor { last_request_status_ &= ~Interrupt::PowerOn; } + /*! + Sets the logical value of the wait line. + */ + inline void set_wait_line(bool value) { + wait_line_ = value; + } + /*! For receivers of perform_machine_cycle only. Temporarily rejects the current machine cycle, causing time to be rewinded to its beginning. diff --git a/Processors/Z80/Z80AllRAM.cpp b/Processors/Z80/Z80AllRAM.cpp index e6b315bfc..0fc493bf4 100644 --- a/Processors/Z80/Z80AllRAM.cpp +++ b/Processors/Z80/Z80AllRAM.cpp @@ -17,45 +17,47 @@ class ConcreteAllRAMProcessor: public AllRAMProcessor, public Processor> 8; - break; - - case MachineCycle::Operation::Internal: - case MachineCycle::Operation::Refresh: - break; - - case MachineCycle::Operation::Interrupt: - // A pick that means LD HL, (nn) if interpreted as an instruction but is otherwise - // arbitrary. - *cycle.value = 0x21; - break; - - default: - printf("???\n"); - break; - } -// } timestamp_ += cycle.length; + if(!cycle.is_terminal()) { + return 0; + } + + uint16_t address = cycle.address ? *cycle.address : 0x0000; + switch(cycle.operation) { + case MachineCycle::Operation::ReadOpcodeStart: + check_address_for_trap(address); + case MachineCycle::Operation::Read: + *cycle.value = memory_[address]; + break; + case MachineCycle::Operation::Write: + memory_[address] = *cycle.value; + break; + + case MachineCycle::Operation::Output: + break; + case MachineCycle::Operation::Input: + // This logic is selected specifically because it seems to match + // the FUSE unit tests. It might need factoring out. + *cycle.value = address >> 8; + break; + + case MachineCycle::Operation::Internal: + case MachineCycle::Operation::Refresh: + break; + + case MachineCycle::Operation::Interrupt: + // A pick that means LD HL, (nn) if interpreted as an instruction but is otherwise + // arbitrary. + *cycle.value = 0x21; + break; + + default: + printf("???\n"); + break; + } if(delegate_ != nullptr) { - delegate_->z80_all_ram_processor_did_perform_bus_operation(*this, cycle.operation, cycle.phase, address, cycle.value ? *cycle.value : 0x00, timestamp_); + delegate_->z80_all_ram_processor_did_perform_bus_operation(*this, cycle.operation, address, cycle.value ? *cycle.value : 0x00, timestamp_); } return 0; diff --git a/Processors/Z80/Z80AllRAM.hpp b/Processors/Z80/Z80AllRAM.hpp index 3c77f943c..e4890fa73 100644 --- a/Processors/Z80/Z80AllRAM.hpp +++ b/Processors/Z80/Z80AllRAM.hpp @@ -22,7 +22,7 @@ class AllRAMProcessor: static AllRAMProcessor *Processor(); struct MemoryAccessDelegate { - virtual void z80_all_ram_processor_did_perform_bus_operation(CPU::Z80::AllRAMProcessor &processor, CPU::Z80::MachineCycle::Operation operation, CPU::Z80::MachineCycle::Phase phase, uint16_t address, uint8_t value, int time_stamp) = 0; + virtual void z80_all_ram_processor_did_perform_bus_operation(CPU::Z80::AllRAMProcessor &processor, CPU::Z80::MachineCycle::Operation operation, uint16_t address, uint8_t value, int time_stamp) = 0; }; inline void set_memory_access_delegate(MemoryAccessDelegate *delegate) { delegate_ = delegate;