diff --git a/Components/6522/6522.hpp b/Components/6522/6522.hpp index 8e0811e66..7d525d4aa 100644 --- a/Components/6522/6522.hpp +++ b/Components/6522/6522.hpp @@ -128,7 +128,7 @@ template class MOS6522: public MOS6522Storage { void access(int address); - uint8_t get_port_input(Port port, uint8_t output_mask, uint8_t output); + uint8_t get_port_input(Port port, uint8_t output_mask, uint8_t output, uint8_t timer_mask); inline void reevaluate_interrupts(); /// Sets the current intended output value for the port and line; diff --git a/Components/6522/Implementation/6522Implementation.hpp b/Components/6522/Implementation/6522Implementation.hpp index 533b8d1be..973bd5bce 100644 --- a/Components/6522/Implementation/6522Implementation.hpp +++ b/Components/6522/Implementation/6522Implementation.hpp @@ -85,6 +85,11 @@ template void MOS6522::write(int address, uint8_t value) { registers_.next_timer[0] = registers_.timer_latch[0]; timer_is_running_[0] = true; + // If PB7 output mode is active, set it low. + if(registers_.auxiliary_control & 0x80) { + registers_.timer_port_b_output &= 0x7f; + } + // Clear existing interrupt flag. registers_.interrupt_flags &= ~InterruptFlag::Timer1; reevaluate_interrupts(); @@ -113,6 +118,12 @@ template void MOS6522::write(int address, uint8_t value) { case 0xb: // Auxiliary control ('ACR'). registers_.auxiliary_control = value; evaluate_cb2_output(); + + // This is a bit of a guess: reset the timer-based PB7 output to its default high level + // any timer that timer-linked PB7 output is disabled. + if(!(registers_.auxiliary_control & 0x80)) { + registers_.timer_port_b_output |= 0x80; + } break; case 0xc: { // Peripheral control ('PCR'). // const auto old_peripheral_control = registers_.peripheral_control; @@ -176,12 +187,12 @@ template uint8_t MOS6522::read(int address) { case 0x0: // Read Port B ('IRB'). registers_.interrupt_flags &= ~(InterruptFlag::CB1ActiveEdge | InterruptFlag::CB2ActiveEdge); reevaluate_interrupts(); - return get_port_input(Port::B, registers_.data_direction[1], registers_.output[1]); + return get_port_input(Port::B, registers_.data_direction[1], registers_.output[1], registers_.auxiliary_control & 0x80); case 0xf: case 0x1: // Read Port A ('IRA'). registers_.interrupt_flags &= ~(InterruptFlag::CA1ActiveEdge | InterruptFlag::CA2ActiveEdge); reevaluate_interrupts(); - return get_port_input(Port::A, registers_.data_direction[0], registers_.output[0]); + return get_port_input(Port::A, registers_.data_direction[0], registers_.output[0], 0); case 0x2: return registers_.data_direction[1]; // Port B direction ('DDRB'). case 0x3: return registers_.data_direction[0]; // Port A direction ('DDRA'). @@ -218,9 +229,10 @@ template uint8_t MOS6522::read(int address) { return 0xff; } -template uint8_t MOS6522::get_port_input(Port port, uint8_t output_mask, uint8_t output) { +template uint8_t MOS6522::get_port_input(Port port, uint8_t output_mask, uint8_t output, uint8_t timer_mask) { bus_handler_.run_for(time_since_bus_handler_call_.flush()); const uint8_t input = bus_handler_.get_port_input(port); + output = (output & ~timer_mask) | (registers_.timer_port_b_output & timer_mask); return (input & ~output_mask) | (output & output_mask); } @@ -354,7 +366,7 @@ template void MOS6522::do_phase1() { // Determine whether to toggle PB7. if(registers_.auxiliary_control&0x80) { - registers_.output[1] ^= 0x80; + registers_.timer_port_b_output ^= 0x80; bus_handler_.run_for(time_since_bus_handler_call_.flush()); bus_handler_.set_port_output(Port::B, registers_.output[1], registers_.data_direction[1]); } diff --git a/Components/6522/Implementation/6522Storage.hpp b/Components/6522/Implementation/6522Storage.hpp index da9e30981..ceff38a8f 100644 --- a/Components/6522/Implementation/6522Storage.hpp +++ b/Components/6522/Implementation/6522Storage.hpp @@ -34,7 +34,9 @@ class MOS6522Storage { uint8_t peripheral_control = 0; uint8_t interrupt_flags = 0; uint8_t interrupt_enable = 0; + bool timer_needs_reload = false; + uint8_t timer_port_b_output = 0xff; } registers_; // Control state. diff --git a/OSBindings/Mac/Clock SignalTests/6522Tests.swift b/OSBindings/Mac/Clock SignalTests/6522Tests.swift index d56ce5903..7126a09c5 100644 --- a/OSBindings/Mac/Clock SignalTests/6522Tests.swift +++ b/OSBindings/Mac/Clock SignalTests/6522Tests.swift @@ -101,6 +101,116 @@ class MOS6522Tests: XCTestCase { } + // MARK: PB7 timer 1 tests + // These follow the same logic and check for the same results as the VICE VIC-20 via_pb7 tests. + + // Perfoms: + // + // (1) establish initial ACR and port B output value, and grab port B input value. + // (2) start timer 1, grab port B input value. + // (3) set final ACR, grab port B input value. + // (4) allow timer 1 to expire, grab port B input value. + private func runTest(startACR: UInt8, endACR: UInt8, portBOutput: UInt8) -> [UInt8] { + var result: [UInt8] = [] + + // Clear all register values. + for n: UInt in 0...15 { + m6522.setValue(0, forRegister: n) + } + m6522.run(forHalfCycles: 2) + + // Set data direction and current port B value. + m6522.setValue(0xff, forRegister: 2) + m6522.run(forHalfCycles: 2) + m6522.setValue(portBOutput, forRegister: 0) + m6522.run(forHalfCycles: 2) + + // Set initial ACR and grab the current port B value. + m6522.setValue(startACR, forRegister: 0xb) + m6522.run(forHalfCycles: 2) + result.append(m6522.value(forRegister: 0)) + m6522.run(forHalfCycles: 2) + + // Start timer 1 and grab the value. + m6522.setValue(1, forRegister: 0x5) + m6522.run(forHalfCycles: 2) + result.append(m6522.value(forRegister: 0)) + m6522.run(forHalfCycles: 2) + + // Set the final ACR value and grab value. + m6522.setValue(endACR, forRegister: 0xb) + m6522.run(forHalfCycles: 2) + result.append(m6522.value(forRegister: 0)) + m6522.run(forHalfCycles: 2) + + // Make sure timer 1 has expired. + m6522.run(forHalfCycles: 512) + + // Grab the final value. + result.append(m6522.value(forRegister: 0)) + + return result + } + + func testTimer1PB7() { + // Original top row. [original Vic-20 screen output in comments on the right] + XCTAssertEqual(runTest(startACR: 0x00, endACR: 0x00, portBOutput: 0x00), [0x00, 0x00, 0x00, 0x00]) // @@@@ (i.e. 0, 0, 0, 0) + XCTAssertEqual(runTest(startACR: 0x00, endACR: 0x40, portBOutput: 0x00), [0x00, 0x00, 0x00, 0x00]) // @@@@ (i.e. 0, 0, 0, 0) + XCTAssertEqual(runTest(startACR: 0x00, endACR: 0x80, portBOutput: 0x00), [0x00, 0x00, 0x80, 0x00]) // @@b@ (i.e. 0, 0, 1, 0) + XCTAssertEqual(runTest(startACR: 0x00, endACR: 0xc0, portBOutput: 0x00), [0x00, 0x00, 0x80, 0x00]) // @@b@ (i.e. 0, 0, 1, 0) + + XCTAssertEqual(runTest(startACR: 0x00, endACR: 0x00, portBOutput: 0xff), [0xff, 0xff, 0xff, 0xff]) // cccc (i.e. 1, 1, 1, 1) + XCTAssertEqual(runTest(startACR: 0x00, endACR: 0x40, portBOutput: 0xff), [0xff, 0xff, 0xff, 0xff]) // cccc (i.e. 1, 1, 1, 1) + XCTAssertEqual(runTest(startACR: 0x00, endACR: 0x80, portBOutput: 0xff), [0xff, 0xff, 0xff, 0x7f]) // ccca (i.e. 1, 1, 1, 0) + XCTAssertEqual(runTest(startACR: 0x00, endACR: 0xc0, portBOutput: 0xff), [0xff, 0xff, 0xff, 0x7f]) // ccca (i.e. 1, 1, 1, 0) + + // Second row. [same output as first row] + XCTAssertEqual(runTest(startACR: 0x40, endACR: 0x00, portBOutput: 0x00), [0x00, 0x00, 0x00, 0x00]) // @@@@ + XCTAssertEqual(runTest(startACR: 0x40, endACR: 0x40, portBOutput: 0x00), [0x00, 0x00, 0x00, 0x00]) // @@@@ + XCTAssertEqual(runTest(startACR: 0x40, endACR: 0x80, portBOutput: 0x00), [0x00, 0x00, 0x80, 0x00]) // @@b@ + XCTAssertEqual(runTest(startACR: 0x40, endACR: 0xc0, portBOutput: 0x00), [0x00, 0x00, 0x80, 0x00]) // @@b@ + + XCTAssertEqual(runTest(startACR: 0x40, endACR: 0x00, portBOutput: 0xff), [0xff, 0xff, 0xff, 0xff]) // cccc + XCTAssertEqual(runTest(startACR: 0x40, endACR: 0x40, portBOutput: 0xff), [0xff, 0xff, 0xff, 0xff]) // cccc + XCTAssertEqual(runTest(startACR: 0x40, endACR: 0x80, portBOutput: 0xff), [0xff, 0xff, 0xff, 0x7f]) // ccca + XCTAssertEqual(runTest(startACR: 0x40, endACR: 0xc0, portBOutput: 0xff), [0xff, 0xff, 0xff, 0x7f]) // ccca + + // Third row. + XCTAssertEqual(runTest(startACR: 0x80, endACR: 0x00, portBOutput: 0x00), [0x80, 0x00, 0x00, 0x00]) // b@@@ (i.e. 1, 0, 0, 0) + XCTAssertEqual(runTest(startACR: 0x80, endACR: 0x40, portBOutput: 0x00), [0x80, 0x00, 0x00, 0x00]) // b@@@ (i.e. 1, 0, 0, 0) + XCTAssertEqual(runTest(startACR: 0x80, endACR: 0x80, portBOutput: 0x00), [0x80, 0x00, 0x00, 0x80]) // b@@b (i.e. 1, 0, 0, 1) + XCTAssertEqual(runTest(startACR: 0x80, endACR: 0xc0, portBOutput: 0x00), [0x80, 0x00, 0x00, 0x80]) // b@@b (i.e. 1, 0, 0, 1) + + XCTAssertEqual(runTest(startACR: 0x80, endACR: 0x00, portBOutput: 0xff), [0xff, 0x7f, 0xff, 0xff]) // cacc (i.e. 1, 0, 1, 1) + XCTAssertEqual(runTest(startACR: 0x80, endACR: 0x40, portBOutput: 0xff), [0xff, 0x7f, 0xff, 0xff]) // cacc (i.e. 1, 0, 1, 1) + XCTAssertEqual(runTest(startACR: 0x80, endACR: 0x80, portBOutput: 0xff), [0xff, 0x7f, 0x7f, 0xff]) // caac (i.e. 1, 0, 0, 1) + XCTAssertEqual(runTest(startACR: 0x80, endACR: 0xc0, portBOutput: 0xff), [0xff, 0x7f, 0x7f, 0xff]) // caac (i.e. 1, 0, 0, 1) + + // Final row. [same output as third row] + XCTAssertEqual(runTest(startACR: 0xc0, endACR: 0x00, portBOutput: 0x00), [0x80, 0x00, 0x00, 0x00]) // b@@@ + XCTAssertEqual(runTest(startACR: 0xc0, endACR: 0x40, portBOutput: 0x00), [0x80, 0x00, 0x00, 0x00]) // b@@@ + XCTAssertEqual(runTest(startACR: 0xc0, endACR: 0x80, portBOutput: 0x00), [0x80, 0x00, 0x00, 0x80]) // b@@b + XCTAssertEqual(runTest(startACR: 0xc0, endACR: 0xc0, portBOutput: 0x00), [0x80, 0x00, 0x00, 0x80]) // b@@b + + XCTAssertEqual(runTest(startACR: 0xc0, endACR: 0x00, portBOutput: 0xff), [0xff, 0x7f, 0xff, 0xff]) // cacc + XCTAssertEqual(runTest(startACR: 0xc0, endACR: 0x40, portBOutput: 0xff), [0xff, 0x7f, 0xff, 0xff]) // cacc + XCTAssertEqual(runTest(startACR: 0xc0, endACR: 0x80, portBOutput: 0xff), [0xff, 0x7f, 0x7f, 0xff]) // caac + XCTAssertEqual(runTest(startACR: 0xc0, endACR: 0xc0, portBOutput: 0xff), [0xff, 0x7f, 0x7f, 0xff]) // caac + + // Conclusions: + // + // after inital ACR and port B value: [original data if not in PB7 output mode, otherwise 1] + // after starting timer 1: [original data if not in PB7 output mode, otherwise 0] + // after final ACR value: [original data if not in PB7 output mode, 1 if has transitioned to PB7 mode, 0 if was already in PB7 mode] + // after timer 1 expiry: [original data if not in PB7 mode, 1 if timer has expired while in PB7 mode] + // + // i.e. + // (1) there is separate storage for the programmer-set PB7 and the timer output; + // (2) the timer output is reset upon a timer write only if PB7 output is enabled; + // (3) expiry toggles the output. + } + + // MARK: Data direction tests func testDataDirection() { // set low four bits of register B as output, the top four as input