DMA: Refactoring, support split block transfers
This commit is contained in:
parent
2d9d999713
commit
88ec178380
|
@ -2,6 +2,7 @@
|
||||||
#include "YBaseLib/Log.h"
|
#include "YBaseLib/Log.h"
|
||||||
#include "common/cd_image.h"
|
#include "common/cd_image.h"
|
||||||
#include "common/state_wrapper.h"
|
#include "common/state_wrapper.h"
|
||||||
|
#include "dma.h"
|
||||||
#include "interrupt_controller.h"
|
#include "interrupt_controller.h"
|
||||||
#include "system.h"
|
#include "system.h"
|
||||||
Log_SetChannel(CDROM);
|
Log_SetChannel(CDROM);
|
||||||
|
@ -394,6 +395,8 @@ void CDROM::UpdateStatusRegister()
|
||||||
m_status.RSLRRDY = !m_response_fifo.IsEmpty();
|
m_status.RSLRRDY = !m_response_fifo.IsEmpty();
|
||||||
m_status.DRQSTS = !m_data_fifo.IsEmpty();
|
m_status.DRQSTS = !m_data_fifo.IsEmpty();
|
||||||
m_status.BUSYSTS = m_command_state == CommandState::WaitForExecute;
|
m_status.BUSYSTS = m_command_state == CommandState::WaitForExecute;
|
||||||
|
|
||||||
|
m_dma->SetRequest(DMA::Channel::CDROM, m_status.DRQSTS);
|
||||||
}
|
}
|
||||||
|
|
||||||
u32 CDROM::GetAckDelayForCommand() const
|
u32 CDROM::GetAckDelayForCommand() const
|
||||||
|
|
142
src/core/dma.cpp
142
src/core/dma.cpp
|
@ -30,7 +30,7 @@ bool DMA::Initialize(System* system, Bus* bus, InterruptController* interrupt_co
|
||||||
void DMA::Reset()
|
void DMA::Reset()
|
||||||
{
|
{
|
||||||
m_transfer_ticks = 0;
|
m_transfer_ticks = 0;
|
||||||
m_transfer_pending = false;
|
m_transfer_in_progress = false;
|
||||||
m_state = {};
|
m_state = {};
|
||||||
m_DPCR.bits = 0x07654321;
|
m_DPCR.bits = 0x07654321;
|
||||||
m_DICR.bits = 0;
|
m_DICR.bits = 0;
|
||||||
|
@ -99,6 +99,7 @@ void DMA::WriteRegister(u32 offset, u32 value)
|
||||||
{
|
{
|
||||||
Log_TracePrintf("DMA channel %u block control <- 0x%08X", channel_index, value);
|
Log_TracePrintf("DMA channel %u block control <- 0x%08X", channel_index, value);
|
||||||
state.block_control.bits = value;
|
state.block_control.bits = value;
|
||||||
|
Transfer();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -107,9 +108,7 @@ 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 %u channel control <- 0x%08X", channel_index, state.channel_control.bits);
|
||||||
if (CanRunChannel(static_cast<Channel>(channel_index)))
|
Transfer();
|
||||||
UpdateTransferPending();
|
|
||||||
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -125,7 +124,7 @@ void DMA::WriteRegister(u32 offset, u32 value)
|
||||||
{
|
{
|
||||||
Log_TracePrintf("DPCR <- 0x%08X", value);
|
Log_TracePrintf("DPCR <- 0x%08X", value);
|
||||||
m_DPCR.bits = value;
|
m_DPCR.bits = value;
|
||||||
UpdateTransferPending();
|
Transfer();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -153,65 +152,70 @@ void DMA::SetRequest(Channel channel, bool request)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
cs.request = request;
|
cs.request = request;
|
||||||
UpdateTransferPending();
|
if (request)
|
||||||
|
Transfer();
|
||||||
}
|
}
|
||||||
|
|
||||||
void DMA::Execute(TickCount ticks)
|
bool DMA::CanTransferChannel(Channel channel) const
|
||||||
{
|
|
||||||
if (!m_transfer_pending)
|
|
||||||
return;
|
|
||||||
|
|
||||||
m_transfer_ticks -= ticks;
|
|
||||||
if (m_transfer_ticks <= 0)
|
|
||||||
{
|
|
||||||
m_transfer_pending = false;
|
|
||||||
|
|
||||||
for (u32 i = 0; i < NUM_CHANNELS; i++)
|
|
||||||
{
|
|
||||||
const Channel channel = static_cast<Channel>(i);
|
|
||||||
if (CanRunChannel(channel))
|
|
||||||
{
|
|
||||||
RunDMA(channel);
|
|
||||||
m_transfer_pending |= CanRunChannel(channel);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (m_transfer_pending)
|
|
||||||
{
|
|
||||||
m_transfer_ticks += TRANSFER_TICKS;
|
|
||||||
m_system->SetDowncount(m_transfer_ticks);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
m_system->SetDowncount(m_transfer_ticks);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
bool DMA::CanRunChannel(Channel channel) const
|
|
||||||
{
|
{
|
||||||
if (!m_DPCR.GetMasterEnable(channel))
|
if (!m_DPCR.GetMasterEnable(channel))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
const ChannelState& cs = m_state[static_cast<u32>(channel)];
|
const ChannelState& cs = m_state[static_cast<u32>(channel)];
|
||||||
if (cs.channel_control.start_trigger)
|
if (!cs.channel_control.enable_busy)
|
||||||
return true;
|
return false;
|
||||||
|
|
||||||
return (cs.channel_control.enable_busy && cs.request);
|
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;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool DMA::CanRunAnyChannels() const
|
bool DMA::CanRunAnyChannels() const
|
||||||
{
|
{
|
||||||
for (u32 i = 0; i < NUM_CHANNELS; i++)
|
for (u32 i = 0; i < NUM_CHANNELS; i++)
|
||||||
{
|
{
|
||||||
if (CanRunChannel(static_cast<Channel>(i)))
|
if (CanTransferChannel(static_cast<Channel>(i)))
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
void DMA::RunDMA(Channel channel)
|
void DMA::Transfer()
|
||||||
|
{
|
||||||
|
if (m_transfer_in_progress)
|
||||||
|
return;
|
||||||
|
|
||||||
|
// prevent recursive calls
|
||||||
|
m_transfer_in_progress = true;
|
||||||
|
|
||||||
|
// keep going until all transfers are done. one channel can start others (e.g. MDEC)
|
||||||
|
for (;;)
|
||||||
|
{
|
||||||
|
bool any_channels_active = false;
|
||||||
|
|
||||||
|
for (u32 i = 0; i < NUM_CHANNELS; i++)
|
||||||
|
{
|
||||||
|
const Channel channel = static_cast<Channel>(i);
|
||||||
|
if (CanTransferChannel(channel))
|
||||||
|
{
|
||||||
|
TransferChannel(channel);
|
||||||
|
any_channels_active = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!any_channels_active)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
m_transfer_in_progress = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
void DMA::TransferChannel(Channel channel)
|
||||||
{
|
{
|
||||||
ChannelState& cs = m_state[static_cast<u32>(channel)];
|
ChannelState& cs = m_state[static_cast<u32>(channel)];
|
||||||
const bool copy_to_device = cs.channel_control.copy_to_device;
|
const bool copy_to_device = cs.channel_control.copy_to_device;
|
||||||
|
@ -305,13 +309,19 @@ void DMA::RunDMA(Channel channel)
|
||||||
|
|
||||||
case SyncMode::Request:
|
case SyncMode::Request:
|
||||||
{
|
{
|
||||||
const u32 block_size = cs.block_control.request.GetBlockSize();
|
Log_DebugPrintf("DMA%u: Copying %u blocks of size %u %s 0x%08X", static_cast<u32>(channel),
|
||||||
const u32 block_count = cs.block_control.request.GetBlockCount();
|
cs.block_control.request.GetBlockCount(), cs.block_control.request.GetBlockSize(),
|
||||||
Log_DebugPrintf("DMA%u: Copying %u blocks of size %u %s 0x%08X", static_cast<u32>(channel), block_count,
|
copy_to_device ? "from" : "to", current_address);
|
||||||
block_size, copy_to_device ? "from" : "to", current_address);
|
|
||||||
|
u32 blocks_remaining = cs.block_control.request.block_count;
|
||||||
|
|
||||||
if (copy_to_device)
|
if (copy_to_device)
|
||||||
{
|
{
|
||||||
u32 words_remaining = block_size * block_count;
|
do
|
||||||
|
{
|
||||||
|
blocks_remaining--;
|
||||||
|
|
||||||
|
u32 words_remaining = cs.block_control.request.block_size;
|
||||||
do
|
do
|
||||||
{
|
{
|
||||||
words_remaining--;
|
words_remaining--;
|
||||||
|
@ -322,10 +332,15 @@ void DMA::RunDMA(Channel channel)
|
||||||
|
|
||||||
current_address = (current_address + increment) & ADDRESS_MASK;
|
current_address = (current_address + increment) & ADDRESS_MASK;
|
||||||
} while (words_remaining > 0);
|
} while (words_remaining > 0);
|
||||||
|
} while (cs.request && blocks_remaining > 0);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
u32 words_remaining = block_size * block_count;
|
do
|
||||||
|
{
|
||||||
|
blocks_remaining--;
|
||||||
|
|
||||||
|
u32 words_remaining = cs.block_control.request.block_size;
|
||||||
do
|
do
|
||||||
{
|
{
|
||||||
words_remaining--;
|
words_remaining--;
|
||||||
|
@ -335,7 +350,15 @@ void DMA::RunDMA(Channel channel)
|
||||||
|
|
||||||
current_address = (current_address + increment) & ADDRESS_MASK;
|
current_address = (current_address + increment) & ADDRESS_MASK;
|
||||||
} while (words_remaining > 0);
|
} while (words_remaining > 0);
|
||||||
|
} while (cs.request && blocks_remaining > 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
cs.base_address = current_address;
|
||||||
|
cs.block_control.request.block_count = blocks_remaining;
|
||||||
|
|
||||||
|
// finish transfer later if the request was cleared
|
||||||
|
if (blocks_remaining > 0)
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
@ -412,22 +435,3 @@ void DMA::DMAWrite(Channel channel, u32 value, PhysicalMemoryAddress src_address
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void DMA::UpdateTransferPending()
|
|
||||||
{
|
|
||||||
if (CanRunAnyChannels())
|
|
||||||
{
|
|
||||||
if (m_transfer_pending)
|
|
||||||
return;
|
|
||||||
|
|
||||||
m_system->Synchronize();
|
|
||||||
m_transfer_pending = true;
|
|
||||||
m_transfer_ticks = TRANSFER_TICKS;
|
|
||||||
m_system->SetDowncount(m_transfer_ticks);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
m_transfer_pending = false;
|
|
||||||
m_transfer_ticks = 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
|
@ -35,8 +35,8 @@ public:
|
||||||
DMA();
|
DMA();
|
||||||
~DMA();
|
~DMA();
|
||||||
|
|
||||||
bool Initialize(System* system, Bus* bus, InterruptController* interrupt_controller, GPU* gpu, CDROM* cdrom,
|
bool Initialize(System* system, Bus* bus, InterruptController* interrupt_controller, GPU* gpu, CDROM* cdrom, SPU* spu,
|
||||||
SPU* spu, MDEC* mdec);
|
MDEC* mdec);
|
||||||
void Reset();
|
void Reset();
|
||||||
bool DoState(StateWrapper& sw);
|
bool DoState(StateWrapper& sw);
|
||||||
|
|
||||||
|
@ -45,8 +45,6 @@ public:
|
||||||
|
|
||||||
void SetRequest(Channel channel, bool request);
|
void SetRequest(Channel channel, bool request);
|
||||||
|
|
||||||
void Execute(TickCount ticks);
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
static constexpr PhysicalMemoryAddress ADDRESS_MASK = UINT32_C(0x00FFFFFF);
|
static constexpr PhysicalMemoryAddress ADDRESS_MASK = UINT32_C(0x00FFFFFF);
|
||||||
static constexpr u32 TRANSFER_TICKS = 10;
|
static constexpr u32 TRANSFER_TICKS = 10;
|
||||||
|
@ -60,10 +58,11 @@ private:
|
||||||
};
|
};
|
||||||
|
|
||||||
// is everything enabled for a channel to operate?
|
// is everything enabled for a channel to operate?
|
||||||
bool CanRunChannel(Channel channel) const;
|
bool CanTransferChannel(Channel channel) const;
|
||||||
bool CanRunAnyChannels() const;
|
bool CanRunAnyChannels() const;
|
||||||
|
|
||||||
void RunDMA(Channel channel);
|
void Transfer();
|
||||||
|
void TransferChannel(Channel channel);
|
||||||
|
|
||||||
// from device -> memory
|
// from device -> memory
|
||||||
u32 DMARead(Channel channel, PhysicalMemoryAddress dst_address, u32 remaining_words);
|
u32 DMARead(Channel channel, PhysicalMemoryAddress dst_address, u32 remaining_words);
|
||||||
|
@ -71,8 +70,6 @@ private:
|
||||||
// from memory -> device
|
// from memory -> device
|
||||||
void DMAWrite(Channel channel, u32 value, PhysicalMemoryAddress src_address, u32 remaining_words);
|
void DMAWrite(Channel channel, u32 value, PhysicalMemoryAddress src_address, u32 remaining_words);
|
||||||
|
|
||||||
void UpdateTransferPending();
|
|
||||||
|
|
||||||
System* m_system = nullptr;
|
System* m_system = nullptr;
|
||||||
Bus* m_bus = nullptr;
|
Bus* m_bus = nullptr;
|
||||||
InterruptController* m_interrupt_controller = nullptr;
|
InterruptController* m_interrupt_controller = nullptr;
|
||||||
|
@ -82,7 +79,7 @@ private:
|
||||||
MDEC* m_mdec = nullptr;
|
MDEC* m_mdec = nullptr;
|
||||||
|
|
||||||
TickCount m_transfer_ticks = 0;
|
TickCount m_transfer_ticks = 0;
|
||||||
bool m_transfer_pending = false;
|
bool m_transfer_in_progress = false;
|
||||||
|
|
||||||
struct ChannelState
|
struct ChannelState
|
||||||
{
|
{
|
||||||
|
|
|
@ -48,7 +48,6 @@ u32 MDEC::ReadRegister(u32 offset)
|
||||||
switch (offset)
|
switch (offset)
|
||||||
{
|
{
|
||||||
case 0:
|
case 0:
|
||||||
UpdateStatusRegister();
|
|
||||||
return ReadDataRegister();
|
return ReadDataRegister();
|
||||||
|
|
||||||
case 4:
|
case 4:
|
||||||
|
@ -85,6 +84,7 @@ void MDEC::WriteRegister(u32 offset, u32 value)
|
||||||
|
|
||||||
m_enable_dma_in = cr.enable_dma_in;
|
m_enable_dma_in = cr.enable_dma_in;
|
||||||
m_enable_dma_out = cr.enable_dma_out;
|
m_enable_dma_out = cr.enable_dma_out;
|
||||||
|
UpdateStatusRegister();
|
||||||
UpdateDMARequest();
|
UpdateDMARequest();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -114,6 +114,7 @@ void MDEC::SoftReset()
|
||||||
m_enable_dma_out = false;
|
m_enable_dma_out = false;
|
||||||
m_data_in_fifo.Clear();
|
m_data_in_fifo.Clear();
|
||||||
m_data_out_fifo.Clear();
|
m_data_out_fifo.Clear();
|
||||||
|
UpdateStatusRegister();
|
||||||
UpdateDMARequest();
|
UpdateDMARequest();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -122,7 +123,7 @@ void MDEC::UpdateStatusRegister()
|
||||||
m_status.data_out_fifo_empty = m_data_out_fifo.IsEmpty();
|
m_status.data_out_fifo_empty = m_data_out_fifo.IsEmpty();
|
||||||
m_status.data_in_fifo_full = m_data_in_fifo.IsFull();
|
m_status.data_in_fifo_full = m_data_in_fifo.IsFull();
|
||||||
|
|
||||||
m_status.command_busy = m_command != Command::None;
|
m_status.command_busy = false;
|
||||||
m_status.parameter_words_remaining = Truncate16(m_remaining_words - 1);
|
m_status.parameter_words_remaining = Truncate16(m_remaining_words - 1);
|
||||||
m_status.current_block = (m_current_block + 4) % NUM_BLOCKS;
|
m_status.current_block = (m_current_block + 4) % NUM_BLOCKS;
|
||||||
}
|
}
|
||||||
|
@ -130,7 +131,7 @@ void MDEC::UpdateStatusRegister()
|
||||||
void MDEC::UpdateDMARequest()
|
void MDEC::UpdateDMARequest()
|
||||||
{
|
{
|
||||||
// we always want data in if it's enabled
|
// we always want data in if it's enabled
|
||||||
const bool data_in_request = m_enable_dma_in && !m_data_in_fifo.IsFull();
|
const bool data_in_request = m_enable_dma_in && m_data_in_fifo.GetSpace() >= (32 * 2) && !m_data_out_fifo.IsFull();
|
||||||
m_status.data_in_request = data_in_request;
|
m_status.data_in_request = data_in_request;
|
||||||
m_dma->SetRequest(DMA::Channel::MDECin, data_in_request);
|
m_dma->SetRequest(DMA::Channel::MDECin, data_in_request);
|
||||||
|
|
||||||
|
@ -142,15 +143,24 @@ void MDEC::UpdateDMARequest()
|
||||||
|
|
||||||
u32 MDEC::ReadDataRegister()
|
u32 MDEC::ReadDataRegister()
|
||||||
{
|
{
|
||||||
|
if (m_data_out_fifo.IsEmpty())
|
||||||
|
{
|
||||||
|
Execute();
|
||||||
|
|
||||||
if (m_data_out_fifo.IsEmpty())
|
if (m_data_out_fifo.IsEmpty())
|
||||||
{
|
{
|
||||||
Log_WarningPrintf("MDEC data out FIFO empty on read");
|
Log_WarningPrintf("MDEC data out FIFO empty on read");
|
||||||
return UINT32_C(0xFFFFFFFF);
|
return UINT32_C(0xFFFFFFFF);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const u32 value = m_data_out_fifo.Pop();
|
const u32 value = m_data_out_fifo.Pop();
|
||||||
|
if (m_data_out_fifo.IsEmpty())
|
||||||
|
{
|
||||||
UpdateStatusRegister();
|
UpdateStatusRegister();
|
||||||
UpdateDMARequest();
|
UpdateDMARequest();
|
||||||
|
}
|
||||||
|
|
||||||
return value;
|
return value;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -197,28 +207,52 @@ void MDEC::WriteCommandRegister(u32 value)
|
||||||
m_data_in_fifo.Push(Truncate16(value));
|
m_data_in_fifo.Push(Truncate16(value));
|
||||||
m_data_in_fifo.Push(Truncate16(value >> 16));
|
m_data_in_fifo.Push(Truncate16(value >> 16));
|
||||||
m_remaining_words--;
|
m_remaining_words--;
|
||||||
UpdateDMARequest();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Execute();
|
||||||
|
}
|
||||||
|
|
||||||
|
void MDEC::Execute()
|
||||||
|
{
|
||||||
switch (m_command)
|
switch (m_command)
|
||||||
{
|
{
|
||||||
case Command::DecodeMacroblock:
|
case Command::DecodeMacroblock:
|
||||||
{
|
{
|
||||||
if (!HandleDecodeMacroblockCommand())
|
if (!HandleDecodeMacroblockCommand())
|
||||||
|
{
|
||||||
|
UpdateStatusRegister();
|
||||||
|
UpdateDMARequest();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case Command::SetIqTab:
|
case Command::SetIqTab:
|
||||||
{
|
{
|
||||||
if (!HandleSetQuantTableCommand())
|
if (!HandleSetQuantTableCommand())
|
||||||
|
{
|
||||||
|
UpdateStatusRegister();
|
||||||
|
UpdateDMARequest();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case Command::SetScale:
|
case Command::SetScale:
|
||||||
{
|
{
|
||||||
if (!HandleSetScaleCommand())
|
if (!HandleSetScaleCommand())
|
||||||
|
{
|
||||||
|
UpdateStatusRegister();
|
||||||
|
UpdateDMARequest();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
{
|
||||||
|
UpdateStatusRegister();
|
||||||
|
UpdateDMARequest();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
@ -229,6 +263,7 @@ void MDEC::WriteCommandRegister(u32 value)
|
||||||
m_current_block = 0;
|
m_current_block = 0;
|
||||||
m_current_coefficient = 64;
|
m_current_coefficient = 64;
|
||||||
m_current_q_scale = 0;
|
m_current_q_scale = 0;
|
||||||
|
UpdateStatusRegister();
|
||||||
UpdateDMARequest();
|
UpdateDMARequest();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -242,7 +277,7 @@ bool MDEC::HandleDecodeMacroblockCommand()
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
return m_remaining_words == 0;
|
return m_data_in_fifo.IsEmpty() && m_remaining_words == 0;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
@ -252,12 +287,24 @@ bool MDEC::HandleDecodeMacroblockCommand()
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
return m_remaining_words == 0;
|
return m_data_in_fifo.IsEmpty() && m_remaining_words == 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool MDEC::DecodeMonoMacroblock()
|
bool MDEC::DecodeMonoMacroblock()
|
||||||
{
|
{
|
||||||
|
// sufficient space in output?
|
||||||
|
if (m_status.data_output_depth == DataOutputDepth_4Bit)
|
||||||
|
{
|
||||||
|
if (m_data_out_fifo.GetSpace() < (64 / 8))
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (m_data_out_fifo.GetSpace() < (64 / 4))
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
if (!rl_decode_block(m_blocks[0].data(), m_iq_y.data()))
|
if (!rl_decode_block(m_blocks[0].data(), m_iq_y.data()))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
|
@ -310,7 +357,17 @@ bool MDEC::DecodeMonoMacroblock()
|
||||||
|
|
||||||
bool MDEC::DecodeColoredMacroblock()
|
bool MDEC::DecodeColoredMacroblock()
|
||||||
{
|
{
|
||||||
std::array<u32, 256> out_rgb;
|
// sufficient space in output?
|
||||||
|
if (m_status.data_output_depth == DataOutputDepth_24Bit)
|
||||||
|
{
|
||||||
|
if (m_data_out_fifo.GetSpace() < (256 - (256 / 4)))
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (m_data_out_fifo.GetSpace() < (256 / 2))
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
for (; m_current_block < NUM_BLOCKS; m_current_block++)
|
for (; m_current_block < NUM_BLOCKS; m_current_block++)
|
||||||
{
|
{
|
||||||
|
@ -322,7 +379,9 @@ bool MDEC::DecodeColoredMacroblock()
|
||||||
|
|
||||||
// done decoding
|
// done decoding
|
||||||
m_current_block = 0;
|
m_current_block = 0;
|
||||||
|
Log_DebugPrintf("Decoded colored macroblock");
|
||||||
|
|
||||||
|
std::array<u32, 256> out_rgb;
|
||||||
yuv_to_rgb(0, 0, m_blocks[0], m_blocks[1], m_blocks[2], out_rgb);
|
yuv_to_rgb(0, 0, m_blocks[0], m_blocks[1], m_blocks[2], out_rgb);
|
||||||
yuv_to_rgb(8, 0, m_blocks[0], m_blocks[1], m_blocks[3], out_rgb);
|
yuv_to_rgb(8, 0, m_blocks[0], m_blocks[1], m_blocks[3], out_rgb);
|
||||||
yuv_to_rgb(0, 8, m_blocks[0], m_blocks[1], m_blocks[4], out_rgb);
|
yuv_to_rgb(0, 8, m_blocks[0], m_blocks[1], m_blocks[4], out_rgb);
|
||||||
|
@ -612,8 +671,8 @@ void MDEC::DrawDebugWindow()
|
||||||
if (ImGui::CollapsingHeader("Status", ImGuiTreeNodeFlags_DefaultOpen))
|
if (ImGui::CollapsingHeader("Status", ImGuiTreeNodeFlags_DefaultOpen))
|
||||||
{
|
{
|
||||||
ImGui::Text("Data-Out FIFO Empty: %s", m_status.data_out_fifo_empty ? "Yes" : "No");
|
ImGui::Text("Data-Out FIFO Empty: %s", m_status.data_out_fifo_empty ? "Yes" : "No");
|
||||||
ImGui::Text("Data-In FIFO Empty: %s", m_status.data_in_fifo_full ? "Yes" : "No");
|
ImGui::Text("Data-In FIFO Full: %s", m_status.data_in_fifo_full ? "Yes" : "No");
|
||||||
ImGui::Text("Command Busy FIFO Empty: %s", m_status.command_busy ? "Yes" : "No");
|
ImGui::Text("Command Busy: %s", m_status.command_busy ? "Yes" : "No");
|
||||||
ImGui::Text("Data-In Request: %s", m_status.data_in_request ? "Yes" : "No");
|
ImGui::Text("Data-In Request: %s", m_status.data_in_request ? "Yes" : "No");
|
||||||
ImGui::Text("Output Depth: %s", output_depths[static_cast<u8>(m_status.data_output_depth.GetValue())]);
|
ImGui::Text("Output Depth: %s", output_depths[static_cast<u8>(m_status.data_output_depth.GetValue())]);
|
||||||
ImGui::Text("Output Signed: %s", m_status.data_output_signed ? "Yes" : "No");
|
ImGui::Text("Output Signed: %s", m_status.data_output_signed ? "Yes" : "No");
|
||||||
|
|
|
@ -91,6 +91,7 @@ private:
|
||||||
|
|
||||||
u32 ReadDataRegister();
|
u32 ReadDataRegister();
|
||||||
void WriteCommandRegister(u32 value);
|
void WriteCommandRegister(u32 value);
|
||||||
|
void Execute();
|
||||||
|
|
||||||
bool HandleDecodeMacroblockCommand();
|
bool HandleDecodeMacroblockCommand();
|
||||||
bool HandleSetQuantTableCommand();
|
bool HandleSetQuantTableCommand();
|
||||||
|
@ -114,8 +115,8 @@ private:
|
||||||
bool m_enable_dma_out = false;
|
bool m_enable_dma_out = false;
|
||||||
|
|
||||||
// Even though the DMA is in words, we access the FIFO as halfwords.
|
// Even though the DMA is in words, we access the FIFO as halfwords.
|
||||||
InlineFIFOQueue<u16, DATA_IN_FIFO_SIZE> m_data_in_fifo;
|
InlineFIFOQueue<u16, DATA_IN_FIFO_SIZE / sizeof(u16)> m_data_in_fifo;
|
||||||
InlineFIFOQueue<u32, DATA_OUT_FIFO_SIZE> m_data_out_fifo;
|
InlineFIFOQueue<u32, DATA_OUT_FIFO_SIZE / sizeof(u32)> m_data_out_fifo;
|
||||||
Command m_command = Command::None;
|
Command m_command = Command::None;
|
||||||
u32 m_remaining_words = 0;
|
u32 m_remaining_words = 0;
|
||||||
|
|
||||||
|
|
|
@ -288,7 +288,6 @@ void System::Synchronize()
|
||||||
m_timers->Execute(pending_ticks);
|
m_timers->Execute(pending_ticks);
|
||||||
m_cdrom->Execute(pending_ticks);
|
m_cdrom->Execute(pending_ticks);
|
||||||
m_pad->Execute(pending_ticks);
|
m_pad->Execute(pending_ticks);
|
||||||
m_dma->Execute(pending_ticks);
|
|
||||||
m_spu->Execute(pending_ticks);
|
m_spu->Execute(pending_ticks);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -96,7 +96,7 @@ int main(int argc, char* argv[])
|
||||||
#else
|
#else
|
||||||
g_pLog->SetConsoleOutputParams(true, nullptr, LOGLEVEL_DEBUG);
|
g_pLog->SetConsoleOutputParams(true, nullptr, LOGLEVEL_DEBUG);
|
||||||
// g_pLog->SetConsoleOutputParams(true, "GPU GPU_HW_OpenGL SPU Pad DigitalController", LOGLEVEL_DEBUG);
|
// g_pLog->SetConsoleOutputParams(true, "GPU GPU_HW_OpenGL SPU Pad DigitalController", LOGLEVEL_DEBUG);
|
||||||
g_pLog->SetConsoleOutputParams(true, "GPU GPU_HW_OpenGL Pad DigitalController InterruptController", LOGLEVEL_DEBUG);
|
g_pLog->SetConsoleOutputParams(true, "Pad DigitalController InterruptController", LOGLEVEL_DEBUG);
|
||||||
// g_pLog->SetFilterLevel(LOGLEVEL_TRACE);
|
// g_pLog->SetFilterLevel(LOGLEVEL_TRACE);
|
||||||
g_pLog->SetFilterLevel(LOGLEVEL_DEBUG);
|
g_pLog->SetFilterLevel(LOGLEVEL_DEBUG);
|
||||||
// g_pLog->SetFilterLevel(LOGLEVEL_DEV);
|
// g_pLog->SetFilterLevel(LOGLEVEL_DEV);
|
||||||
|
|
Loading…
Reference in New Issue