Duckstation/src/core/dma.cpp

523 lines
14 KiB
C++
Raw Normal View History

2019-09-09 07:01:26 +00:00
#include "dma.h"
#include "bus.h"
2019-09-21 15:12:16 +00:00
#include "cdrom.h"
#include "common/log.h"
2019-09-14 10:28:47 +00:00
#include "common/state_wrapper.h"
#include "common/string_util.h"
2019-09-11 04:59:41 +00:00
#include "gpu.h"
2019-09-24 09:43:10 +00:00
#include "interrupt_controller.h"
2019-09-29 02:51:34 +00:00
#include "mdec.h"
#include "spu.h"
2019-09-24 11:38:58 +00:00
#include "system.h"
2019-09-09 07:01:26 +00:00
Log_SetChannel(DMA);
DMA::DMA() = default;
DMA::~DMA() = default;
2019-11-11 08:19:57 +00:00
void DMA::Initialize(System* system, Bus* bus, InterruptController* interrupt_controller, GPU* gpu, CDROM* cdrom,
2019-09-29 02:51:34 +00:00
SPU* spu, MDEC* mdec)
2019-09-09 07:01:26 +00:00
{
2019-09-24 11:38:58 +00:00
m_system = system;
2019-09-09 07:01:26 +00:00
m_bus = bus;
2019-09-24 09:43:10 +00:00
m_interrupt_controller = interrupt_controller;
2019-09-09 07:01:26 +00:00
m_gpu = gpu;
2019-09-21 15:12:16 +00:00
m_cdrom = cdrom;
m_spu = spu;
2019-09-29 02:51:34 +00:00
m_mdec = mdec;
m_transfer_buffer.resize(32);
for (u32 i = 0; i < NUM_CHANNELS; i++)
{
m_state[i].transfer_event = system->CreateTimingEvent(
StringUtil::StdStringFromFormat("DMA%u Transfer", i), 1, 1,
std::bind(&DMA::TransferChannel, this, static_cast<Channel>(i), std::placeholders::_2), false);
}
2019-09-09 07:01:26 +00:00
}
void DMA::Reset()
{
m_DPCR.bits = 0x07654321;
2019-09-24 09:43:10 +00:00
m_DICR.bits = 0;
for (u32 i = 0; i < NUM_CHANNELS; i++)
{
ChannelState& cs = m_state[i];
cs.base_address = 0;
cs.block_control.bits = 0;
cs.channel_control.bits = 0;
cs.request = false;
cs.transfer_event->Deactivate();
}
2019-09-09 07:01:26 +00:00
}
2019-09-14 10:28:47 +00:00
bool DMA::DoState(StateWrapper& sw)
{
for (u32 i = 0; i < NUM_CHANNELS; i++)
{
ChannelState& cs = m_state[i];
sw.Do(&cs.base_address);
sw.Do(&cs.block_control.bits);
sw.Do(&cs.channel_control.bits);
sw.Do(&cs.request);
}
sw.Do(&m_DPCR.bits);
2019-09-24 09:43:10 +00:00
sw.Do(&m_DICR.bits);
if (sw.IsReading())
{
for (u32 i = 0; i < NUM_CHANNELS; i++)
{
m_state[i].transfer_event->Deactivate();
UpdateChannelTransferEvent(static_cast<Channel>(i));
}
}
2019-09-14 10:28:47 +00:00
return !sw.HasError();
}
2019-09-09 07:01:26 +00:00
u32 DMA::ReadRegister(u32 offset)
{
const u32 channel_index = offset >> 4;
if (channel_index < 7)
{
switch (offset & UINT32_C(0x0F))
{
case 0x00:
{
Log_TracePrintf("DMA%u base address -> 0x%08X", channel_index, m_state[channel_index].base_address);
2019-09-09 07:01:26 +00:00
return m_state[channel_index].base_address;
}
2019-09-09 07:01:26 +00:00
case 0x04:
{
Log_TracePrintf("DMA%u block control -> 0x%08X", channel_index, m_state[channel_index].block_control.bits);
2019-09-09 07:01:26 +00:00
return m_state[channel_index].block_control.bits;
}
2019-09-09 07:01:26 +00:00
case 0x08:
{
Log_TracePrintf("DMA%u channel control -> 0x%08X", channel_index, m_state[channel_index].channel_control.bits);
2019-09-09 07:01:26 +00:00
return m_state[channel_index].channel_control.bits;
}
2019-09-09 07:01:26 +00:00
default:
break;
}
}
else
{
if (offset == 0x70)
{
Log_TracePrintf("DPCR -> 0x%08X", m_DPCR.bits);
2019-09-09 07:01:26 +00:00
return m_DPCR.bits;
}
2019-09-09 07:01:26 +00:00
else if (offset == 0x74)
{
Log_TracePrintf("DPCR -> 0x%08X", m_DPCR.bits);
2019-09-24 09:43:10 +00:00
return m_DICR.bits;
}
2019-09-09 07:01:26 +00:00
}
Log_ErrorPrintf("Unhandled register read: %02X", offset);
return UINT32_C(0xFFFFFFFF);
}
void DMA::WriteRegister(u32 offset, u32 value)
{
const u32 channel_index = offset >> 4;
if (channel_index < 7)
{
ChannelState& state = m_state[channel_index];
switch (offset & UINT32_C(0x0F))
{
case 0x00:
{
2019-11-11 10:34:41 +00:00
state.base_address = value & BASE_ADDRESS_MASK;
2019-09-30 10:01:41 +00:00
Log_TracePrintf("DMA channel %u base address <- 0x%08X", channel_index, state.base_address);
return;
}
case 0x04:
{
2019-09-30 10:01:41 +00:00
Log_TracePrintf("DMA channel %u block control <- 0x%08X", channel_index, value);
state.block_control.bits = value;
UpdateChannelTransferEvent(static_cast<Channel>(channel_index));
return;
}
case 0x08:
{
state.channel_control.bits = (state.channel_control.bits & ~ChannelState::ChannelControl::WRITE_MASK) |
(value & ChannelState::ChannelControl::WRITE_MASK);
2019-09-30 10:01:41 +00:00
Log_TracePrintf("DMA channel %u channel control <- 0x%08X", channel_index, state.channel_control.bits);
UpdateChannelTransferEvent(static_cast<Channel>(channel_index));
return;
}
default:
break;
}
}
else
{
switch (offset)
{
case 0x70:
{
2019-09-30 10:01:41 +00:00
Log_TracePrintf("DPCR <- 0x%08X", value);
m_DPCR.bits = value;
for (u32 i = 0; i < NUM_CHANNELS; i++)
UpdateChannelTransferEvent(static_cast<Channel>(i));
return;
}
case 0x74:
{
2019-09-30 10:01:41 +00:00
Log_TracePrintf("DCIR <- 0x%08X", value);
2019-09-24 09:43:10 +00:00
m_DICR.bits = (m_DICR.bits & ~DICR_WRITE_MASK) | (value & DICR_WRITE_MASK);
m_DICR.bits = m_DICR.bits & ~(value & DICR_RESET_MASK);
2019-09-24 09:43:10 +00:00
m_DICR.UpdateMasterFlag();
return;
}
default:
break;
}
}
2019-09-09 07:01:26 +00:00
Log_ErrorPrintf("Unhandled register write: %02X <- %08X", offset, value);
}
void DMA::SetRequest(Channel channel, bool request)
{
ChannelState& cs = m_state[static_cast<u32>(channel)];
if (cs.request == request)
return;
cs.request = request;
if (request)
UpdateChannelTransferEvent(channel);
}
TickCount DMA::GetTransferDelay(Channel channel) const
{
const ChannelState& cs = m_state[static_cast<u32>(channel)];
switch (channel)
{
case Channel::MDECin:
case Channel::MDECout:
return 1;
case Channel::SPU:
{
if (cs.channel_control.sync_mode == SyncMode::Request)
return (cs.block_control.request.GetBlockCount() * (cs.block_control.request.GetBlockSize() / 2));
else
return 1;
}
break;
default:
return 0;
}
}
bool DMA::CanTransferChannel(Channel channel) const
{
if (!m_DPCR.GetMasterEnable(channel))
return false;
const ChannelState& cs = m_state[static_cast<u32>(channel)];
if (!cs.channel_control.enable_busy)
return false;
if (!cs.request && channel != Channel::OTC)
return false;
if (cs.channel_control.sync_mode == SyncMode::Manual && !cs.channel_control.start_trigger)
return false;
return true;
}
2019-09-24 11:38:58 +00:00
bool DMA::CanRunAnyChannels() const
{
for (u32 i = 0; i < NUM_CHANNELS; i++)
{
if (CanTransferChannel(static_cast<Channel>(i)))
2019-09-24 11:38:58 +00:00
return true;
}
return false;
}
void DMA::UpdateIRQ()
{
m_DICR.UpdateMasterFlag();
if (m_DICR.master_flag)
{
Log_TracePrintf("Firing DMA master interrupt");
m_interrupt_controller->InterruptRequest(InterruptController::IRQ::DMA);
}
}
void DMA::UpdateChannelTransferEvent(Channel channel)
{
ChannelState& cs = m_state[static_cast<u32>(channel)];
if (!CanTransferChannel(channel))
{
cs.transfer_event->Deactivate();
return;
}
if (cs.transfer_event->IsActive())
return;
const TickCount ticks = GetTransferDelay(channel);
if (ticks == 0)
{
// immediate transfer
TransferChannel(channel, 0);
return;
}
cs.transfer_event->SetPeriodAndSchedule(ticks);
}
void DMA::TransferChannel(Channel channel, TickCount ticks_late)
{
ChannelState& cs = m_state[static_cast<u32>(channel)];
cs.transfer_event->Deactivate();
const bool copy_to_device = cs.channel_control.copy_to_device;
// start/trigger bit is cleared on beginning of transfer
cs.channel_control.start_trigger = false;
2019-11-11 10:34:41 +00:00
PhysicalMemoryAddress current_address = cs.base_address;
2019-09-11 04:59:41 +00:00
const PhysicalMemoryAddress increment = cs.channel_control.address_step_reverse ? static_cast<u32>(-4) : UINT32_C(4);
switch (cs.channel_control.sync_mode)
{
case SyncMode::Manual:
{
const u32 word_count = cs.block_control.manual.GetWordCount();
2019-09-30 10:01:41 +00:00
Log_DebugPrintf("DMA%u: Copying %u words %s 0x%08X", static_cast<u32>(channel), word_count,
copy_to_device ? "from" : "to", current_address);
if (copy_to_device)
TransferMemoryToDevice(channel, current_address, increment, word_count);
else
TransferDeviceToMemory(channel, current_address, increment, word_count);
2019-09-11 04:59:41 +00:00
}
break;
case SyncMode::LinkedList:
{
if (!copy_to_device)
{
Panic("Linked list not implemented for DMA reads");
}
else
{
2019-09-30 10:01:41 +00:00
Log_DebugPrintf("DMA%u: Copying linked list starting at 0x%08X to device", static_cast<u32>(channel),
current_address);
2019-09-11 04:59:41 +00:00
for (;;)
{
u32 header;
2019-11-11 10:34:41 +00:00
m_bus->DispatchAccess<MemoryAccessType::Read, MemoryAccessSize::Word>(current_address & ADDRESS_MASK, header);
2019-09-11 04:59:41 +00:00
const u32 word_count = header >> 24;
2019-11-11 10:34:41 +00:00
const u32 next_address = header & UINT32_C(0x00FFFFFF);
Log_TracePrintf(" .. linked list entry at 0x%08X size=%u(%u words) next=0x%08X", current_address,
2019-09-11 04:59:41 +00:00
word_count * UINT32_C(4), word_count, next_address);
if (word_count > 0)
TransferMemoryToDevice(channel, (current_address + sizeof(header)) & ADDRESS_MASK, 4, word_count);
// Self-referencing DMA loops.. not sure how these are happening?
if (current_address == next_address)
{
Log_ErrorPrintf("HACK: Aborting self-referencing DMA loop @ 0x%08X. Something went wrong to generate this.",
current_address);
break;
}
2019-09-11 04:59:41 +00:00
2019-11-11 10:34:41 +00:00
current_address = next_address;
if (current_address & UINT32_C(0x800000))
2019-09-11 04:59:41 +00:00
break;
}
}
2019-11-11 10:34:41 +00:00
cs.base_address = current_address;
}
break;
case SyncMode::Request:
2019-09-12 15:09:44 +00:00
{
Log_DebugPrintf("DMA%u: Copying %u blocks of size %u (%u total words) %s 0x%08X", static_cast<u32>(channel),
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);
const u32 block_size = cs.block_control.request.GetBlockSize();
u32 blocks_remaining = cs.block_control.request.block_count;
2019-09-12 15:09:44 +00:00
if (copy_to_device)
{
do
{
blocks_remaining--;
2019-11-11 10:34:41 +00:00
TransferMemoryToDevice(channel, current_address & ADDRESS_MASK, increment, block_size);
current_address = (current_address + (increment * block_size));
} while (cs.request && blocks_remaining > 0);
2019-09-12 15:09:44 +00:00
}
else
{
do
{
blocks_remaining--;
2019-11-11 10:34:41 +00:00
TransferDeviceToMemory(channel, current_address & ADDRESS_MASK, increment, block_size);
current_address = (current_address + (increment * block_size));
} while (cs.request && blocks_remaining > 0);
2019-09-12 15:09:44 +00:00
}
2019-11-11 10:34:41 +00:00
cs.base_address = current_address & BASE_ADDRESS_MASK;
cs.block_control.request.block_count = blocks_remaining;
// finish transfer later if the request was cleared
if (blocks_remaining > 0)
return;
2019-09-12 15:09:44 +00:00
}
break;
default:
Panic("Unimplemented sync mode");
break;
}
// start/busy bit is cleared on end of transfer
cs.channel_control.enable_busy = false;
2019-09-24 09:43:10 +00:00
if (m_DICR.IsIRQEnabled(channel))
{
Log_DebugPrintf("Set DMA interrupt for channel %u", static_cast<u32>(channel));
2019-09-24 09:43:10 +00:00
m_DICR.SetIRQFlag(channel);
UpdateIRQ();
2019-09-24 09:43:10 +00:00
}
}
void DMA::TransferMemoryToDevice(Channel channel, u32 address, u32 increment, u32 word_count)
{
// Read from memory. Wrap-around?
if (m_transfer_buffer.size() < word_count)
m_transfer_buffer.resize(word_count);
if (increment > 0 && ((address + (increment * word_count)) & ADDRESS_MASK) > address)
{
m_bus->ReadWords(address, m_transfer_buffer.data(), word_count);
}
else
{
for (u32 i = 0; i < word_count; i++)
{
m_bus->DispatchAccess<MemoryAccessType::Read, MemoryAccessSize::Word>(address, m_transfer_buffer[i]);
address = (address + increment) & ADDRESS_MASK;
}
}
2019-09-11 04:59:41 +00:00
switch (channel)
{
2019-09-11 04:59:41 +00:00
case Channel::GPU:
m_gpu->DMAWrite(m_transfer_buffer.data(), word_count);
break;
2019-09-21 15:12:16 +00:00
case Channel::SPU:
m_spu->DMAWrite(m_transfer_buffer.data(), word_count);
break;
2019-09-29 02:51:34 +00:00
case Channel::MDECin:
m_mdec->DMAWrite(m_transfer_buffer.data(), word_count);
break;
case Channel::CDROM:
case Channel::MDECout:
case Channel::PIO:
default:
Panic("Unhandled DMA channel for device write");
break;
}
}
void DMA::TransferDeviceToMemory(Channel channel, u32 address, u32 increment, u32 word_count)
{
if (m_transfer_buffer.size() < word_count)
m_transfer_buffer.resize(word_count);
// Read from device.
2019-09-11 04:59:41 +00:00
switch (channel)
{
case Channel::OTC:
{
// clear ordering table
// this always goes in reverse, so we can generate values in reverse order and write it forwards
if (((address - (4 * word_count)) & ADDRESS_MASK) < address)
{
const u32 end_address = (address - (4 * (word_count - 1))) & ADDRESS_MASK;
u32 value = end_address;
m_transfer_buffer[0] = UINT32_C(0xFFFFFF);
for (u32 i = 1; i < word_count; i++)
{
m_transfer_buffer[i] = value;
value = (value + 4) & ADDRESS_MASK;
}
m_bus->WriteWords(end_address, m_transfer_buffer.data(), word_count);
}
else
{
for (u32 i = 0; i < word_count; i++)
{
u32 value = (i == word_count - 1) ? UINT32_C(0xFFFFFFF) : ((address - 4) & ADDRESS_MASK);
m_bus->DispatchAccess<MemoryAccessType::Write, MemoryAccessSize::Word>(address, value);
address = (address - 4) & ADDRESS_MASK;
}
}
2019-09-11 04:59:41 +00:00
return;
}
break;
2019-09-11 04:59:41 +00:00
case Channel::GPU:
m_gpu->DMARead(m_transfer_buffer.data(), word_count);
break;
case Channel::CDROM:
m_cdrom->DMARead(m_transfer_buffer.data(), word_count);
break;
case Channel::SPU:
m_spu->DMARead(m_transfer_buffer.data(), word_count);
2019-09-29 02:51:34 +00:00
break;
2019-09-11 04:59:41 +00:00
case Channel::MDECout:
m_mdec->DMARead(m_transfer_buffer.data(), word_count);
break;
case Channel::MDECin:
2019-09-11 04:59:41 +00:00
case Channel::PIO:
default:
Panic("Unhandled DMA channel for device read");
std::fill_n(m_transfer_buffer.begin(), word_count, UINT32_C(0xFFFFFFFF));
2019-09-11 04:59:41 +00:00
break;
}
if (increment > 0 && ((address + (increment * word_count)) & ADDRESS_MASK) > address)
{
m_bus->WriteWords(address, m_transfer_buffer.data(), word_count);
}
else
{
for (u32 i = 0; i < word_count; i++)
{
m_bus->DispatchAccess<MemoryAccessType::Write, MemoryAccessSize::Word>(address, m_transfer_buffer[i]);
address = (address + increment) & ADDRESS_MASK;
}
}
}