1
0
mirror of https://github.com/TomHarte/CLK.git synced 2025-01-26 15:32:04 +00:00

Switched to clocking the 6522 by the half-cycle. Very trivial test now passes.

This commit is contained in:
Thomas Harte 2016-06-18 13:57:10 -04:00
parent 06fb2ff1c7
commit 394902f409
4 changed files with 54 additions and 24 deletions

View File

@ -144,30 +144,38 @@ template <class T> class MOS6522 {
{
}
void run_for_cycles(unsigned int number_of_cycles)
void run_for_half_cycles(unsigned int number_of_cycles)
{
_registers.timer[0] -= number_of_cycles;
_registers.timer[1] -= number_of_cycles;
if(!_registers.timer[1] && _timer_is_running[1])
while(number_of_cycles--)
{
_timer_is_running[1] = false;
_registers.interrupt_flags |= InterruptFlag::Timer2;
reevaluate_interrupts();
}
if(_is_phase2)
{
_registers.timer[0] --;
_registers.timer[1] --;
if(!_registers.timer[0] && _timer_is_running[0])
{
_registers.interrupt_flags |= InterruptFlag::Timer1;
reevaluate_interrupts();
if(!_registers.timer[1] && _timer_is_running[1])
{
_timer_is_running[1] = false;
_registers.interrupt_flags |= InterruptFlag::Timer2;
reevaluate_interrupts();
}
// TODO: reload shouldn't occur for a further 1.5 cycles
if(_registers.auxiliary_control&0x40)
_registers.timer[0] = _registers.timer_latch[0];
else
_timer_is_running[0] = false;
if(!_registers.timer[0] && _timer_is_running[0])
{
_registers.interrupt_flags |= InterruptFlag::Timer1;
reevaluate_interrupts();
// TODO: reload shouldn't occur for a further 1.5 cycles
if(_registers.auxiliary_control&0x40)
_registers.timer[0] = _registers.timer_latch[0];
else
_timer_is_running[0] = false;
}
// TODO: lots of other status effects
}
_is_phase2 ^= true;
}
// TODO: lots of other status effects
}
bool get_interrupt_line()
@ -178,13 +186,18 @@ template <class T> class MOS6522 {
MOS6522() :
_timer_is_running{false, false},
_last_posted_interrupt_status(false)
_last_posted_interrupt_status(false),
_is_phase2(false)
{}
private:
// Intended to be overwritten
// Intended to be overridden
uint8_t get_port_input(int port) { return 0xff; }
void set_port_output(int port, uint8_t value) {}
// void set_interrupt_status(bool status) {}
// Phase toggle
bool _is_phase2;
// Delegate and communications
bool _last_posted_interrupt_status;

View File

@ -79,8 +79,8 @@ unsigned int Machine::perform_bus_operation(CPU6502::BusOperation operation, uin
}
}
_userPortVIA.run_for_cycles(1);
_keyboardVIA.run_for_cycles(1);
_userPortVIA.run_for_half_cycles(2);
_keyboardVIA.run_for_half_cycles(2);
return 1;
}

View File

@ -10,4 +10,21 @@ import XCTest
import Foundation
class MOS6522Tests: XCTestCase {
private func with6522(action: (MOS6522Bridge) -> ()) {
let bridge = MOS6522Bridge()
action(bridge)
}
func testTimerCount() {
with6522 {
$0.setValue(10, forRegister: 4)
$0.setValue(0, forRegister: 5)
$0.runForHalfCycles(10)
XCTAssert($0.valueForRegister(4) == 5, "Low order byte of timer should be 5; was \($0.valueForRegister(4))")
XCTAssert($0.valueForRegister(5) == 0, "High order byte of timer should be 5; was \($0.valueForRegister(5))")
}
}
}

View File

@ -49,7 +49,7 @@ class VanillaVIA: public MOS::MOS6522<VanillaVIA> {
- (void)runForHalfCycles:(NSUInteger)numberOfHalfCycles
{
_via.run_for_cycles(numberOfHalfCycles);
_via.run_for_half_cycles(numberOfHalfCycles);
}
@end