mirror of
https://github.com/TomHarte/CLK.git
synced 2025-04-04 13:31:26 +00:00
Takes a run at timer-linked PB7 output behaviour.
Seemingly sufficiently to pass the VICE test (which I've transcribed), though with some guesswork.
This commit is contained in:
parent
291aa42fe1
commit
703065a0a5
@ -128,7 +128,7 @@ template <class T> 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;
|
||||
|
@ -85,6 +85,11 @@ template <typename T> void MOS6522<T>::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 <typename T> void MOS6522<T>::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 <typename T> uint8_t MOS6522<T>::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 <typename T> uint8_t MOS6522<T>::read(int address) {
|
||||
return 0xff;
|
||||
}
|
||||
|
||||
template <typename T> uint8_t MOS6522<T>::get_port_input(Port port, uint8_t output_mask, uint8_t output) {
|
||||
template <typename T> uint8_t MOS6522<T>::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<HalfCycles>());
|
||||
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 <typename T> void MOS6522<T>::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<HalfCycles>());
|
||||
bus_handler_.set_port_output(Port::B, registers_.output[1], registers_.data_direction[1]);
|
||||
}
|
||||
|
@ -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.
|
||||
|
@ -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
|
||||
|
Loading…
x
Reference in New Issue
Block a user