// SPDX-FileCopyrightText: 2019-2022 Connor McLaughlin // SPDX-License-Identifier: (GPL-3.0 OR CC-BY-NC-ND-4.0) #include "timers.h" #include "gpu.h" #include "host.h" #include "interrupt_controller.h" #include "system.h" #include "util/imgui_manager.h" #include "util/state_wrapper.h" #include "common/bitfield.h" #include "common/log.h" #include "imgui.h" #include #include Log_SetChannel(Timers); namespace Timers { static constexpr u32 NUM_TIMERS = 3; enum class SyncMode : u8 { PauseOnGate = 0, ResetOnGate = 1, ResetAndRunOnGate = 2, FreeRunOnGate = 3 }; union CounterMode { u32 bits; BitField sync_enable; BitField sync_mode; BitField reset_at_target; BitField irq_at_target; BitField irq_on_overflow; BitField irq_repeat; BitField irq_pulse_n; BitField clock_source; BitField interrupt_request_n; BitField reached_target; BitField 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; }; static void UpdateCountingEnabled(CounterState& cs); static void CheckForIRQ(u32 index, u32 old_counter); static void UpdateIRQ(u32 index); static void AddSysClkTicks(void*, TickCount sysclk_ticks, TickCount ticks_late); static TickCount GetTicksUntilNextInterrupt(); static void UpdateSysClkEvent(); static std::unique_ptr s_sysclk_event; static std::array 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 void Timers::Initialize() { s_sysclk_event = TimingEvents::CreateTimingEvent("Timer SysClk Interrupt", 1, 1, &Timers::AddSysClkTicks, nullptr, false); Reset(); } void Timers::Shutdown() { s_sysclk_event.reset(); } void Timers::Reset() { for (CounterState& cs : s_states) { cs.mode.bits = 0; cs.mode.interrupt_request_n = true; cs.counter = 0; cs.target = 0; cs.gate = false; cs.external_counting_enabled = false; cs.counting_enabled = true; cs.irq_done = false; } s_syclk_ticks_carry = 0; s_sysclk_div_8_carry = 0; UpdateSysClkEvent(); } bool Timers::DoState(StateWrapper& sw) { for (CounterState& cs : s_states) { sw.Do(&cs.mode.bits); sw.Do(&cs.counter); sw.Do(&cs.target); sw.Do(&cs.gate); sw.Do(&cs.use_external_clock); sw.Do(&cs.external_counting_enabled); sw.Do(&cs.counting_enabled); sw.Do(&cs.irq_done); } sw.Do(&s_syclk_ticks_carry); sw.Do(&s_sysclk_div_8_carry); if (sw.IsReading()) UpdateSysClkEvent(); return !sw.HasError(); } void Timers::CPUClocksChanged() { 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); } void Timers::SetGate(u32 timer, bool state) { CounterState& cs = s_states[timer]; if (cs.gate == state) return; cs.gate = state; if (!cs.mode.sync_enable) return; if (cs.counting_enabled && !cs.use_external_clock) s_sysclk_event->InvokeEarly(); if (state) { switch (cs.mode.sync_mode) { case SyncMode::ResetOnGate: case SyncMode::ResetAndRunOnGate: cs.counter = 0; break; case SyncMode::FreeRunOnGate: cs.mode.sync_enable = false; break; default: break; } } UpdateCountingEnabled(cs); UpdateSysClkEvent(); } TickCount Timers::GetTicksUntilIRQ(u32 timer) { const CounterState& cs = s_states[timer]; if (!cs.counting_enabled) return std::numeric_limits::max(); TickCount ticks_until_irq = std::numeric_limits::max(); if (cs.mode.irq_at_target && cs.counter < cs.target) ticks_until_irq = static_cast(cs.target - cs.counter); if (cs.mode.irq_on_overflow) ticks_until_irq = std::min(ticks_until_irq, static_cast(0xFFFFu - cs.counter)); return ticks_until_irq; } void Timers::AddTicks(u32 timer, TickCount count) { CounterState& cs = s_states[timer]; const u32 old_counter = cs.counter; cs.counter += static_cast(count); CheckForIRQ(timer, old_counter); } void Timers::CheckForIRQ(u32 timer, u32 old_counter) { CounterState& cs = s_states[timer]; bool interrupt_request = false; if (cs.counter >= cs.target && (old_counter < cs.target || cs.target == 0)) { interrupt_request |= cs.mode.irq_at_target; cs.mode.reached_target = true; if (cs.mode.reset_at_target && cs.target > 0) cs.counter %= cs.target; } if (cs.counter >= 0xFFFF) { interrupt_request |= cs.mode.irq_on_overflow; cs.mode.reached_overflow = true; cs.counter %= 0xFFFFu; } if (interrupt_request) { if (!cs.mode.irq_pulse_n) { // this is actually low for a few cycles cs.mode.interrupt_request_n = false; UpdateIRQ(timer); cs.mode.interrupt_request_n = true; } else { cs.mode.interrupt_request_n ^= true; UpdateIRQ(timer); } } } void Timers::AddSysClkTicks(void*, TickCount sysclk_ticks, TickCount ticks_late) { sysclk_ticks = System::UnscaleTicksToOverclock(sysclk_ticks, &s_syclk_ticks_carry); if (!s_states[0].external_counting_enabled && s_states[0].counting_enabled) AddTicks(0, sysclk_ticks); if (!s_states[1].external_counting_enabled && s_states[1].counting_enabled) AddTicks(1, sysclk_ticks); if (s_states[2].external_counting_enabled) { 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; AddTicks(2, sysclk_div_8_ticks); } else if (s_states[2].counting_enabled) { AddTicks(2, sysclk_ticks); } UpdateSysClkEvent(); } 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); } CounterState& cs = s_states[timer_index]; switch (port_offset) { case 0x00: { if (timer_index < 2 && cs.external_counting_enabled) { // timers 0/1 depend on the GPU if (timer_index == 0 || g_gpu->IsCRTCScanlinePending()) g_gpu->SynchronizeCRTC(); } s_sysclk_event->InvokeEarly(); return cs.counter; } case 0x04: { if (timer_index < 2 && cs.external_counting_enabled) { // timers 0/1 depend on the GPU if (timer_index == 0 || g_gpu->IsCRTCScanlinePending()) g_gpu->SynchronizeCRTC(); } s_sysclk_event->InvokeEarly(); const u32 bits = cs.mode.bits; cs.mode.reached_overflow = false; cs.mode.reached_target = false; return bits; } case 0x08: return cs.target; default: Log_ErrorPrintf("Read unknown register in timer %u (offset 0x%02X)", timer_index, offset); 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; } CounterState& cs = s_states[timer_index]; if (timer_index < 2 && cs.external_counting_enabled) { // timers 0/1 depend on the GPU if (timer_index == 0 || g_gpu->IsCRTCScanlinePending()) g_gpu->SynchronizeCRTC(); } s_sysclk_event->InvokeEarly(); // Strictly speaking these IRQ checks should probably happen on the next tick. switch (port_offset) { case 0x00: { const u32 old_counter = cs.counter; 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; case 0x04: { static constexpr u32 WRITE_MASK = 0b1110001111111111; Log_DebugPrintf("Timer %u write mode register 0x%04X", timer_index, value); cs.mode.bits = (value & WRITE_MASK) | (cs.mode.bits & ~WRITE_MASK); cs.use_external_clock = (cs.mode.clock_source & (timer_index == 2 ? 2 : 1)) != 0; cs.counter = 0; cs.irq_done = false; UpdateCountingEnabled(cs); CheckForIRQ(timer_index, cs.counter); UpdateIRQ(timer_index); UpdateSysClkEvent(); } break; case 0x08: { 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; default: Log_ErrorPrintf("Write unknown register in timer %u (offset 0x%02X, value 0x%X)", timer_index, offset, value); 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: cs.counting_enabled = cs.gate; break; } } else { cs.counting_enabled = true; } cs.external_counting_enabled = cs.use_external_clock && cs.counting_enabled; } void Timers::UpdateIRQ(u32 index) { CounterState& cs = s_states[index]; if (cs.mode.interrupt_request_n || (!cs.mode.irq_repeat && cs.irq_done)) return; Log_DebugPrintf("Raising timer %u IRQ", index); cs.irq_done = true; InterruptController::InterruptRequest( static_cast(static_cast(InterruptController::IRQ::TMR0) + index)); } TickCount Timers::GetTicksUntilNextInterrupt() { TickCount min_ticks = System::GetMaxSliceTicks(); for (u32 i = 0; i < NUM_TIMERS; i++) { 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))) { continue; } if (cs.mode.irq_at_target) { TickCount ticks = (cs.counter <= cs.target) ? static_cast(cs.target - cs.counter) : static_cast((0xFFFFu - cs.counter) + cs.target); if (cs.external_counting_enabled) // sysclk/8 for timer 2 ticks *= 8; min_ticks = std::min(min_ticks, ticks); } if (cs.mode.irq_on_overflow) { TickCount ticks = static_cast(0xFFFFu - cs.counter); if (cs.external_counting_enabled) // sysclk/8 for timer 2 ticks *= 8; min_ticks = std::min(min_ticks, ticks); } } return System::ScaleTicksToOverclock(std::max(1, min_ticks)); } void Timers::UpdateSysClkEvent() { s_sysclk_event->Schedule(GetTicksUntilNextInterrupt()); } void Timers::DrawDebugStateWindow() { static constexpr u32 NUM_COLUMNS = 10; static constexpr std::array column_names = { {"#", "Value", "Target", "Sync", "Reset", "IRQ", "IRQRepeat", "IRQToggle", "Clock Source", "Reached"}}; static constexpr std::array sync_mode_names = { {"PauseOnGate", "ResetOnGate", "ResetAndRunOnGate", "FreeRunOnGate"}}; static constexpr std::array, 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)) { 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); for (const char* title : column_names) { ImGui::TextUnformatted(title); ImGui::NextColumn(); } for (u32 i = 0; i < NUM_TIMERS; i++) { const CounterState& cs = s_states[i]; 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(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(); ImGui::Text("%s%s", cs.mode.reached_target ? "Target " : "", cs.mode.reached_overflow ? "Overflow" : ""); ImGui::NextColumn(); ImGui::PopStyleColor(); } ImGui::Columns(1); ImGui::End(); }