Fixed obscure timer behavior, fixed regression in rapid_toggle.gb.
This commit is contained in:
parent
9bde98dede
commit
72d26c7046
@ -333,6 +333,9 @@ struct GB_gameboy_internal_s {
|
|||||||
uint16_t serial_cycles; /* This field changed its meaning in v0.10 */
|
uint16_t serial_cycles; /* This field changed its meaning in v0.10 */
|
||||||
uint16_t serial_length;
|
uint16_t serial_length;
|
||||||
uint8_t delayed_interrupts; /* When an interrupt occurs while not aligned to a T-cycle, it must be "delayed" */
|
uint8_t delayed_interrupts; /* When an interrupt occurs while not aligned to a T-cycle, it must be "delayed" */
|
||||||
|
bool dont_delay_timer_interrupt; /* If the timer glitch causes a TIMA overflow, it causes the timer to overflow
|
||||||
|
with different timing, so the triggered interrupt is not delayed.
|
||||||
|
Todo: needs test ROM. */
|
||||||
);
|
);
|
||||||
|
|
||||||
/* APU */
|
/* APU */
|
||||||
|
@ -88,7 +88,9 @@ static void advance_tima_state_machine(GB_gameboy_t *gb)
|
|||||||
}
|
}
|
||||||
else if (gb->tima_reload_state == GB_TIMA_RELOADING) {
|
else if (gb->tima_reload_state == GB_TIMA_RELOADING) {
|
||||||
gb->io_registers[GB_IO_IF] |= 4;
|
gb->io_registers[GB_IO_IF] |= 4;
|
||||||
|
if (!gb->dont_delay_timer_interrupt) {
|
||||||
gb->delayed_interrupts |= 4; // Timer interrupt is not aligned to a T-cycle and therefore is effective only the next one.
|
gb->delayed_interrupts |= 4; // Timer interrupt is not aligned to a T-cycle and therefore is effective only the next one.
|
||||||
|
}
|
||||||
gb->tima_reload_state = GB_TIMA_RELOADED;
|
gb->tima_reload_state = GB_TIMA_RELOADED;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -156,6 +158,7 @@ static void increase_tima(GB_gameboy_t *gb)
|
|||||||
{
|
{
|
||||||
gb->io_registers[GB_IO_TIMA]++;
|
gb->io_registers[GB_IO_TIMA]++;
|
||||||
if (gb->io_registers[GB_IO_TIMA] == 0) {
|
if (gb->io_registers[GB_IO_TIMA] == 0) {
|
||||||
|
gb->dont_delay_timer_interrupt = false;
|
||||||
gb->io_registers[GB_IO_TIMA] = gb->io_registers[GB_IO_TMA];
|
gb->io_registers[GB_IO_TIMA] = gb->io_registers[GB_IO_TMA];
|
||||||
gb->tima_reload_state = GB_TIMA_RELOADING;
|
gb->tima_reload_state = GB_TIMA_RELOADING;
|
||||||
}
|
}
|
||||||
@ -195,6 +198,7 @@ void GB_emulate_timer_glitch(GB_gameboy_t *gb, uint8_t old_tac, uint8_t new_tac)
|
|||||||
/* And now either the timer must be disabled, or the new bit used for overflow testing be 0. */
|
/* And now either the timer must be disabled, or the new bit used for overflow testing be 0. */
|
||||||
if (!(new_tac & 4) || gb->div_cycles & (new_clocks >> 1)) {
|
if (!(new_tac & 4) || gb->div_cycles & (new_clocks >> 1)) {
|
||||||
increase_tima(gb);
|
increase_tima(gb);
|
||||||
|
gb->dont_delay_timer_interrupt = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user