Duckstation/src/core/timers.cpp

548 lines
15 KiB
C++
Raw Normal View History

// SPDX-FileCopyrightText: 2019-2022 Connor McLaughlin <stenzek@gmail.com>
// SPDX-License-Identifier: (GPL-3.0 OR CC-BY-NC-ND-4.0)
2019-09-20 13:40:19 +00:00
#include "timers.h"
#include "gpu.h"
#include "host.h"
2019-09-20 13:40:19 +00:00
#include "interrupt_controller.h"
#include "system.h"
2023-08-27 06:00:06 +00:00
#include "util/imgui_manager.h"
#include "util/state_wrapper.h"
2023-08-27 06:00:06 +00:00
#include "common/bitfield.h"
#include "common/log.h"
#include "imgui.h"
2023-01-11 08:58:25 +00:00
#include <array>
#include <memory>
2023-08-27 06:00:06 +00:00
2019-09-20 13:40:19 +00:00
Log_SetChannel(Timers);
2023-01-11 08:58:25 +00:00
namespace Timers {
namespace {
2023-01-11 08:58:25 +00:00
static constexpr u32 NUM_TIMERS = 3;
JIT optimizations and refactoring (#675) * CPU/Recompiler: Use rel32 call where possible for no-args * JitCodeBuffer: Support using preallocated buffer * CPU/Recompiler/AArch64: Use bl instead of blr for short branches * CPU/CodeCache: Allocate recompiler buffer in program space This means we don't need 64-bit moves for every call out of the recompiler. * GTE: Don't store as u16 and load as u32 * CPU/Recompiler: Add methods to emit global load/stores * GTE: Convert class to namespace * CPU/Recompiler: Call GTE functions directly * Settings: Turn into a global variable * GPU: Replace local pointers with global * InterruptController: Turn into a global pointer * System: Replace local pointers with global * Timers: Turn into a global instance * DMA: Turn into a global instance * SPU: Turn into a global instance * CDROM: Turn into a global instance * MDEC: Turn into a global instance * Pad: Turn into a global instance * SIO: Turn into a global instance * CDROM: Move audio FIFO to the heap * CPU/Recompiler: Drop ASMFunctions No longer needed since we have code in the same 4GB window. * CPUCodeCache: Turn class into namespace * Bus: Local pointer -> global pointers * CPU: Turn class into namespace * Bus: Turn into namespace * GTE: Store registers in CPU state struct Allows relative addressing on ARM. * CPU/Recompiler: Align code storage to page size * CPU/Recompiler: Fix relative branches on A64 * HostInterface: Local references to global * System: Turn into a namespace, move events out * Add guard pages * Android: Fix build
2020-07-31 07:09:18 +00:00
2023-01-11 08:58:25 +00:00
enum class SyncMode : u8
{
PauseOnGate = 0,
ResetOnGate = 1,
ResetAndRunOnGate = 2,
FreeRunOnGate = 3
};
union CounterMode
{
u32 bits;
BitField<u32, bool, 0, 1> sync_enable;
BitField<u32, SyncMode, 1, 2> sync_mode;
BitField<u32, bool, 3, 1> reset_at_target;
BitField<u32, bool, 4, 1> irq_at_target;
BitField<u32, bool, 5, 1> irq_on_overflow;
BitField<u32, bool, 6, 1> irq_repeat;
BitField<u32, bool, 7, 1> irq_pulse_n;
BitField<u32, u8, 8, 2> clock_source;
BitField<u32, bool, 10, 1> interrupt_request_n;
BitField<u32, bool, 11, 1> reached_target;
BitField<u32, bool, 12, 1> reached_overflow;
};
struct CounterState
{
CounterMode mode;
u32 counter;
u32 target;
bool gate;
bool use_external_clock;
bool external_counting_enabled;
bool counting_enabled;
bool irq_done;
};
} // namespace
2023-01-11 08:58:25 +00:00
static void UpdateCountingEnabled(CounterState& cs);
static void CheckForIRQ(u32 index, u32 old_counter);
2019-09-20 13:40:19 +00:00
2023-01-11 08:58:25 +00:00
static void AddSysClkTicks(void*, TickCount sysclk_ticks, TickCount ticks_late);
static TickCount GetTicksUntilNextInterrupt();
static void UpdateSysClkEvent();
static std::unique_ptr<TimingEvent> s_sysclk_event;
static std::array<CounterState, NUM_TIMERS> s_states{};
static TickCount s_syclk_ticks_carry = 0; // 0 unless overclocking is enabled
static u32 s_sysclk_div_8_carry = 0; // partial ticks for timer 3 with sysclk/8
}; // namespace Timers
2019-09-20 13:40:19 +00:00
JIT optimizations and refactoring (#675) * CPU/Recompiler: Use rel32 call where possible for no-args * JitCodeBuffer: Support using preallocated buffer * CPU/Recompiler/AArch64: Use bl instead of blr for short branches * CPU/CodeCache: Allocate recompiler buffer in program space This means we don't need 64-bit moves for every call out of the recompiler. * GTE: Don't store as u16 and load as u32 * CPU/Recompiler: Add methods to emit global load/stores * GTE: Convert class to namespace * CPU/Recompiler: Call GTE functions directly * Settings: Turn into a global variable * GPU: Replace local pointers with global * InterruptController: Turn into a global pointer * System: Replace local pointers with global * Timers: Turn into a global instance * DMA: Turn into a global instance * SPU: Turn into a global instance * CDROM: Turn into a global instance * MDEC: Turn into a global instance * Pad: Turn into a global instance * SIO: Turn into a global instance * CDROM: Move audio FIFO to the heap * CPU/Recompiler: Drop ASMFunctions No longer needed since we have code in the same 4GB window. * CPUCodeCache: Turn class into namespace * Bus: Local pointer -> global pointers * CPU: Turn class into namespace * Bus: Turn into namespace * GTE: Store registers in CPU state struct Allows relative addressing on ARM. * CPU/Recompiler: Align code storage to page size * CPU/Recompiler: Fix relative branches on A64 * HostInterface: Local references to global * System: Turn into a namespace, move events out * Add guard pages * Android: Fix build
2020-07-31 07:09:18 +00:00
void Timers::Initialize()
{
2023-01-11 08:58:25 +00:00
s_sysclk_event =
TimingEvents::CreateTimingEvent("Timer SysClk Interrupt", 1, 1, &Timers::AddSysClkTicks, nullptr, false);
JIT optimizations and refactoring (#675) * CPU/Recompiler: Use rel32 call where possible for no-args * JitCodeBuffer: Support using preallocated buffer * CPU/Recompiler/AArch64: Use bl instead of blr for short branches * CPU/CodeCache: Allocate recompiler buffer in program space This means we don't need 64-bit moves for every call out of the recompiler. * GTE: Don't store as u16 and load as u32 * CPU/Recompiler: Add methods to emit global load/stores * GTE: Convert class to namespace * CPU/Recompiler: Call GTE functions directly * Settings: Turn into a global variable * GPU: Replace local pointers with global * InterruptController: Turn into a global pointer * System: Replace local pointers with global * Timers: Turn into a global instance * DMA: Turn into a global instance * SPU: Turn into a global instance * CDROM: Turn into a global instance * MDEC: Turn into a global instance * Pad: Turn into a global instance * SIO: Turn into a global instance * CDROM: Move audio FIFO to the heap * CPU/Recompiler: Drop ASMFunctions No longer needed since we have code in the same 4GB window. * CPUCodeCache: Turn class into namespace * Bus: Local pointer -> global pointers * CPU: Turn class into namespace * Bus: Turn into namespace * GTE: Store registers in CPU state struct Allows relative addressing on ARM. * CPU/Recompiler: Align code storage to page size * CPU/Recompiler: Fix relative branches on A64 * HostInterface: Local references to global * System: Turn into a namespace, move events out * Add guard pages * Android: Fix build
2020-07-31 07:09:18 +00:00
Reset();
}
void Timers::Shutdown()
2019-09-20 13:40:19 +00:00
{
2023-01-11 08:58:25 +00:00
s_sysclk_event.reset();
2019-09-20 13:40:19 +00:00
}
void Timers::Reset()
{
2023-01-11 08:58:25 +00:00
for (CounterState& cs : s_states)
2019-09-20 13:40:19 +00:00
{
cs.mode.bits = 0;
cs.mode.interrupt_request_n = true;
2019-09-20 13:40:19 +00:00
cs.counter = 0;
cs.target = 0;
cs.gate = false;
cs.external_counting_enabled = false;
cs.counting_enabled = true;
2019-10-08 08:21:15 +00:00
cs.irq_done = false;
2019-09-20 13:40:19 +00:00
}
2019-10-08 08:21:15 +00:00
2023-01-11 08:58:25 +00:00
s_syclk_ticks_carry = 0;
s_sysclk_div_8_carry = 0;
UpdateSysClkEvent();
2019-09-20 13:40:19 +00:00
}
bool Timers::DoState(StateWrapper& sw)
{
2023-01-11 08:58:25 +00:00
for (CounterState& cs : s_states)
2019-09-20 13:40:19 +00:00
{
sw.Do(&cs.mode.bits);
sw.Do(&cs.counter);
sw.Do(&cs.target);
sw.Do(&cs.gate);
2019-09-26 14:03:28 +00:00
sw.Do(&cs.use_external_clock);
2019-09-20 13:40:19 +00:00
sw.Do(&cs.external_counting_enabled);
sw.Do(&cs.counting_enabled);
2019-10-08 08:21:15 +00:00
sw.Do(&cs.irq_done);
2019-09-20 13:40:19 +00:00
}
2023-01-11 08:58:25 +00:00
sw.Do(&s_syclk_ticks_carry);
sw.Do(&s_sysclk_div_8_carry);
if (sw.IsReading())
UpdateSysClkEvent();
2019-09-20 13:40:19 +00:00
return !sw.HasError();
}
void Timers::CPUClocksChanged()
{
2023-01-11 08:58:25 +00:00
s_syclk_ticks_carry = 0;
}
bool Timers::IsUsingExternalClock(u32 timer)
{
return s_states[timer].external_counting_enabled;
}
bool Timers::IsSyncEnabled(u32 timer)
{
return s_states[timer].mode.sync_enable;
}
bool Timers::IsExternalIRQEnabled(u32 timer)
{
const CounterState& cs = s_states[timer];
return (cs.external_counting_enabled && (cs.mode.bits & ((1u << 4) | (1u << 5))) != 0);
}
2019-09-20 13:40:19 +00:00
void Timers::SetGate(u32 timer, bool state)
{
2023-01-11 08:58:25 +00:00
CounterState& cs = s_states[timer];
2019-09-20 13:40:19 +00:00
if (cs.gate == state)
return;
cs.gate = state;
if (!cs.mode.sync_enable)
return;
if (cs.counting_enabled && !cs.use_external_clock)
2023-01-11 08:58:25 +00:00
s_sysclk_event->InvokeEarly();
if (state)
2019-09-20 13:40:19 +00:00
{
switch (cs.mode.sync_mode)
2019-09-20 13:40:19 +00:00
{
case SyncMode::ResetOnGate:
case SyncMode::ResetAndRunOnGate:
cs.counter = 0;
break;
case SyncMode::FreeRunOnGate:
cs.mode.sync_enable = false;
break;
2023-09-03 04:30:26 +00:00
default:
break;
2019-09-20 13:40:19 +00:00
}
}
UpdateCountingEnabled(cs);
UpdateSysClkEvent();
2019-09-20 13:40:19 +00:00
}
2023-01-11 08:58:25 +00:00
TickCount Timers::GetTicksUntilIRQ(u32 timer)
{
2023-01-11 08:58:25 +00:00
const CounterState& cs = s_states[timer];
if (!cs.counting_enabled)
return std::numeric_limits<TickCount>::max();
TickCount ticks_until_irq = std::numeric_limits<TickCount>::max();
if (cs.mode.irq_at_target && cs.counter < cs.target)
ticks_until_irq = static_cast<TickCount>(cs.target - cs.counter);
if (cs.mode.irq_on_overflow)
ticks_until_irq = std::min(ticks_until_irq, static_cast<TickCount>(0xFFFFu - cs.counter));
return ticks_until_irq;
}
2019-10-08 08:21:15 +00:00
void Timers::AddTicks(u32 timer, TickCount count)
2019-09-20 13:40:19 +00:00
{
2023-01-11 08:58:25 +00:00
CounterState& cs = s_states[timer];
2019-10-08 08:21:15 +00:00
const u32 old_counter = cs.counter;
cs.counter += static_cast<u32>(count);
CheckForIRQ(timer, old_counter);
}
void Timers::CheckForIRQ(u32 timer, u32 old_counter)
{
2023-01-11 08:58:25 +00:00
CounterState& cs = s_states[timer];
2019-09-20 13:40:19 +00:00
2019-10-08 08:21:15 +00:00
bool interrupt_request = false;
if (cs.counter >= cs.target && (old_counter < cs.target || cs.target == 0))
2019-10-08 08:21:15 +00:00
{
interrupt_request |= cs.mode.irq_at_target;
2019-09-20 13:40:19 +00:00
cs.mode.reached_target = true;
if (cs.mode.reset_at_target && cs.target > 0)
cs.counter %= cs.target;
2019-10-08 08:21:15 +00:00
}
if (cs.counter >= 0xFFFF)
{
interrupt_request |= cs.mode.irq_on_overflow;
2019-09-20 13:40:19 +00:00
cs.mode.reached_overflow = true;
cs.counter %= 0xFFFFu;
2019-10-08 08:21:15 +00:00
}
2019-09-20 13:40:19 +00:00
2019-10-08 08:21:15 +00:00
if (interrupt_request)
2019-09-20 13:40:19 +00:00
{
const InterruptController::IRQ irqnum =
static_cast<InterruptController::IRQ>(static_cast<u32>(InterruptController::IRQ::TMR0) + timer);
2019-10-08 08:21:15 +00:00
if (!cs.mode.irq_pulse_n)
{
if (!cs.irq_done || cs.mode.irq_repeat)
{
// this is actually low for a few cycles
Log_DebugPrintf("Raising timer %u pulse IRQ", timer);
InterruptController::SetLineState(irqnum, false);
InterruptController::SetLineState(irqnum, true);
}
cs.irq_done = true;
2019-10-08 08:21:15 +00:00
cs.mode.interrupt_request_n = true;
}
else
{
// TODO: How does the non-repeat mode work here?
2019-10-08 08:21:15 +00:00
cs.mode.interrupt_request_n ^= true;
if (!cs.mode.interrupt_request_n)
Log_DebugPrintf("Raising timer %u alternate IRQ", timer);
InterruptController::SetLineState(irqnum, !cs.mode.interrupt_request_n);
2019-10-08 08:21:15 +00:00
}
2019-09-20 13:40:19 +00:00
}
}
2023-01-11 08:58:25 +00:00
void Timers::AddSysClkTicks(void*, TickCount sysclk_ticks, TickCount ticks_late)
{
2023-01-11 08:58:25 +00:00
sysclk_ticks = System::UnscaleTicksToOverclock(sysclk_ticks, &s_syclk_ticks_carry);
2023-01-11 08:58:25 +00:00
if (!s_states[0].external_counting_enabled && s_states[0].counting_enabled)
2019-10-08 08:21:15 +00:00
AddTicks(0, sysclk_ticks);
2023-01-11 08:58:25 +00:00
if (!s_states[1].external_counting_enabled && s_states[1].counting_enabled)
2019-10-08 08:21:15 +00:00
AddTicks(1, sysclk_ticks);
2023-01-11 08:58:25 +00:00
if (s_states[2].external_counting_enabled)
2019-10-08 08:21:15 +00:00
{
2023-01-11 08:58:25 +00:00
TickCount sysclk_div_8_ticks = (sysclk_ticks + s_sysclk_div_8_carry) / 8;
s_sysclk_div_8_carry = (sysclk_ticks + s_sysclk_div_8_carry) % 8;
2019-10-08 08:21:15 +00:00
AddTicks(2, sysclk_div_8_ticks);
}
2023-01-11 08:58:25 +00:00
else if (s_states[2].counting_enabled)
2019-10-08 08:21:15 +00:00
{
2019-10-12 12:15:38 +00:00
AddTicks(2, sysclk_ticks);
2019-10-08 08:21:15 +00:00
}
UpdateSysClkEvent();
}
2019-09-20 13:40:19 +00:00
u32 Timers::ReadRegister(u32 offset)
{
const u32 timer_index = (offset >> 4) & u32(0x03);
const u32 port_offset = offset & u32(0x0F);
if (timer_index >= 3)
{
Log_ErrorPrintf("Timer read out of range: offset 0x%02X", offset);
return UINT32_C(0xFFFFFFFF);
}
2019-09-20 13:40:19 +00:00
2023-01-11 08:58:25 +00:00
CounterState& cs = s_states[timer_index];
2019-09-20 13:40:19 +00:00
switch (port_offset)
{
case 0x00:
{
if (timer_index < 2 && cs.external_counting_enabled)
{
// timers 0/1 depend on the GPU
JIT optimizations and refactoring (#675) * CPU/Recompiler: Use rel32 call where possible for no-args * JitCodeBuffer: Support using preallocated buffer * CPU/Recompiler/AArch64: Use bl instead of blr for short branches * CPU/CodeCache: Allocate recompiler buffer in program space This means we don't need 64-bit moves for every call out of the recompiler. * GTE: Don't store as u16 and load as u32 * CPU/Recompiler: Add methods to emit global load/stores * GTE: Convert class to namespace * CPU/Recompiler: Call GTE functions directly * Settings: Turn into a global variable * GPU: Replace local pointers with global * InterruptController: Turn into a global pointer * System: Replace local pointers with global * Timers: Turn into a global instance * DMA: Turn into a global instance * SPU: Turn into a global instance * CDROM: Turn into a global instance * MDEC: Turn into a global instance * Pad: Turn into a global instance * SIO: Turn into a global instance * CDROM: Move audio FIFO to the heap * CPU/Recompiler: Drop ASMFunctions No longer needed since we have code in the same 4GB window. * CPUCodeCache: Turn class into namespace * Bus: Local pointer -> global pointers * CPU: Turn class into namespace * Bus: Turn into namespace * GTE: Store registers in CPU state struct Allows relative addressing on ARM. * CPU/Recompiler: Align code storage to page size * CPU/Recompiler: Fix relative branches on A64 * HostInterface: Local references to global * System: Turn into a namespace, move events out * Add guard pages * Android: Fix build
2020-07-31 07:09:18 +00:00
if (timer_index == 0 || g_gpu->IsCRTCScanlinePending())
g_gpu->SynchronizeCRTC();
}
2023-01-11 08:58:25 +00:00
s_sysclk_event->InvokeEarly();
2019-09-20 13:40:19 +00:00
return cs.counter;
}
2019-09-20 13:40:19 +00:00
case 0x04:
{
if (timer_index < 2 && cs.external_counting_enabled)
{
// timers 0/1 depend on the GPU
JIT optimizations and refactoring (#675) * CPU/Recompiler: Use rel32 call where possible for no-args * JitCodeBuffer: Support using preallocated buffer * CPU/Recompiler/AArch64: Use bl instead of blr for short branches * CPU/CodeCache: Allocate recompiler buffer in program space This means we don't need 64-bit moves for every call out of the recompiler. * GTE: Don't store as u16 and load as u32 * CPU/Recompiler: Add methods to emit global load/stores * GTE: Convert class to namespace * CPU/Recompiler: Call GTE functions directly * Settings: Turn into a global variable * GPU: Replace local pointers with global * InterruptController: Turn into a global pointer * System: Replace local pointers with global * Timers: Turn into a global instance * DMA: Turn into a global instance * SPU: Turn into a global instance * CDROM: Turn into a global instance * MDEC: Turn into a global instance * Pad: Turn into a global instance * SIO: Turn into a global instance * CDROM: Move audio FIFO to the heap * CPU/Recompiler: Drop ASMFunctions No longer needed since we have code in the same 4GB window. * CPUCodeCache: Turn class into namespace * Bus: Local pointer -> global pointers * CPU: Turn class into namespace * Bus: Turn into namespace * GTE: Store registers in CPU state struct Allows relative addressing on ARM. * CPU/Recompiler: Align code storage to page size * CPU/Recompiler: Fix relative branches on A64 * HostInterface: Local references to global * System: Turn into a namespace, move events out * Add guard pages * Android: Fix build
2020-07-31 07:09:18 +00:00
if (timer_index == 0 || g_gpu->IsCRTCScanlinePending())
g_gpu->SynchronizeCRTC();
}
2023-01-11 08:58:25 +00:00
s_sysclk_event->InvokeEarly();
2019-09-20 13:40:19 +00:00
const u32 bits = cs.mode.bits;
cs.mode.reached_overflow = false;
cs.mode.reached_target = false;
return bits;
2019-09-20 13:40:19 +00:00
}
case 0x08:
return cs.target;
default:
2021-02-05 00:54:51 +00:00
Log_ErrorPrintf("Read unknown register in timer %u (offset 0x%02X)", timer_index, offset);
2019-09-20 13:40:19 +00:00
return UINT32_C(0xFFFFFFFF);
}
}
void Timers::WriteRegister(u32 offset, u32 value)
{
const u32 timer_index = (offset >> 4) & u32(0x03);
const u32 port_offset = offset & u32(0x0F);
if (timer_index >= 3)
{
Log_ErrorPrintf("Timer write out of range: offset 0x%02X value 0x%08X", offset, value);
return;
}
2019-09-20 13:40:19 +00:00
2023-01-11 08:58:25 +00:00
CounterState& cs = s_states[timer_index];
2019-09-20 13:40:19 +00:00
if (timer_index < 2 && cs.external_counting_enabled)
{
// timers 0/1 depend on the GPU
JIT optimizations and refactoring (#675) * CPU/Recompiler: Use rel32 call where possible for no-args * JitCodeBuffer: Support using preallocated buffer * CPU/Recompiler/AArch64: Use bl instead of blr for short branches * CPU/CodeCache: Allocate recompiler buffer in program space This means we don't need 64-bit moves for every call out of the recompiler. * GTE: Don't store as u16 and load as u32 * CPU/Recompiler: Add methods to emit global load/stores * GTE: Convert class to namespace * CPU/Recompiler: Call GTE functions directly * Settings: Turn into a global variable * GPU: Replace local pointers with global * InterruptController: Turn into a global pointer * System: Replace local pointers with global * Timers: Turn into a global instance * DMA: Turn into a global instance * SPU: Turn into a global instance * CDROM: Turn into a global instance * MDEC: Turn into a global instance * Pad: Turn into a global instance * SIO: Turn into a global instance * CDROM: Move audio FIFO to the heap * CPU/Recompiler: Drop ASMFunctions No longer needed since we have code in the same 4GB window. * CPUCodeCache: Turn class into namespace * Bus: Local pointer -> global pointers * CPU: Turn class into namespace * Bus: Turn into namespace * GTE: Store registers in CPU state struct Allows relative addressing on ARM. * CPU/Recompiler: Align code storage to page size * CPU/Recompiler: Fix relative branches on A64 * HostInterface: Local references to global * System: Turn into a namespace, move events out * Add guard pages * Android: Fix build
2020-07-31 07:09:18 +00:00
if (timer_index == 0 || g_gpu->IsCRTCScanlinePending())
g_gpu->SynchronizeCRTC();
}
2023-01-11 08:58:25 +00:00
s_sysclk_event->InvokeEarly();
2020-03-23 14:20:27 +00:00
// Strictly speaking these IRQ checks should probably happen on the next tick.
2019-09-20 13:40:19 +00:00
switch (port_offset)
{
case 0x00:
{
const u32 old_counter = cs.counter;
2019-09-20 13:40:19 +00:00
Log_DebugPrintf("Timer %u write counter %u", timer_index, value);
cs.counter = value & u32(0xFFFF);
CheckForIRQ(timer_index, old_counter);
if (timer_index == 2 || !cs.external_counting_enabled)
UpdateSysClkEvent();
}
break;
2019-09-20 13:40:19 +00:00
case 0x04:
{
static constexpr u32 WRITE_MASK = 0b1110001111111111;
2019-09-20 13:40:19 +00:00
Log_DebugPrintf("Timer %u write mode register 0x%04X", timer_index, value);
cs.mode.bits = (value & WRITE_MASK) | (cs.mode.bits & ~WRITE_MASK);
2019-09-20 13:40:19 +00:00
cs.use_external_clock = (cs.mode.clock_source & (timer_index == 2 ? 2 : 1)) != 0;
cs.counter = 0;
2019-10-08 08:21:15 +00:00
cs.irq_done = false;
InterruptController::SetLineState(
static_cast<InterruptController::IRQ>(static_cast<u32>(InterruptController::IRQ::TMR0) + timer_index), false);
2019-10-08 08:21:15 +00:00
2019-09-20 13:40:19 +00:00
UpdateCountingEnabled(cs);
CheckForIRQ(timer_index, cs.counter);
UpdateSysClkEvent();
2019-09-20 13:40:19 +00:00
}
break;
case 0x08:
{
2019-09-20 13:40:19 +00:00
Log_DebugPrintf("Timer %u write target 0x%04X", timer_index, ZeroExtend32(Truncate16(value)));
cs.target = value & u32(0xFFFF);
CheckForIRQ(timer_index, cs.counter);
if (timer_index == 2 || !cs.external_counting_enabled)
UpdateSysClkEvent();
}
break;
2019-09-20 13:40:19 +00:00
default:
2021-02-05 00:54:51 +00:00
Log_ErrorPrintf("Write unknown register in timer %u (offset 0x%02X, value 0x%X)", timer_index, offset, value);
2019-09-20 13:40:19 +00:00
break;
}
}
void Timers::UpdateCountingEnabled(CounterState& cs)
{
if (cs.mode.sync_enable)
{
switch (cs.mode.sync_mode)
{
case SyncMode::PauseOnGate:
cs.counting_enabled = !cs.gate;
break;
case SyncMode::ResetOnGate:
cs.counting_enabled = true;
break;
case SyncMode::ResetAndRunOnGate:
case SyncMode::FreeRunOnGate:
2019-09-20 13:40:19 +00:00
cs.counting_enabled = cs.gate;
break;
}
}
else
{
cs.counting_enabled = true;
}
cs.external_counting_enabled = cs.use_external_clock && cs.counting_enabled;
}
2023-01-11 08:58:25 +00:00
TickCount Timers::GetTicksUntilNextInterrupt()
2019-09-20 13:40:19 +00:00
{
TickCount min_ticks = System::GetMaxSliceTicks();
2019-10-08 08:21:15 +00:00
for (u32 i = 0; i < NUM_TIMERS; i++)
{
2023-01-11 08:58:25 +00:00
const CounterState& cs = s_states[i];
if (!cs.counting_enabled || (i < 2 && cs.external_counting_enabled) ||
(!cs.mode.irq_at_target && !cs.mode.irq_on_overflow && (cs.mode.irq_repeat || !cs.irq_done)))
{
2019-10-08 08:21:15 +00:00
continue;
}
2019-10-08 08:21:15 +00:00
if (cs.mode.irq_at_target)
{
TickCount ticks = (cs.counter <= cs.target) ? static_cast<TickCount>(cs.target - cs.counter) :
static_cast<TickCount>((0xFFFFu - cs.counter) + cs.target);
if (cs.external_counting_enabled) // sysclk/8 for timer 2
ticks *= 8;
2019-10-08 08:21:15 +00:00
min_ticks = std::min(min_ticks, ticks);
}
if (cs.mode.irq_on_overflow)
{
TickCount ticks = static_cast<TickCount>(0xFFFFu - cs.counter);
if (cs.external_counting_enabled) // sysclk/8 for timer 2
ticks *= 8;
2019-10-08 08:21:15 +00:00
min_ticks = std::min(min_ticks, ticks);
}
2019-10-08 08:21:15 +00:00
}
return System::ScaleTicksToOverclock(std::max<TickCount>(1, min_ticks));
}
void Timers::UpdateSysClkEvent()
{
2023-01-11 08:58:25 +00:00
s_sysclk_event->Schedule(GetTicksUntilNextInterrupt());
2019-09-20 13:40:19 +00:00
}
2019-10-12 12:15:38 +00:00
void Timers::DrawDebugStateWindow()
2019-10-12 12:15:38 +00:00
{
static constexpr u32 NUM_COLUMNS = 10;
static constexpr std::array<const char*, NUM_COLUMNS> column_names = {
{"#", "Value", "Target", "Sync", "Reset", "IRQ", "IRQRepeat", "IRQToggle", "Clock Source", "Reached"}};
static constexpr std::array<const char*, 4> sync_mode_names = {
{"PauseOnGate", "ResetOnGate", "ResetAndRunOnGate", "FreeRunOnGate"}};
static constexpr std::array<std::array<const char*, 4>, 3> clock_source_names = {
{{{"SysClk", "DotClk", "SysClk", "DotClk"}},
{{"SysClk", "HBlank", "SysClk", "HBlank"}},
{{"SysClk", "DotClk", "SysClk/8", "SysClk/8"}}}};
const float framebuffer_scale = Host::GetOSDScale();
ImGui::SetNextWindowSize(ImVec2(800.0f * framebuffer_scale, 100.0f * framebuffer_scale), ImGuiCond_FirstUseEver);
if (!ImGui::Begin("Timer State", nullptr))
2019-10-12 12:15:38 +00:00
{
ImGui::End();
return;
}
ImGui::Columns(NUM_COLUMNS);
ImGui::SetColumnWidth(0, 20.0f * framebuffer_scale);
ImGui::SetColumnWidth(1, 50.0f * framebuffer_scale);
ImGui::SetColumnWidth(2, 50.0f * framebuffer_scale);
ImGui::SetColumnWidth(3, 100.0f * framebuffer_scale);
ImGui::SetColumnWidth(4, 80.0f * framebuffer_scale);
ImGui::SetColumnWidth(5, 80.0f * framebuffer_scale);
ImGui::SetColumnWidth(6, 80.0f * framebuffer_scale);
ImGui::SetColumnWidth(7, 80.0f * framebuffer_scale);
ImGui::SetColumnWidth(8, 80.0f * framebuffer_scale);
ImGui::SetColumnWidth(9, 80.0f * framebuffer_scale);
2019-10-12 12:15:38 +00:00
for (const char* title : column_names)
{
ImGui::TextUnformatted(title);
ImGui::NextColumn();
}
for (u32 i = 0; i < NUM_TIMERS; i++)
{
2023-01-11 08:58:25 +00:00
const CounterState& cs = s_states[i];
2019-10-12 12:15:38 +00:00
ImGui::PushStyleColor(ImGuiCol_Text,
cs.counting_enabled ? ImVec4(1.0f, 1.0f, 1.0f, 1.0f) : ImVec4(0.5f, 0.5f, 0.5f, 1.0f));
ImGui::Text("%u", i);
ImGui::NextColumn();
ImGui::Text("%u", cs.counter);
ImGui::NextColumn();
ImGui::Text("%u", cs.target);
ImGui::NextColumn();
ImGui::Text("%s",
cs.mode.sync_enable ? sync_mode_names[static_cast<u8>(cs.mode.sync_mode.GetValue())] : "Disabled");
ImGui::NextColumn();
ImGui::Text("%s", cs.mode.reset_at_target ? "@Target" : "@Overflow");
ImGui::NextColumn();
ImGui::Text("%s%s", cs.mode.irq_at_target ? "Target " : "", cs.mode.irq_on_overflow ? "Overflow" : "");
ImGui::NextColumn();
ImGui::Text("%s", cs.mode.irq_repeat ? "Yes" : "No");
ImGui::NextColumn();
ImGui::Text("%s", cs.mode.irq_pulse_n ? "Yes" : "No");
ImGui::NextColumn();
ImGui::Text("%s%s", clock_source_names[i][cs.mode.clock_source], cs.external_counting_enabled ? " (External)" : "");
ImGui::NextColumn();
2019-10-22 13:07:51 +00:00
ImGui::Text("%s%s", cs.mode.reached_target ? "Target " : "", cs.mode.reached_overflow ? "Overflow" : "");
2019-10-12 12:15:38 +00:00
ImGui::NextColumn();
ImGui::PopStyleColor();
}
ImGui::Columns(1);
ImGui::End();
}