mirror of
https://github.com/TomHarte/CLK.git
synced 2025-01-23 11:30:24 +00:00
Beefs up interrupt awareness.
This commit is contained in:
parent
d898a43dff
commit
c733a4dbf8
@ -63,6 +63,7 @@ template <typename PortHandlerT, Personality personality> class MOS6526:
|
||||
template <int port> void set_port_output();
|
||||
template <int port> uint8_t get_port_input();
|
||||
void update_interrupts();
|
||||
void posit_interrupt(uint8_t mask);
|
||||
};
|
||||
|
||||
}
|
||||
|
@ -27,8 +27,20 @@ template <int port> uint8_t MOS6526<BusHandlerT, personality>::get_port_input()
|
||||
return (input & ~data_direction_[port]) | (output_[port] & data_direction_[port]);
|
||||
}
|
||||
|
||||
template <typename BusHandlerT, Personality personality>
|
||||
void MOS6526<BusHandlerT, personality>::posit_interrupt(uint8_t mask) {
|
||||
interrupt_state_ |= mask;
|
||||
update_interrupts();
|
||||
}
|
||||
|
||||
template <typename BusHandlerT, Personality personality>
|
||||
void MOS6526<BusHandlerT, personality>::update_interrupts() {
|
||||
if(interrupt_state_ & interrupt_control_) {
|
||||
interrupt_state_ |= 0x80;
|
||||
|
||||
printf("6526 should signal interrupt\n");
|
||||
assert(false);
|
||||
}
|
||||
}
|
||||
|
||||
template <typename BusHandlerT, Personality personality>
|
||||
@ -67,8 +79,12 @@ void MOS6526<BusHandlerT, personality>::write(int address, uint8_t value) {
|
||||
// upon a write to the LSB.
|
||||
case 8:
|
||||
if constexpr (personality == Personality::P8250) {
|
||||
tod_ = (tod_ & 0xffff00) | uint32_t(value);
|
||||
tod_increment_mask_ = uint32_t(~0);
|
||||
if(control_[1] & 0x80) {
|
||||
tod_alarm_ = (tod_alarm_ & 0xffff00) | uint32_t(value);
|
||||
} else {
|
||||
tod_ = (tod_ & 0xffff00) | uint32_t(value);
|
||||
tod_increment_mask_ = uint32_t(~0);
|
||||
}
|
||||
} else {
|
||||
printf("6526 TOD clock not implemented\n");
|
||||
assert(false);
|
||||
@ -76,8 +92,12 @@ void MOS6526<BusHandlerT, personality>::write(int address, uint8_t value) {
|
||||
break;
|
||||
case 9:
|
||||
if constexpr (personality == Personality::P8250) {
|
||||
tod_ = (tod_ & 0xff00ff) | uint32_t(value << 8);
|
||||
tod_increment_mask_ = 0;
|
||||
if(control_[1] & 0x80) {
|
||||
tod_alarm_ = (tod_alarm_ & 0xff00ff) | uint32_t(value << 8);
|
||||
} else {
|
||||
tod_ = (tod_ & 0xff00ff) | uint32_t(value << 8);
|
||||
tod_increment_mask_ = 0;
|
||||
}
|
||||
} else {
|
||||
printf("6526 TOD clock not implemented\n");
|
||||
assert(false);
|
||||
@ -85,8 +105,12 @@ void MOS6526<BusHandlerT, personality>::write(int address, uint8_t value) {
|
||||
break;
|
||||
case 10:
|
||||
if constexpr (personality == Personality::P8250) {
|
||||
tod_ = (tod_ & 0x00ffff) | uint32_t(value << 16);
|
||||
tod_increment_mask_ = 0;
|
||||
if(control_[1] & 0x80) {
|
||||
tod_alarm_ = (tod_alarm_ & 0x00ffff) | uint32_t(value << 16);
|
||||
} else {
|
||||
tod_ = (tod_ & 0x00ffff) | uint32_t(value << 16);
|
||||
tod_increment_mask_ = 0;
|
||||
}
|
||||
} else {
|
||||
printf("6526 TOD clock not implemented\n");
|
||||
assert(false);
|
||||
@ -101,10 +125,14 @@ void MOS6526<BusHandlerT, personality>::write(int address, uint8_t value) {
|
||||
break;
|
||||
|
||||
// Interrupt control.
|
||||
case 13:
|
||||
interrupt_control_ = value;
|
||||
case 13: {
|
||||
if(interrupt_control_ & 0x80) {
|
||||
interrupt_control_ |= value & 0x7f;
|
||||
} else {
|
||||
interrupt_control_ &= ~(value & 0x7f);
|
||||
}
|
||||
update_interrupts();
|
||||
break;
|
||||
} break;
|
||||
|
||||
// Control.
|
||||
case 14:
|
||||
@ -126,7 +154,6 @@ uint8_t MOS6526<BusHandlerT, personality>::read(int address) {
|
||||
switch(address) {
|
||||
case 0: return get_port_input<0>();
|
||||
case 1: return get_port_input<1>();
|
||||
case 13: return interrupt_control_;
|
||||
|
||||
case 2: case 3:
|
||||
return data_direction_[address - 2];
|
||||
@ -137,6 +164,14 @@ uint8_t MOS6526<BusHandlerT, personality>::read(int address) {
|
||||
case 6: return uint8_t(counters_[1].value >> 0);
|
||||
case 7: return uint8_t(counters_[1].value >> 0);
|
||||
|
||||
// Interrupt state.
|
||||
case 13: {
|
||||
const uint8_t result = interrupt_state_;
|
||||
interrupt_state_ = 0;
|
||||
update_interrupts();
|
||||
return result;
|
||||
} break;
|
||||
|
||||
case 14: case 15:
|
||||
return control_[address - 14];
|
||||
|
||||
@ -211,14 +246,17 @@ void MOS6526<BusHandlerT, personality>::advance_tod(int count) {
|
||||
if constexpr(personality == Personality::P8250) {
|
||||
// The 8250 uses a simple binary counter to replace the
|
||||
// 6526's time-of-day clock. So this is easy.
|
||||
const uint32_t distance_to_alarm_ = (tod_alarm_ - tod_) & 0xffffff;
|
||||
tod_ += uint32_t(count) & tod_increment_mask_;
|
||||
if(distance_to_alarm_ <= uint32_t(count)) {
|
||||
posit_interrupt(0x04);
|
||||
}
|
||||
} else {
|
||||
// The 6526 uses a time-of-day clock. This may or may not
|
||||
// be accurate.
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -17,14 +17,15 @@ struct MOS6526Storage {
|
||||
|
||||
uint8_t output_[2] = {0, 0};
|
||||
uint8_t data_direction_[2] = {0, 0};
|
||||
|
||||
uint8_t interrupt_control_ = 0;
|
||||
uint8_t interrupt_state_ = 0;
|
||||
|
||||
uint8_t control_[2] = {0, 0};
|
||||
|
||||
uint32_t tod_increment_mask_ = uint32_t(~0);
|
||||
uint32_t tod_latch_ = 0;
|
||||
uint32_t tod_ = 0;
|
||||
|
||||
bool write_tod_alarm_ = false;
|
||||
uint32_t tod_alarm_ = 0;
|
||||
|
||||
struct Counter {
|
||||
|
Loading…
x
Reference in New Issue
Block a user