mirror of
https://github.com/RetroDECK/Duckstation.git
synced 2025-01-20 15:25:38 +00:00
CPU: Make interrupts actually edge-triggered
This commit is contained in:
parent
3702a533f2
commit
fa6850902a
|
@ -986,6 +986,7 @@ void CDROM::WriteRegister(u32 offset, u8 value)
|
||||||
s_interrupt_flag_register &= ~(value & INTERRUPT_REGISTER_MASK);
|
s_interrupt_flag_register &= ~(value & INTERRUPT_REGISTER_MASK);
|
||||||
if (s_interrupt_flag_register == 0)
|
if (s_interrupt_flag_register == 0)
|
||||||
{
|
{
|
||||||
|
InterruptController::SetLineState(InterruptController::IRQ::CDROM, false);
|
||||||
if (HasPendingAsyncInterrupt())
|
if (HasPendingAsyncInterrupt())
|
||||||
QueueDeliverAsyncInterrupt();
|
QueueDeliverAsyncInterrupt();
|
||||||
else
|
else
|
||||||
|
@ -1212,10 +1213,8 @@ void CDROM::UpdateStatusRegister()
|
||||||
|
|
||||||
void CDROM::UpdateInterruptRequest()
|
void CDROM::UpdateInterruptRequest()
|
||||||
{
|
{
|
||||||
if ((s_interrupt_flag_register & s_interrupt_enable_register) == 0)
|
InterruptController::SetLineState(InterruptController::IRQ::CDROM,
|
||||||
return;
|
(s_interrupt_flag_register & s_interrupt_enable_register) != 0);
|
||||||
|
|
||||||
InterruptController::InterruptRequest(InterruptController::IRQ::CDROM);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool CDROM::HasPendingDiscEvent()
|
bool CDROM::HasPendingDiscEvent()
|
||||||
|
|
|
@ -398,17 +398,16 @@ void CPU::RaiseBreakException(u32 CAUSE_bits, u32 EPC, u32 instruction_bits)
|
||||||
RaiseException(CAUSE_bits, EPC, GetExceptionVector());
|
RaiseException(CAUSE_bits, EPC, GetExceptionVector());
|
||||||
}
|
}
|
||||||
|
|
||||||
void CPU::SetExternalInterrupt(u8 bit)
|
void CPU::SetIRQRequest(bool state)
|
||||||
{
|
{
|
||||||
g_state.cop0_regs.cause.Ip |= static_cast<u8>(1u << bit);
|
// Only uses bit 10.
|
||||||
|
constexpr u32 bit = (1u << 10);
|
||||||
|
const u32 old_cause = g_state.cop0_regs.cause.bits;
|
||||||
|
g_state.cop0_regs.cause.bits = (g_state.cop0_regs.cause.bits & ~bit) | (state ? bit : 0u);
|
||||||
|
if (old_cause ^ g_state.cop0_regs.cause.bits && state)
|
||||||
CheckForPendingInterrupt();
|
CheckForPendingInterrupt();
|
||||||
}
|
}
|
||||||
|
|
||||||
void CPU::ClearExternalInterrupt(u8 bit)
|
|
||||||
{
|
|
||||||
g_state.cop0_regs.cause.Ip &= static_cast<u8>(~(1u << bit));
|
|
||||||
}
|
|
||||||
|
|
||||||
ALWAYS_INLINE_RELEASE void CPU::UpdateLoadDelay()
|
ALWAYS_INLINE_RELEASE void CPU::UpdateLoadDelay()
|
||||||
{
|
{
|
||||||
// the old value is needed in case the delay slot instruction overwrites the same register
|
// the old value is needed in case the delay slot instruction overwrites the same register
|
||||||
|
|
|
@ -176,8 +176,7 @@ bool SafeWriteMemoryHalfWord(VirtualMemoryAddress addr, u16 value);
|
||||||
bool SafeWriteMemoryWord(VirtualMemoryAddress addr, u32 value);
|
bool SafeWriteMemoryWord(VirtualMemoryAddress addr, u32 value);
|
||||||
|
|
||||||
// External IRQs
|
// External IRQs
|
||||||
void SetExternalInterrupt(u8 bit);
|
void SetIRQRequest(bool state);
|
||||||
void ClearExternalInterrupt(u8 bit);
|
|
||||||
|
|
||||||
void DisassembleAndPrint(u32 addr);
|
void DisassembleAndPrint(u32 addr);
|
||||||
void DisassembleAndLog(u32 addr);
|
void DisassembleAndLog(u32 addr);
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
// SPDX-FileCopyrightText: 2019-2022 Connor McLaughlin <stenzek@gmail.com>
|
// SPDX-FileCopyrightText: 2019-2024 Connor McLaughlin <stenzek@gmail.com>
|
||||||
// SPDX-License-Identifier: (GPL-3.0 OR CC-BY-NC-ND-4.0)
|
// SPDX-License-Identifier: (GPL-3.0 OR CC-BY-NC-ND-4.0)
|
||||||
|
|
||||||
#include "dma.h"
|
#include "dma.h"
|
||||||
|
@ -22,6 +22,8 @@
|
||||||
#include "common/log.h"
|
#include "common/log.h"
|
||||||
#include "common/string_util.h"
|
#include "common/string_util.h"
|
||||||
|
|
||||||
|
#include "fmt/format.h"
|
||||||
|
|
||||||
#include <array>
|
#include <array>
|
||||||
#include <memory>
|
#include <memory>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
@ -203,8 +205,20 @@ static constexpr std::array<bool (*)(), NUM_CHANNELS> s_channel_transfer_functio
|
||||||
&TransferChannel<Channel::OTC>,
|
&TransferChannel<Channel::OTC>,
|
||||||
}};
|
}};
|
||||||
|
|
||||||
|
[[maybe_unused]] static constexpr std::array<const char*, NUM_CHANNELS> s_channel_names = {
|
||||||
|
{"MDECin", "MDECout", "GPU", "CDROM", "SPU", "PIO", "OTC"}};
|
||||||
|
|
||||||
}; // namespace DMA
|
}; // namespace DMA
|
||||||
|
|
||||||
|
template<>
|
||||||
|
struct fmt::formatter<DMA::Channel> : fmt::formatter<fmt::string_view>
|
||||||
|
{
|
||||||
|
auto format(DMA::Channel channel, fmt::format_context& ctx) const
|
||||||
|
{
|
||||||
|
return formatter<fmt::string_view>::format(DMA::s_channel_names[static_cast<u32>(channel)], ctx);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
u32 DMA::GetAddressMask()
|
u32 DMA::GetAddressMask()
|
||||||
{
|
{
|
||||||
return Bus::g_ram_mask & 0xFFFFFFFCu;
|
return Bus::g_ram_mask & 0xFFFFFFFCu;
|
||||||
|
@ -332,12 +346,13 @@ void DMA::WriteRegister(u32 offset, u32 value)
|
||||||
case 0x00:
|
case 0x00:
|
||||||
{
|
{
|
||||||
state.base_address = value & BASE_ADDRESS_MASK;
|
state.base_address = value & BASE_ADDRESS_MASK;
|
||||||
Log_TracePrintf("DMA channel %u base address <- 0x%08X", channel_index, state.base_address);
|
Log_TraceFmt("DMA channel {} base address <- 0x{:08X}", static_cast<Channel>(channel_index),
|
||||||
|
state.base_address);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
case 0x04:
|
case 0x04:
|
||||||
{
|
{
|
||||||
Log_TracePrintf("DMA channel %u block control <- 0x%08X", channel_index, value);
|
Log_TraceFmt("DMA channel {} block control <- 0x{:08X}", static_cast<Channel>(channel_index), value);
|
||||||
state.block_control.bits = value;
|
state.block_control.bits = value;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -352,14 +367,42 @@ void DMA::WriteRegister(u32 offset, u32 value)
|
||||||
|
|
||||||
state.channel_control.bits = (state.channel_control.bits & ~ChannelState::ChannelControl::WRITE_MASK) |
|
state.channel_control.bits = (state.channel_control.bits & ~ChannelState::ChannelControl::WRITE_MASK) |
|
||||||
(value & ChannelState::ChannelControl::WRITE_MASK);
|
(value & ChannelState::ChannelControl::WRITE_MASK);
|
||||||
Log_TracePrintf("DMA channel %u channel control <- 0x%08X", channel_index, state.channel_control.bits);
|
Log_TracePrintf("DMA channel {} channel control <- 0x{:08X}", static_cast<Channel>(channel_index),
|
||||||
|
state.channel_control.bits);
|
||||||
|
|
||||||
// start/trigger bit must be enabled for OTC
|
// start/trigger bit must be enabled for OTC
|
||||||
if (static_cast<Channel>(channel_index) == Channel::OTC)
|
if (static_cast<Channel>(channel_index) == Channel::OTC)
|
||||||
SetRequest(static_cast<Channel>(channel_index), state.channel_control.start_trigger);
|
SetRequest(static_cast<Channel>(channel_index), state.channel_control.start_trigger);
|
||||||
|
|
||||||
if (CanTransferChannel(static_cast<Channel>(channel_index), ignore_halt))
|
if (CanTransferChannel(static_cast<Channel>(channel_index), ignore_halt))
|
||||||
|
{
|
||||||
|
if (static_cast<Channel>(channel_index) != Channel::OTC &&
|
||||||
|
state.channel_control.sync_mode == SyncMode::Manual && state.channel_control.chopping_enable)
|
||||||
|
{
|
||||||
|
// Figure out how roughly many CPU cycles it'll take for the transfer to complete, and delay the transfer.
|
||||||
|
// Needed for Lagnacure Legend, which sets DICR to enable interrupts after CHCR to kickstart the transfer.
|
||||||
|
// This has an artificial 500 cycle cap, setting it too high causes Namco Museum Vol. 4 and a couple of
|
||||||
|
// other games to crash... so clearly something is missing here.
|
||||||
|
const u32 block_words = (1u << state.channel_control.chopping_dma_window_size);
|
||||||
|
const u32 cpu_cycles_per_block = (1u << state.channel_control.chopping_cpu_window_size);
|
||||||
|
const u32 blocks = state.block_control.manual.word_count / block_words;
|
||||||
|
const TickCount delay_cycles = std::min(static_cast<TickCount>(cpu_cycles_per_block * blocks), 500);
|
||||||
|
if (delay_cycles > 1 && true)
|
||||||
|
{
|
||||||
|
Log_DevFmt("Delaying {} transfer by {} cycles due to chopping", static_cast<Channel>(channel_index),
|
||||||
|
delay_cycles);
|
||||||
|
HaltTransfer(delay_cycles);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
s_channel_transfer_functions[channel_index]();
|
s_channel_transfer_functions[channel_index]();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
s_channel_transfer_functions[channel_index]();
|
||||||
|
}
|
||||||
|
}
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -393,7 +436,7 @@ void DMA::WriteRegister(u32 offset, u32 value)
|
||||||
Log_TracePrintf("DCIR <- 0x%08X", value);
|
Log_TracePrintf("DCIR <- 0x%08X", value);
|
||||||
s_DICR.bits = (s_DICR.bits & ~DICR_WRITE_MASK) | (value & DICR_WRITE_MASK);
|
s_DICR.bits = (s_DICR.bits & ~DICR_WRITE_MASK) | (value & DICR_WRITE_MASK);
|
||||||
s_DICR.bits = s_DICR.bits & ~(value & DICR_RESET_MASK);
|
s_DICR.bits = s_DICR.bits & ~(value & DICR_RESET_MASK);
|
||||||
s_DICR.UpdateMasterFlag();
|
UpdateIRQ();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -450,10 +493,8 @@ void DMA::UpdateIRQ()
|
||||||
{
|
{
|
||||||
s_DICR.UpdateMasterFlag();
|
s_DICR.UpdateMasterFlag();
|
||||||
if (s_DICR.master_flag)
|
if (s_DICR.master_flag)
|
||||||
{
|
|
||||||
Log_TracePrintf("Firing DMA master interrupt");
|
Log_TracePrintf("Firing DMA master interrupt");
|
||||||
InterruptController::InterruptRequest(InterruptController::IRQ::DMA);
|
InterruptController::SetLineState(InterruptController::IRQ::DMA, s_DICR.master_flag);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Plenty of games seem to suffer from this issue where they have a linked list DMA going while polling the
|
// Plenty of games seem to suffer from this issue where they have a linked list DMA going while polling the
|
||||||
|
@ -576,10 +617,10 @@ bool DMA::TransferChannel()
|
||||||
|
|
||||||
case SyncMode::Request:
|
case SyncMode::Request:
|
||||||
{
|
{
|
||||||
Log_DebugPrintf("DMA%u: Copying %u blocks of size %u (%u total words) %s 0x%08X", static_cast<u32>(channel),
|
Log_DebugFmt("DMA[{}]: Copying {} blocks of size {} ({} total words) {} 0x{:08X}", channel,
|
||||||
cs.block_control.request.GetBlockCount(), cs.block_control.request.GetBlockSize(),
|
cs.block_control.request.GetBlockCount(), cs.block_control.request.GetBlockSize(),
|
||||||
cs.block_control.request.GetBlockCount() * cs.block_control.request.GetBlockSize(),
|
cs.block_control.request.GetBlockCount() * cs.block_control.request.GetBlockSize(),
|
||||||
copy_to_device ? "from" : "to", current_address & mask);
|
copy_to_device ? "from" : "to", current_address);
|
||||||
|
|
||||||
const u32 block_size = cs.block_control.request.GetBlockSize();
|
const u32 block_size = cs.block_control.request.GetBlockSize();
|
||||||
u32 blocks_remaining = cs.block_control.request.GetBlockCount();
|
u32 blocks_remaining = cs.block_control.request.GetBlockCount();
|
||||||
|
@ -638,10 +679,11 @@ bool DMA::TransferChannel()
|
||||||
}
|
}
|
||||||
|
|
||||||
// start/busy bit is cleared on end of transfer
|
// start/busy bit is cleared on end of transfer
|
||||||
|
Log_DebugFmt("DMA transfer for channel {} complete", channel);
|
||||||
cs.channel_control.enable_busy = false;
|
cs.channel_control.enable_busy = false;
|
||||||
if (s_DICR.IsIRQEnabled(channel))
|
if (s_DICR.IsIRQEnabled(channel))
|
||||||
{
|
{
|
||||||
Log_DebugPrintf("Set DMA interrupt for channel %u", static_cast<u32>(channel));
|
Log_DebugFmt("Setting DMA interrupt for channel {}", channel);
|
||||||
s_DICR.SetIRQFlag(channel);
|
s_DICR.SetIRQFlag(channel);
|
||||||
UpdateIRQ();
|
UpdateIRQ();
|
||||||
}
|
}
|
||||||
|
@ -816,8 +858,6 @@ void DMA::DrawDebugStateWindow()
|
||||||
static constexpr u32 NUM_COLUMNS = 10;
|
static constexpr u32 NUM_COLUMNS = 10;
|
||||||
static constexpr std::array<const char*, NUM_COLUMNS> column_names = {
|
static constexpr std::array<const char*, NUM_COLUMNS> column_names = {
|
||||||
{"#", "Req", "Direction", "Chopping", "Mode", "Busy", "Enable", "Priority", "IRQ", "Flag"}};
|
{"#", "Req", "Direction", "Chopping", "Mode", "Busy", "Enable", "Priority", "IRQ", "Flag"}};
|
||||||
static constexpr std::array<const char*, NUM_CHANNELS> channel_names = {
|
|
||||||
{"MDECin", "MDECout", "GPU", "CDROM", "SPU", "PIO", "OTC"}};
|
|
||||||
static constexpr std::array<const char*, 4> sync_mode_names = {{"Manual", "Request", "LinkedList", "Reserved"}};
|
static constexpr std::array<const char*, 4> sync_mode_names = {{"Manual", "Request", "LinkedList", "Reserved"}};
|
||||||
|
|
||||||
const float framebuffer_scale = Host::GetOSDScale();
|
const float framebuffer_scale = Host::GetOSDScale();
|
||||||
|
@ -854,7 +894,7 @@ void DMA::DrawDebugStateWindow()
|
||||||
{
|
{
|
||||||
const ChannelState& cs = s_state[i];
|
const ChannelState& cs = s_state[i];
|
||||||
|
|
||||||
ImGui::TextColored(cs.channel_control.enable_busy ? active : inactive, "%u[%s]", i, channel_names[i]);
|
ImGui::TextColored(cs.channel_control.enable_busy ? active : inactive, "%u[%s]", i, s_channel_names[i]);
|
||||||
ImGui::NextColumn();
|
ImGui::NextColumn();
|
||||||
ImGui::TextColored(cs.request ? active : inactive, cs.request ? "Yes" : "No");
|
ImGui::TextColored(cs.request ? active : inactive, cs.request ? "Yes" : "No");
|
||||||
ImGui::NextColumn();
|
ImGui::NextColumn();
|
||||||
|
|
|
@ -961,6 +961,7 @@ void GPU::CRTCTickEvent(TickCount ticks)
|
||||||
m_crtc_state.current_scanline >= m_crtc_state.vertical_display_end)
|
m_crtc_state.current_scanline >= m_crtc_state.vertical_display_end)
|
||||||
{
|
{
|
||||||
Timers::SetGate(HBLANK_TIMER_INDEX, false);
|
Timers::SetGate(HBLANK_TIMER_INDEX, false);
|
||||||
|
InterruptController::SetLineState(InterruptController::IRQ::VBLANK, false);
|
||||||
m_crtc_state.in_vblank = false;
|
m_crtc_state.in_vblank = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -971,7 +972,6 @@ void GPU::CRTCTickEvent(TickCount ticks)
|
||||||
if (new_vblank)
|
if (new_vblank)
|
||||||
{
|
{
|
||||||
Log_DebugPrintf("Now in v-blank");
|
Log_DebugPrintf("Now in v-blank");
|
||||||
InterruptController::InterruptRequest(InterruptController::IRQ::VBLANK);
|
|
||||||
|
|
||||||
// flush any pending draws and "scan out" the image
|
// flush any pending draws and "scan out" the image
|
||||||
// TODO: move present in here I guess
|
// TODO: move present in here I guess
|
||||||
|
@ -987,6 +987,7 @@ void GPU::CRTCTickEvent(TickCount ticks)
|
||||||
}
|
}
|
||||||
|
|
||||||
Timers::SetGate(HBLANK_TIMER_INDEX, new_vblank);
|
Timers::SetGate(HBLANK_TIMER_INDEX, new_vblank);
|
||||||
|
InterruptController::SetLineState(InterruptController::IRQ::VBLANK, new_vblank);
|
||||||
m_crtc_state.in_vblank = new_vblank;
|
m_crtc_state.in_vblank = new_vblank;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -208,11 +208,9 @@ bool GPU::HandleClearCacheCommand()
|
||||||
bool GPU::HandleInterruptRequestCommand()
|
bool GPU::HandleInterruptRequestCommand()
|
||||||
{
|
{
|
||||||
Log_DebugPrintf("GP0 interrupt request");
|
Log_DebugPrintf("GP0 interrupt request");
|
||||||
if (!m_GPUSTAT.interrupt_request)
|
|
||||||
{
|
|
||||||
m_GPUSTAT.interrupt_request = true;
|
m_GPUSTAT.interrupt_request = true;
|
||||||
InterruptController::InterruptRequest(InterruptController::IRQ::GPU);
|
InterruptController::SetLineState(InterruptController::IRQ::GPU, m_GPUSTAT.interrupt_request);
|
||||||
}
|
|
||||||
|
|
||||||
m_fifo.RemoveOne();
|
m_fifo.RemoveOne();
|
||||||
AddCommandTicks(1);
|
AddCommandTicks(1);
|
||||||
|
|
|
@ -1,54 +1,64 @@
|
||||||
// SPDX-FileCopyrightText: 2019-2022 Connor McLaughlin <stenzek@gmail.com>
|
// SPDX-FileCopyrightText: 2019-2024 Connor McLaughlin <stenzek@gmail.com>
|
||||||
// SPDX-License-Identifier: (GPL-3.0 OR CC-BY-NC-ND-4.0)
|
// SPDX-License-Identifier: (GPL-3.0 OR CC-BY-NC-ND-4.0)
|
||||||
|
|
||||||
#include "interrupt_controller.h"
|
#include "interrupt_controller.h"
|
||||||
#include "common/log.h"
|
|
||||||
#include "cpu_core.h"
|
#include "cpu_core.h"
|
||||||
|
|
||||||
#include "util/state_wrapper.h"
|
#include "util/state_wrapper.h"
|
||||||
|
|
||||||
|
#include "common/log.h"
|
||||||
|
|
||||||
Log_SetChannel(InterruptController);
|
Log_SetChannel(InterruptController);
|
||||||
|
|
||||||
namespace InterruptController {
|
namespace InterruptController {
|
||||||
|
|
||||||
static constexpr u32 REGISTER_WRITE_MASK = (u32(1) << NUM_IRQS) - 1;
|
static constexpr u32 REGISTER_WRITE_MASK = (u32(1) << NUM_IRQS) - 1;
|
||||||
static constexpr u32 DEFAULT_INTERRUPT_MASK = 0; //(u32(1) << NUM_IRQS) - 1;
|
static constexpr u32 DEFAULT_INTERRUPT_MASK = 0;
|
||||||
|
|
||||||
static void UpdateCPUInterruptRequest();
|
static void UpdateCPUInterruptRequest();
|
||||||
|
|
||||||
static u32 s_interrupt_status_register = 0;
|
static u32 s_interrupt_status_register = 0;
|
||||||
static u32 s_interrupt_mask_register = DEFAULT_INTERRUPT_MASK;
|
static u32 s_interrupt_mask_register = DEFAULT_INTERRUPT_MASK;
|
||||||
|
static u32 s_interrupt_line_state = 0;
|
||||||
|
|
||||||
|
[[maybe_unused]] static constexpr std::array<const char*, static_cast<size_t>(IRQ::MaxCount)> s_irq_names = {
|
||||||
|
{"VBLANK", "GPU", "CDROM", "DMA", "TMR0", "TMR1", "TMR2", "PAD", "SIO", "SPU", "IRQ10"}};
|
||||||
|
|
||||||
} // namespace InterruptController
|
} // namespace InterruptController
|
||||||
|
|
||||||
void InterruptController::Initialize()
|
|
||||||
{
|
|
||||||
Reset();
|
|
||||||
}
|
|
||||||
|
|
||||||
void InterruptController::Shutdown() {}
|
|
||||||
|
|
||||||
void InterruptController::Reset()
|
void InterruptController::Reset()
|
||||||
{
|
{
|
||||||
s_interrupt_status_register = 0;
|
s_interrupt_status_register = 0;
|
||||||
s_interrupt_mask_register = DEFAULT_INTERRUPT_MASK;
|
s_interrupt_mask_register = DEFAULT_INTERRUPT_MASK;
|
||||||
|
s_interrupt_line_state = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool InterruptController::DoState(StateWrapper& sw)
|
bool InterruptController::DoState(StateWrapper& sw)
|
||||||
{
|
{
|
||||||
sw.Do(&s_interrupt_status_register);
|
sw.Do(&s_interrupt_status_register);
|
||||||
sw.Do(&s_interrupt_mask_register);
|
sw.Do(&s_interrupt_mask_register);
|
||||||
|
sw.DoEx(&s_interrupt_line_state, 63, s_interrupt_status_register);
|
||||||
|
|
||||||
return !sw.HasError();
|
return !sw.HasError();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool InterruptController::GetIRQLineState()
|
void InterruptController::SetLineState(IRQ irq, bool state)
|
||||||
{
|
{
|
||||||
return (s_interrupt_status_register != 0);
|
// Interupts are edge-triggered, so only set the flag in the status register on a 0-1 transition.
|
||||||
}
|
const u32 bit = (1u << static_cast<u32>(irq));
|
||||||
|
const u32 prev_state = s_interrupt_line_state;
|
||||||
|
s_interrupt_line_state = (s_interrupt_line_state & ~bit) | (state ? bit : 0u);
|
||||||
|
if (s_interrupt_line_state == prev_state)
|
||||||
|
return;
|
||||||
|
|
||||||
void InterruptController::InterruptRequest(IRQ irq)
|
#ifdef _DEBUG
|
||||||
{
|
if (!(prev_state & bit) && state)
|
||||||
const u32 bit = (u32(1) << static_cast<u32>(irq));
|
Log_DebugFmt("{} IRQ triggered", s_irq_names[static_cast<size_t>(irq)]);
|
||||||
s_interrupt_status_register |= bit;
|
else if ((prev_state & bit) && !state)
|
||||||
|
Log_DebugFmt("{} IRQ line inactive", s_irq_names[static_cast<size_t>(irq)]);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
s_interrupt_status_register |= (state ? (prev_state ^ s_interrupt_line_state) : 0u) & s_interrupt_line_state;
|
||||||
UpdateCPUInterruptRequest();
|
UpdateCPUInterruptRequest();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -74,8 +84,14 @@ void InterruptController::WriteRegister(u32 offset, u32 value)
|
||||||
{
|
{
|
||||||
case 0x00: // I_STATUS
|
case 0x00: // I_STATUS
|
||||||
{
|
{
|
||||||
if ((s_interrupt_status_register & ~value) != 0)
|
#ifdef _DEBUG
|
||||||
Log_DebugPrintf("Clearing bits 0x%08X", (s_interrupt_status_register & ~value));
|
const u32 cleared_bits = (s_interrupt_status_register & ~value);
|
||||||
|
for (u32 i = 0; i < static_cast<u32>(IRQ::MaxCount); i++)
|
||||||
|
{
|
||||||
|
if (cleared_bits & (1u << i))
|
||||||
|
Log_DebugFmt("{} IRQ cleared", s_irq_names[i]);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
s_interrupt_status_register = s_interrupt_status_register & (value & REGISTER_WRITE_MASK);
|
s_interrupt_status_register = s_interrupt_status_register & (value & REGISTER_WRITE_MASK);
|
||||||
UpdateCPUInterruptRequest();
|
UpdateCPUInterruptRequest();
|
||||||
|
@ -96,11 +112,8 @@ void InterruptController::WriteRegister(u32 offset, u32 value)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void InterruptController::UpdateCPUInterruptRequest()
|
ALWAYS_INLINE_RELEASE void InterruptController::UpdateCPUInterruptRequest()
|
||||||
{
|
{
|
||||||
// external interrupts set bit 10 only?
|
const bool state = (s_interrupt_status_register & s_interrupt_mask_register) != 0;
|
||||||
if ((s_interrupt_status_register & s_interrupt_mask_register) != 0)
|
CPU::SetIRQRequest(state);
|
||||||
CPU::SetExternalInterrupt(2);
|
|
||||||
else
|
|
||||||
CPU::ClearExternalInterrupt(2);
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
// SPDX-FileCopyrightText: 2019-2022 Connor McLaughlin <stenzek@gmail.com>
|
// SPDX-FileCopyrightText: 2019-2024 Connor McLaughlin <stenzek@gmail.com>
|
||||||
// SPDX-License-Identifier: (GPL-3.0 OR CC-BY-NC-ND-4.0)
|
// SPDX-License-Identifier: (GPL-3.0 OR CC-BY-NC-ND-4.0)
|
||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
@ -19,24 +19,19 @@ enum class IRQ : u32
|
||||||
TMR0 = 4, // IRQ4 - TMR0 - Sysclk or Dotclk
|
TMR0 = 4, // IRQ4 - TMR0 - Sysclk or Dotclk
|
||||||
TMR1 = 5, // IRQ5 - TMR1 - Sysclk Hblank
|
TMR1 = 5, // IRQ5 - TMR1 - Sysclk Hblank
|
||||||
TMR2 = 6, // IRQ6 - TMR2 - Sysclk or Sysclk / 8
|
TMR2 = 6, // IRQ6 - TMR2 - Sysclk or Sysclk / 8
|
||||||
IRQ7 = 7, // IRQ7 - Controller and Memory Card Byte Received
|
PAD = 7, // IRQ7 - Controller and Memory Card Byte Received
|
||||||
SIO = 8, // IRQ8 - SIO
|
SIO = 8, // IRQ8 - SIO
|
||||||
SPU = 9, // IRQ9 - SPU
|
SPU = 9, // IRQ9 - SPU
|
||||||
IRQ10 = 10 // IRQ10 - Lightpen interrupt, PIO
|
IRQ10 = 10, // IRQ10 - Lightpen interrupt, PIO
|
||||||
|
|
||||||
|
MaxCount
|
||||||
};
|
};
|
||||||
|
|
||||||
void Initialize();
|
|
||||||
void Shutdown();
|
|
||||||
void Reset();
|
void Reset();
|
||||||
bool DoState(StateWrapper& sw);
|
bool DoState(StateWrapper& sw);
|
||||||
|
|
||||||
// Should mirror CPU state.
|
void SetLineState(IRQ irq, bool state);
|
||||||
bool GetIRQLineState();
|
|
||||||
|
|
||||||
// Interupts are edge-triggered, so if it is masked when TriggerInterrupt() is called, it will be lost.
|
|
||||||
void InterruptRequest(IRQ irq);
|
|
||||||
|
|
||||||
// I/O
|
|
||||||
u32 ReadRegister(u32 offset);
|
u32 ReadRegister(u32 offset);
|
||||||
void WriteRegister(u32 offset, u32 value);
|
void WriteRegister(u32 offset, u32 value);
|
||||||
|
|
||||||
|
|
|
@ -639,6 +639,7 @@ void Pad::WriteRegister(u32 offset, u32 value)
|
||||||
{
|
{
|
||||||
// reset stat bits
|
// reset stat bits
|
||||||
s_JOY_STAT.INTR = false;
|
s_JOY_STAT.INTR = false;
|
||||||
|
InterruptController::SetLineState(InterruptController::IRQ::PAD, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!s_JOY_CTRL.SELECT)
|
if (!s_JOY_CTRL.SELECT)
|
||||||
|
@ -883,7 +884,7 @@ void Pad::DoACK()
|
||||||
{
|
{
|
||||||
Log_DebugPrintf("Triggering ACK interrupt");
|
Log_DebugPrintf("Triggering ACK interrupt");
|
||||||
s_JOY_STAT.INTR = true;
|
s_JOY_STAT.INTR = true;
|
||||||
InterruptController::InterruptRequest(InterruptController::IRQ::IRQ7);
|
InterruptController::SetLineState(InterruptController::IRQ::PAD, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
EndTransfer();
|
EndTransfer();
|
||||||
|
|
|
@ -1,11 +1,11 @@
|
||||||
// SPDX-FileCopyrightText: 2019-2022 Connor McLaughlin <stenzek@gmail.com>
|
// SPDX-FileCopyrightText: 2019-2024 Connor McLaughlin <stenzek@gmail.com>
|
||||||
// SPDX-License-Identifier: (GPL-3.0 OR CC-BY-NC-ND-4.0)
|
// SPDX-License-Identifier: (GPL-3.0 OR CC-BY-NC-ND-4.0)
|
||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
#include "types.h"
|
#include "types.h"
|
||||||
|
|
||||||
static constexpr u32 SAVE_STATE_MAGIC = 0x43435544;
|
static constexpr u32 SAVE_STATE_MAGIC = 0x43435544;
|
||||||
static constexpr u32 SAVE_STATE_VERSION = 62;
|
static constexpr u32 SAVE_STATE_VERSION = 63;
|
||||||
static constexpr u32 SAVE_STATE_MINIMUM_VERSION = 42;
|
static constexpr u32 SAVE_STATE_MINIMUM_VERSION = 42;
|
||||||
|
|
||||||
static_assert(SAVE_STATE_VERSION >= SAVE_STATE_MINIMUM_VERSION);
|
static_assert(SAVE_STATE_VERSION >= SAVE_STATE_MINIMUM_VERSION);
|
||||||
|
|
|
@ -929,9 +929,14 @@ void SPU::WriteRegister(u32 offset, u16 value)
|
||||||
s_SPUSTAT.mode = s_SPUCNT.mode.GetValue();
|
s_SPUSTAT.mode = s_SPUCNT.mode.GetValue();
|
||||||
|
|
||||||
if (!s_SPUCNT.irq9_enable)
|
if (!s_SPUCNT.irq9_enable)
|
||||||
|
{
|
||||||
s_SPUSTAT.irq9_flag = false;
|
s_SPUSTAT.irq9_flag = false;
|
||||||
|
InterruptController::SetLineState(InterruptController::IRQ::SPU, false);
|
||||||
|
}
|
||||||
else if (IsRAMIRQTriggerable())
|
else if (IsRAMIRQTriggerable())
|
||||||
|
{
|
||||||
CheckForLateRAMIRQs();
|
CheckForLateRAMIRQs();
|
||||||
|
}
|
||||||
|
|
||||||
UpdateEventInterval();
|
UpdateEventInterval();
|
||||||
UpdateDMARequest();
|
UpdateDMARequest();
|
||||||
|
@ -1155,7 +1160,7 @@ void SPU::TriggerRAMIRQ()
|
||||||
{
|
{
|
||||||
DebugAssert(IsRAMIRQTriggerable());
|
DebugAssert(IsRAMIRQTriggerable());
|
||||||
s_SPUSTAT.irq9_flag = true;
|
s_SPUSTAT.irq9_flag = true;
|
||||||
InterruptController::InterruptRequest(InterruptController::IRQ::SPU);
|
InterruptController::SetLineState(InterruptController::IRQ::SPU, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
void SPU::CheckForLateRAMIRQs()
|
void SPU::CheckForLateRAMIRQs()
|
||||||
|
|
|
@ -1622,8 +1622,6 @@ bool System::Initialize(bool force_software_renderer)
|
||||||
}
|
}
|
||||||
|
|
||||||
DMA::Initialize();
|
DMA::Initialize();
|
||||||
InterruptController::Initialize();
|
|
||||||
|
|
||||||
CDROM::Initialize();
|
CDROM::Initialize();
|
||||||
Pad::Initialize();
|
Pad::Initialize();
|
||||||
Timers::Initialize();
|
Timers::Initialize();
|
||||||
|
@ -1675,7 +1673,6 @@ void System::DestroySystem()
|
||||||
Pad::Shutdown();
|
Pad::Shutdown();
|
||||||
CDROM::Shutdown();
|
CDROM::Shutdown();
|
||||||
g_gpu.reset();
|
g_gpu.reset();
|
||||||
InterruptController::Shutdown();
|
|
||||||
DMA::Shutdown();
|
DMA::Shutdown();
|
||||||
CPU::PGXP::Shutdown();
|
CPU::PGXP::Shutdown();
|
||||||
CPU::CodeCache::Shutdown();
|
CPU::CodeCache::Shutdown();
|
||||||
|
|
|
@ -66,7 +66,6 @@ struct CounterState
|
||||||
|
|
||||||
static void UpdateCountingEnabled(CounterState& cs);
|
static void UpdateCountingEnabled(CounterState& cs);
|
||||||
static void CheckForIRQ(u32 index, u32 old_counter);
|
static void CheckForIRQ(u32 index, u32 old_counter);
|
||||||
static void UpdateIRQ(u32 index);
|
|
||||||
|
|
||||||
static void AddSysClkTicks(void*, TickCount sysclk_ticks, TickCount ticks_late);
|
static void AddSysClkTicks(void*, TickCount sysclk_ticks, TickCount ticks_late);
|
||||||
|
|
||||||
|
@ -236,17 +235,29 @@ void Timers::CheckForIRQ(u32 timer, u32 old_counter)
|
||||||
|
|
||||||
if (interrupt_request)
|
if (interrupt_request)
|
||||||
{
|
{
|
||||||
|
const InterruptController::IRQ irqnum =
|
||||||
|
static_cast<InterruptController::IRQ>(static_cast<u32>(InterruptController::IRQ::TMR0) + timer);
|
||||||
if (!cs.mode.irq_pulse_n)
|
if (!cs.mode.irq_pulse_n)
|
||||||
|
{
|
||||||
|
if (!cs.irq_done || cs.mode.irq_repeat)
|
||||||
{
|
{
|
||||||
// this is actually low for a few cycles
|
// this is actually low for a few cycles
|
||||||
cs.mode.interrupt_request_n = false;
|
Log_DebugPrintf("Raising timer %u pulse IRQ", timer);
|
||||||
UpdateIRQ(timer);
|
InterruptController::SetLineState(irqnum, false);
|
||||||
|
InterruptController::SetLineState(irqnum, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
cs.irq_done = true;
|
||||||
cs.mode.interrupt_request_n = true;
|
cs.mode.interrupt_request_n = true;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
// TODO: How does the non-repeat mode work here?
|
||||||
cs.mode.interrupt_request_n ^= true;
|
cs.mode.interrupt_request_n ^= true;
|
||||||
UpdateIRQ(timer);
|
if (!cs.mode.interrupt_request_n)
|
||||||
|
Log_DebugPrintf("Raising timer %u alternate IRQ", timer);
|
||||||
|
|
||||||
|
InterruptController::SetLineState(irqnum, !cs.mode.interrupt_request_n);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -371,10 +382,11 @@ void Timers::WriteRegister(u32 offset, u32 value)
|
||||||
cs.use_external_clock = (cs.mode.clock_source & (timer_index == 2 ? 2 : 1)) != 0;
|
cs.use_external_clock = (cs.mode.clock_source & (timer_index == 2 ? 2 : 1)) != 0;
|
||||||
cs.counter = 0;
|
cs.counter = 0;
|
||||||
cs.irq_done = false;
|
cs.irq_done = false;
|
||||||
|
InterruptController::SetLineState(
|
||||||
|
static_cast<InterruptController::IRQ>(static_cast<u32>(InterruptController::IRQ::TMR0) + timer_index), false);
|
||||||
|
|
||||||
UpdateCountingEnabled(cs);
|
UpdateCountingEnabled(cs);
|
||||||
CheckForIRQ(timer_index, cs.counter);
|
CheckForIRQ(timer_index, cs.counter);
|
||||||
UpdateIRQ(timer_index);
|
|
||||||
UpdateSysClkEvent();
|
UpdateSysClkEvent();
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
@ -423,18 +435,6 @@ void Timers::UpdateCountingEnabled(CounterState& cs)
|
||||||
cs.external_counting_enabled = cs.use_external_clock && cs.counting_enabled;
|
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<InterruptController::IRQ>(static_cast<u32>(InterruptController::IRQ::TMR0) + index));
|
|
||||||
}
|
|
||||||
|
|
||||||
TickCount Timers::GetTicksUntilNextInterrupt()
|
TickCount Timers::GetTicksUntilNextInterrupt()
|
||||||
{
|
{
|
||||||
TickCount min_ticks = System::GetMaxSliceTicks();
|
TickCount min_ticks = System::GetMaxSliceTicks();
|
||||||
|
|
Loading…
Reference in a new issue