2015-05-24 06:55:12 +02:00
|
|
|
// Copyright 2008 Dolphin Emulator Project
|
2021-07-05 03:22:19 +02:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
2010-06-09 01:37:08 +00:00
|
|
|
|
2021-12-09 18:22:16 -08:00
|
|
|
#include "VideoCommon/CommandProcessor.h"
|
|
|
|
|
2015-05-27 03:08:48 -04:00
|
|
|
#include <atomic>
|
2016-01-17 16:54:31 -05:00
|
|
|
#include <cstring>
|
2021-04-22 20:57:56 -07:00
|
|
|
#include <fmt/format.h>
|
2015-05-27 03:08:48 -04:00
|
|
|
|
2016-01-17 16:54:31 -05:00
|
|
|
#include "Common/Assert.h"
|
2014-02-19 02:27:20 +01:00
|
|
|
#include "Common/ChunkFile.h"
|
2014-09-07 20:06:58 -05:00
|
|
|
#include "Common/CommonTypes.h"
|
2016-08-05 16:04:39 +02:00
|
|
|
#include "Common/Flag.h"
|
2016-01-17 16:54:31 -05:00
|
|
|
#include "Common/Logging/Log.h"
|
2014-02-18 12:18:47 +01:00
|
|
|
#include "Core/ConfigManager.h"
|
|
|
|
#include "Core/CoreTiming.h"
|
|
|
|
#include "Core/HW/GPFifo.h"
|
|
|
|
#include "Core/HW/MMIO.h"
|
|
|
|
#include "Core/HW/ProcessorInterface.h"
|
|
|
|
#include "VideoCommon/Fifo.h"
|
2010-06-09 01:37:08 +00:00
|
|
|
|
|
|
|
namespace CommandProcessor
|
|
|
|
{
|
2016-09-01 10:54:18 +00:00
|
|
|
static CoreTiming::EventType* et_UpdateInterrupts;
|
2010-06-09 01:37:08 +00:00
|
|
|
|
|
|
|
// TODO(ector): Warn on bbox read/write
|
|
|
|
|
|
|
|
// STATE_TO_SAVE
|
|
|
|
SCPFifoStruct fifo;
|
2014-07-08 15:58:25 +02:00
|
|
|
static UCPStatusReg m_CPStatusReg;
|
2016-06-24 10:43:46 +02:00
|
|
|
static UCPCtrlReg m_CPCtrlReg;
|
2014-07-08 15:58:25 +02:00
|
|
|
static UCPClearReg m_CPClearReg;
|
2010-06-09 01:37:08 +00:00
|
|
|
|
2014-07-08 15:58:25 +02:00
|
|
|
static u16 m_bboxleft;
|
|
|
|
static u16 m_bboxtop;
|
|
|
|
static u16 m_bboxright;
|
|
|
|
static u16 m_bboxbottom;
|
|
|
|
static u16 m_tokenReg;
|
2010-06-09 01:37:08 +00:00
|
|
|
|
2016-08-05 16:04:39 +02:00
|
|
|
static Common::Flag s_interrupt_set;
|
|
|
|
static Common::Flag s_interrupt_waiting;
|
2010-06-24 13:28:54 +00:00
|
|
|
|
2014-07-08 14:29:26 +02:00
|
|
|
static bool IsOnThread()
|
2011-02-08 10:37:47 +00:00
|
|
|
{
|
2016-06-24 10:43:46 +02:00
|
|
|
return SConfig::GetInstance().bCPUThread;
|
2011-02-08 10:37:47 +00:00
|
|
|
}
|
|
|
|
|
2016-04-10 02:06:09 +12:00
|
|
|
static void UpdateInterrupts_Wrapper(u64 userdata, s64 cyclesLate)
|
2011-02-08 10:37:47 +00:00
|
|
|
{
|
2016-06-24 10:43:46 +02:00
|
|
|
UpdateInterrupts(userdata);
|
2011-02-08 10:37:47 +00:00
|
|
|
}
|
|
|
|
|
2021-05-17 21:55:03 +02:00
|
|
|
void SCPFifoStruct::Init()
|
|
|
|
{
|
|
|
|
CPBase = 0;
|
|
|
|
CPEnd = 0;
|
|
|
|
CPHiWatermark = 0;
|
|
|
|
CPLoWatermark = 0;
|
|
|
|
CPReadWriteDistance = 0;
|
|
|
|
CPWritePointer = 0;
|
|
|
|
CPReadPointer = 0;
|
|
|
|
CPBreakpoint = 0;
|
|
|
|
SafeCPReadPointer = 0;
|
|
|
|
|
|
|
|
bFF_GPLinkEnable = 0;
|
|
|
|
bFF_GPReadEnable = 0;
|
|
|
|
bFF_BPEnable = 0;
|
|
|
|
bFF_BPInt = 0;
|
|
|
|
|
|
|
|
bFF_Breakpoint.store(0, std::memory_order_relaxed);
|
|
|
|
bFF_HiWatermark.store(0, std::memory_order_relaxed);
|
|
|
|
bFF_HiWatermarkInt.store(0, std::memory_order_relaxed);
|
|
|
|
bFF_LoWatermark.store(0, std::memory_order_relaxed);
|
|
|
|
bFF_LoWatermarkInt.store(0, std::memory_order_relaxed);
|
|
|
|
}
|
|
|
|
|
2017-11-12 17:20:59 +01:00
|
|
|
void SCPFifoStruct::DoState(PointerWrap& p)
|
|
|
|
{
|
|
|
|
p.Do(CPBase);
|
|
|
|
p.Do(CPEnd);
|
|
|
|
p.Do(CPHiWatermark);
|
|
|
|
p.Do(CPLoWatermark);
|
|
|
|
p.Do(CPReadWriteDistance);
|
|
|
|
p.Do(CPWritePointer);
|
|
|
|
p.Do(CPReadPointer);
|
|
|
|
p.Do(CPBreakpoint);
|
|
|
|
p.Do(SafeCPReadPointer);
|
|
|
|
|
|
|
|
p.Do(bFF_GPLinkEnable);
|
|
|
|
p.Do(bFF_GPReadEnable);
|
|
|
|
p.Do(bFF_BPEnable);
|
|
|
|
p.Do(bFF_BPInt);
|
|
|
|
p.Do(bFF_Breakpoint);
|
|
|
|
|
|
|
|
p.Do(bFF_LoWatermarkInt);
|
|
|
|
p.Do(bFF_HiWatermarkInt);
|
|
|
|
|
|
|
|
p.Do(bFF_LoWatermark);
|
|
|
|
p.Do(bFF_HiWatermark);
|
|
|
|
}
|
|
|
|
|
2016-06-24 10:43:46 +02:00
|
|
|
void DoState(PointerWrap& p)
|
2010-06-09 01:37:08 +00:00
|
|
|
{
|
2016-06-24 10:43:46 +02:00
|
|
|
p.DoPOD(m_CPStatusReg);
|
|
|
|
p.DoPOD(m_CPCtrlReg);
|
|
|
|
p.DoPOD(m_CPClearReg);
|
|
|
|
p.Do(m_bboxleft);
|
|
|
|
p.Do(m_bboxtop);
|
|
|
|
p.Do(m_bboxright);
|
|
|
|
p.Do(m_bboxbottom);
|
|
|
|
p.Do(m_tokenReg);
|
2017-11-12 17:20:59 +01:00
|
|
|
fifo.DoState(p);
|
2016-06-24 10:43:46 +02:00
|
|
|
|
|
|
|
p.Do(s_interrupt_set);
|
|
|
|
p.Do(s_interrupt_waiting);
|
2010-06-09 01:37:08 +00:00
|
|
|
}
|
|
|
|
|
2021-05-13 18:44:59 +02:00
|
|
|
static inline void WriteLow(std::atomic<u32>& reg, u16 lowbits)
|
2014-08-25 21:09:26 +02:00
|
|
|
{
|
2021-05-13 18:44:59 +02:00
|
|
|
reg.store((reg.load(std::memory_order_relaxed) & 0xFFFF0000) | lowbits,
|
|
|
|
std::memory_order_relaxed);
|
2014-08-25 21:09:26 +02:00
|
|
|
}
|
2021-05-13 18:44:59 +02:00
|
|
|
static inline void WriteHigh(std::atomic<u32>& reg, u16 highbits)
|
2014-08-25 21:09:26 +02:00
|
|
|
{
|
2021-05-13 18:44:59 +02:00
|
|
|
reg.store((reg.load(std::memory_order_relaxed) & 0x0000FFFF) | (static_cast<u32>(highbits) << 16),
|
|
|
|
std::memory_order_relaxed);
|
2014-08-25 21:09:26 +02:00
|
|
|
}
|
2010-06-09 01:37:08 +00:00
|
|
|
|
|
|
|
void Init()
|
|
|
|
{
|
2016-06-24 10:43:46 +02:00
|
|
|
m_CPStatusReg.Hex = 0;
|
|
|
|
m_CPStatusReg.CommandIdle = 1;
|
|
|
|
m_CPStatusReg.ReadIdle = 1;
|
2010-06-09 01:37:08 +00:00
|
|
|
|
2016-06-24 10:43:46 +02:00
|
|
|
m_CPCtrlReg.Hex = 0;
|
2010-06-09 01:37:08 +00:00
|
|
|
|
2016-06-24 10:43:46 +02:00
|
|
|
m_CPClearReg.Hex = 0;
|
2012-01-02 02:20:22 -08:00
|
|
|
|
2016-06-24 10:43:46 +02:00
|
|
|
m_bboxleft = 0;
|
|
|
|
m_bboxtop = 0;
|
|
|
|
m_bboxright = 640;
|
|
|
|
m_bboxbottom = 480;
|
2010-06-09 01:37:08 +00:00
|
|
|
|
2016-06-24 10:43:46 +02:00
|
|
|
m_tokenReg = 0;
|
2013-10-29 01:23:17 -04:00
|
|
|
|
2021-05-17 21:55:03 +02:00
|
|
|
fifo.Init();
|
2010-12-11 12:42:55 +00:00
|
|
|
|
2016-08-05 16:04:39 +02:00
|
|
|
s_interrupt_set.Clear();
|
|
|
|
s_interrupt_waiting.Clear();
|
2010-06-09 01:37:08 +00:00
|
|
|
|
2016-06-24 10:43:46 +02:00
|
|
|
et_UpdateInterrupts = CoreTiming::RegisterEvent("CPInterrupt", UpdateInterrupts_Wrapper);
|
2010-06-09 01:37:08 +00:00
|
|
|
}
|
|
|
|
|
2019-05-31 02:46:17 -04:00
|
|
|
u32 GetPhysicalAddressMask()
|
|
|
|
{
|
|
|
|
// Physical addresses in CP seem to ignore some of the upper bits (depending on platform)
|
|
|
|
// This can be observed in CP MMIO registers by setting to 0xffffffff and then reading back.
|
|
|
|
return SConfig::GetInstance().bWii ? 0x1fffffff : 0x03ffffff;
|
|
|
|
}
|
|
|
|
|
2014-02-02 14:16:43 +01:00
|
|
|
void RegisterMMIO(MMIO::Mapping* mmio, u32 base)
|
2010-06-09 01:37:08 +00:00
|
|
|
{
|
2018-07-12 03:08:30 -04:00
|
|
|
constexpr u16 WMASK_NONE = 0x0000;
|
|
|
|
constexpr u16 WMASK_ALL = 0xffff;
|
|
|
|
constexpr u16 WMASK_LO_ALIGN_32BIT = 0xffe0;
|
2019-05-31 02:46:17 -04:00
|
|
|
const u16 WMASK_HI_RESTRICT = GetPhysicalAddressMask() >> 16;
|
2018-07-12 03:08:30 -04:00
|
|
|
|
2016-06-24 10:43:46 +02:00
|
|
|
struct
|
|
|
|
{
|
|
|
|
u32 addr;
|
|
|
|
u16* ptr;
|
|
|
|
bool readonly;
|
2018-07-12 03:08:30 -04:00
|
|
|
// FIFO mmio regs in the range [cc000020-cc00003e] have certain bits that always read as 0
|
|
|
|
// For _LO registers in this range, only bits 0xffe0 can be set
|
|
|
|
// For _HI registers in this range, only bits 0x03ff can be set on GCN and 0x1fff on Wii
|
|
|
|
u16 wmask;
|
2016-06-24 10:43:46 +02:00
|
|
|
} directly_mapped_vars[] = {
|
2018-07-12 03:08:30 -04:00
|
|
|
{FIFO_TOKEN_REGISTER, &m_tokenReg, false, WMASK_ALL},
|
2016-06-24 10:43:46 +02:00
|
|
|
|
|
|
|
// Bounding box registers are read only.
|
2018-07-12 03:08:30 -04:00
|
|
|
{FIFO_BOUNDING_BOX_LEFT, &m_bboxleft, true, WMASK_NONE},
|
|
|
|
{FIFO_BOUNDING_BOX_RIGHT, &m_bboxright, true, WMASK_NONE},
|
|
|
|
{FIFO_BOUNDING_BOX_TOP, &m_bboxtop, true, WMASK_NONE},
|
|
|
|
{FIFO_BOUNDING_BOX_BOTTOM, &m_bboxbottom, true, WMASK_NONE},
|
|
|
|
{FIFO_BASE_LO, MMIO::Utils::LowPart(&fifo.CPBase), false, WMASK_LO_ALIGN_32BIT},
|
|
|
|
{FIFO_BASE_HI, MMIO::Utils::HighPart(&fifo.CPBase), false, WMASK_HI_RESTRICT},
|
|
|
|
{FIFO_END_LO, MMIO::Utils::LowPart(&fifo.CPEnd), false, WMASK_LO_ALIGN_32BIT},
|
|
|
|
{FIFO_END_HI, MMIO::Utils::HighPart(&fifo.CPEnd), false, WMASK_HI_RESTRICT},
|
|
|
|
{FIFO_HI_WATERMARK_LO, MMIO::Utils::LowPart(&fifo.CPHiWatermark), false,
|
|
|
|
WMASK_LO_ALIGN_32BIT},
|
|
|
|
{FIFO_HI_WATERMARK_HI, MMIO::Utils::HighPart(&fifo.CPHiWatermark), false, WMASK_HI_RESTRICT},
|
|
|
|
{FIFO_LO_WATERMARK_LO, MMIO::Utils::LowPart(&fifo.CPLoWatermark), false,
|
|
|
|
WMASK_LO_ALIGN_32BIT},
|
|
|
|
{FIFO_LO_WATERMARK_HI, MMIO::Utils::HighPart(&fifo.CPLoWatermark), false, WMASK_HI_RESTRICT},
|
2016-06-24 10:43:46 +02:00
|
|
|
// FIFO_RW_DISTANCE has some complex read code different for
|
|
|
|
// single/dual core.
|
2018-07-12 03:08:30 -04:00
|
|
|
{FIFO_WRITE_POINTER_LO, MMIO::Utils::LowPart(&fifo.CPWritePointer), false,
|
|
|
|
WMASK_LO_ALIGN_32BIT},
|
|
|
|
{FIFO_WRITE_POINTER_HI, MMIO::Utils::HighPart(&fifo.CPWritePointer), false,
|
|
|
|
WMASK_HI_RESTRICT},
|
2016-06-24 10:43:46 +02:00
|
|
|
// FIFO_READ_POINTER has different code for single/dual core.
|
|
|
|
};
|
|
|
|
|
|
|
|
for (auto& mapped_var : directly_mapped_vars)
|
|
|
|
{
|
|
|
|
mmio->Register(base | mapped_var.addr, MMIO::DirectRead<u16>(mapped_var.ptr),
|
|
|
|
mapped_var.readonly ? MMIO::InvalidWrite<u16>() :
|
2018-07-12 03:08:30 -04:00
|
|
|
MMIO::DirectWrite<u16>(mapped_var.ptr, mapped_var.wmask));
|
2016-06-24 10:43:46 +02:00
|
|
|
}
|
|
|
|
|
2018-07-12 03:08:30 -04:00
|
|
|
mmio->Register(base | FIFO_BP_LO, MMIO::DirectRead<u16>(MMIO::Utils::LowPart(&fifo.CPBreakpoint)),
|
2020-06-25 13:11:29 +02:00
|
|
|
MMIO::ComplexWrite<u16>([](u32, u16 val) {
|
2018-07-12 03:08:30 -04:00
|
|
|
WriteLow(fifo.CPBreakpoint, val & WMASK_LO_ALIGN_32BIT);
|
|
|
|
}));
|
2016-06-24 10:43:46 +02:00
|
|
|
mmio->Register(base | FIFO_BP_HI,
|
|
|
|
MMIO::DirectRead<u16>(MMIO::Utils::HighPart(&fifo.CPBreakpoint)),
|
2018-07-12 03:08:30 -04:00
|
|
|
MMIO::ComplexWrite<u16>([WMASK_HI_RESTRICT](u32, u16 val) {
|
|
|
|
WriteHigh(fifo.CPBreakpoint, val & WMASK_HI_RESTRICT);
|
|
|
|
}));
|
2016-06-24 10:43:46 +02:00
|
|
|
|
|
|
|
// Timing and metrics MMIOs are stubbed with fixed values.
|
|
|
|
struct
|
|
|
|
{
|
|
|
|
u32 addr;
|
|
|
|
u16 value;
|
|
|
|
} metrics_mmios[] = {
|
|
|
|
{XF_RASBUSY_L, 0},
|
|
|
|
{XF_RASBUSY_H, 0},
|
|
|
|
{XF_CLKS_L, 0},
|
|
|
|
{XF_CLKS_H, 0},
|
|
|
|
{XF_WAIT_IN_L, 0},
|
|
|
|
{XF_WAIT_IN_H, 0},
|
|
|
|
{XF_WAIT_OUT_L, 0},
|
|
|
|
{XF_WAIT_OUT_H, 0},
|
|
|
|
{VCACHE_METRIC_CHECK_L, 0},
|
|
|
|
{VCACHE_METRIC_CHECK_H, 0},
|
|
|
|
{VCACHE_METRIC_MISS_L, 0},
|
|
|
|
{VCACHE_METRIC_MISS_H, 0},
|
|
|
|
{VCACHE_METRIC_STALL_L, 0},
|
|
|
|
{VCACHE_METRIC_STALL_H, 0},
|
|
|
|
{CLKS_PER_VTX_OUT, 4},
|
|
|
|
};
|
|
|
|
for (auto& metrics_mmio : metrics_mmios)
|
|
|
|
{
|
|
|
|
mmio->Register(base | metrics_mmio.addr, MMIO::Constant<u16>(metrics_mmio.value),
|
|
|
|
MMIO::InvalidWrite<u16>());
|
|
|
|
}
|
|
|
|
|
|
|
|
mmio->Register(base | STATUS_REGISTER, MMIO::ComplexRead<u16>([](u32) {
|
2021-04-17 15:54:50 +10:00
|
|
|
Fifo::SyncGPUForRegisterAccess();
|
2016-06-24 10:43:46 +02:00
|
|
|
SetCpStatusRegister();
|
|
|
|
return m_CPStatusReg.Hex;
|
|
|
|
}),
|
|
|
|
MMIO::InvalidWrite<u16>());
|
|
|
|
|
|
|
|
mmio->Register(base | CTRL_REGISTER, MMIO::DirectRead<u16>(&m_CPCtrlReg.Hex),
|
|
|
|
MMIO::ComplexWrite<u16>([](u32, u16 val) {
|
|
|
|
UCPCtrlReg tmp(val);
|
|
|
|
m_CPCtrlReg.Hex = tmp.Hex;
|
|
|
|
SetCpControlRegister();
|
|
|
|
Fifo::RunGpu();
|
|
|
|
}));
|
|
|
|
|
|
|
|
mmio->Register(base | CLEAR_REGISTER, MMIO::DirectRead<u16>(&m_CPClearReg.Hex),
|
|
|
|
MMIO::ComplexWrite<u16>([](u32, u16 val) {
|
|
|
|
UCPClearReg tmp(val);
|
|
|
|
m_CPClearReg.Hex = tmp.Hex;
|
|
|
|
SetCpClearRegister();
|
|
|
|
Fifo::RunGpu();
|
|
|
|
}));
|
|
|
|
|
|
|
|
mmio->Register(base | PERF_SELECT, MMIO::InvalidRead<u16>(), MMIO::Nop<u16>());
|
|
|
|
|
|
|
|
// Some MMIOs have different handlers for single core vs. dual core mode.
|
2021-05-13 18:44:59 +02:00
|
|
|
mmio->Register(
|
|
|
|
base | FIFO_RW_DISTANCE_LO,
|
|
|
|
IsOnThread() ? MMIO::ComplexRead<u16>([](u32) {
|
|
|
|
if (fifo.CPWritePointer.load(std::memory_order_relaxed) >=
|
|
|
|
fifo.SafeCPReadPointer.load(std::memory_order_relaxed))
|
|
|
|
{
|
|
|
|
return static_cast<u16>(fifo.CPWritePointer.load(std::memory_order_relaxed) -
|
|
|
|
fifo.SafeCPReadPointer.load(std::memory_order_relaxed));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return static_cast<u16>(fifo.CPEnd.load(std::memory_order_relaxed) -
|
|
|
|
fifo.SafeCPReadPointer.load(std::memory_order_relaxed) +
|
|
|
|
fifo.CPWritePointer.load(std::memory_order_relaxed) -
|
|
|
|
fifo.CPBase.load(std::memory_order_relaxed) + 32);
|
|
|
|
}
|
|
|
|
}) :
|
|
|
|
MMIO::DirectRead<u16>(MMIO::Utils::LowPart(&fifo.CPReadWriteDistance)),
|
|
|
|
MMIO::DirectWrite<u16>(MMIO::Utils::LowPart(&fifo.CPReadWriteDistance),
|
|
|
|
WMASK_LO_ALIGN_32BIT));
|
|
|
|
mmio->Register(base | FIFO_RW_DISTANCE_HI,
|
2016-06-24 10:43:46 +02:00
|
|
|
IsOnThread() ?
|
|
|
|
MMIO::ComplexRead<u16>([](u32) {
|
2021-05-13 18:44:59 +02:00
|
|
|
Fifo::SyncGPUForRegisterAccess();
|
|
|
|
if (fifo.CPWritePointer.load(std::memory_order_relaxed) >=
|
|
|
|
fifo.SafeCPReadPointer.load(std::memory_order_relaxed))
|
|
|
|
{
|
|
|
|
return (fifo.CPWritePointer.load(std::memory_order_relaxed) -
|
|
|
|
fifo.SafeCPReadPointer.load(std::memory_order_relaxed)) >>
|
|
|
|
16;
|
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
else
|
2021-05-13 18:44:59 +02:00
|
|
|
{
|
|
|
|
return (fifo.CPEnd.load(std::memory_order_relaxed) -
|
|
|
|
fifo.SafeCPReadPointer.load(std::memory_order_relaxed) +
|
|
|
|
fifo.CPWritePointer.load(std::memory_order_relaxed) -
|
|
|
|
fifo.CPBase.load(std::memory_order_relaxed) + 32) >>
|
|
|
|
16;
|
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
}) :
|
2021-05-13 18:44:59 +02:00
|
|
|
MMIO::ComplexRead<u16>([](u32) {
|
|
|
|
Fifo::SyncGPUForRegisterAccess();
|
|
|
|
return fifo.CPReadWriteDistance.load(std::memory_order_relaxed) >> 16;
|
|
|
|
}),
|
2018-07-12 03:08:30 -04:00
|
|
|
MMIO::ComplexWrite<u16>([WMASK_HI_RESTRICT](u32, u16 val) {
|
2021-04-17 15:54:50 +10:00
|
|
|
Fifo::SyncGPUForRegisterAccess();
|
2018-07-12 03:08:30 -04:00
|
|
|
WriteHigh(fifo.CPReadWriteDistance, val & WMASK_HI_RESTRICT);
|
2016-06-24 10:43:46 +02:00
|
|
|
Fifo::RunGpu();
|
|
|
|
}));
|
2018-07-12 03:08:30 -04:00
|
|
|
mmio->Register(
|
|
|
|
base | FIFO_READ_POINTER_LO,
|
|
|
|
IsOnThread() ? MMIO::DirectRead<u16>(MMIO::Utils::LowPart(&fifo.SafeCPReadPointer)) :
|
2016-06-24 10:43:46 +02:00
|
|
|
MMIO::DirectRead<u16>(MMIO::Utils::LowPart(&fifo.CPReadPointer)),
|
2018-07-12 03:08:30 -04:00
|
|
|
MMIO::DirectWrite<u16>(MMIO::Utils::LowPart(&fifo.CPReadPointer), WMASK_LO_ALIGN_32BIT));
|
2021-04-17 15:54:50 +10:00
|
|
|
mmio->Register(base | FIFO_READ_POINTER_HI,
|
|
|
|
IsOnThread() ? MMIO::ComplexRead<u16>([](u32) {
|
|
|
|
Fifo::SyncGPUForRegisterAccess();
|
2021-05-13 18:44:59 +02:00
|
|
|
return fifo.SafeCPReadPointer.load(std::memory_order_relaxed) >> 16;
|
2021-04-17 15:54:50 +10:00
|
|
|
}) :
|
|
|
|
MMIO::ComplexRead<u16>([](u32) {
|
|
|
|
Fifo::SyncGPUForRegisterAccess();
|
2021-05-13 18:44:59 +02:00
|
|
|
return fifo.CPReadPointer.load(std::memory_order_relaxed) >> 16;
|
2021-04-17 15:54:50 +10:00
|
|
|
}),
|
|
|
|
IsOnThread() ? MMIO::ComplexWrite<u16>([WMASK_HI_RESTRICT](u32, u16 val) {
|
|
|
|
Fifo::SyncGPUForRegisterAccess();
|
|
|
|
WriteHigh(fifo.CPReadPointer, val & WMASK_HI_RESTRICT);
|
2021-05-13 18:44:59 +02:00
|
|
|
fifo.SafeCPReadPointer.store(fifo.CPReadPointer.load(std::memory_order_relaxed),
|
|
|
|
std::memory_order_relaxed);
|
2021-04-17 15:54:50 +10:00
|
|
|
}) :
|
|
|
|
MMIO::ComplexWrite<u16>([WMASK_HI_RESTRICT](u32, u16 val) {
|
|
|
|
Fifo::SyncGPUForRegisterAccess();
|
|
|
|
WriteHigh(fifo.CPReadPointer, val & WMASK_HI_RESTRICT);
|
|
|
|
}));
|
2014-02-02 14:16:43 +01:00
|
|
|
}
|
2010-06-09 01:37:08 +00:00
|
|
|
|
2014-09-30 01:22:57 -04:00
|
|
|
void GatherPipeBursted()
|
2010-06-09 01:37:08 +00:00
|
|
|
{
|
2016-01-10 12:18:45 +01:00
|
|
|
SetCPStatusFromCPU();
|
2016-06-24 10:43:46 +02:00
|
|
|
|
|
|
|
// if we aren't linked, we don't care about gather pipe data
|
|
|
|
if (!m_CPCtrlReg.GPLinkEnable)
|
|
|
|
{
|
|
|
|
if (IsOnThread() && !Fifo::UseDeterministicGPUThread())
|
|
|
|
{
|
|
|
|
// In multibuffer mode is not allowed write in the same FIFO attached to the GPU.
|
|
|
|
// Fix Pokemon XD in DC mode.
|
2021-05-13 18:44:59 +02:00
|
|
|
if ((ProcessorInterface::Fifo_CPUEnd == fifo.CPEnd.load(std::memory_order_relaxed)) &&
|
|
|
|
(ProcessorInterface::Fifo_CPUBase == fifo.CPBase.load(std::memory_order_relaxed)) &&
|
|
|
|
fifo.CPReadWriteDistance.load(std::memory_order_relaxed) > 0)
|
2016-06-24 10:43:46 +02:00
|
|
|
{
|
|
|
|
Fifo::FlushGpu();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Fifo::RunGpu();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// update the fifo pointer
|
2021-05-13 18:44:59 +02:00
|
|
|
if (fifo.CPWritePointer.load(std::memory_order_relaxed) ==
|
|
|
|
fifo.CPEnd.load(std::memory_order_relaxed))
|
|
|
|
{
|
|
|
|
fifo.CPWritePointer.store(fifo.CPBase, std::memory_order_relaxed);
|
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
else
|
2021-05-13 18:44:59 +02:00
|
|
|
{
|
|
|
|
fifo.CPWritePointer.fetch_add(GATHER_PIPE_SIZE, std::memory_order_relaxed);
|
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
|
|
|
if (m_CPCtrlReg.GPReadEnable && m_CPCtrlReg.GPLinkEnable)
|
|
|
|
{
|
2021-05-13 18:44:59 +02:00
|
|
|
ProcessorInterface::Fifo_CPUWritePointer = fifo.CPWritePointer.load(std::memory_order_relaxed);
|
|
|
|
ProcessorInterface::Fifo_CPUBase = fifo.CPBase.load(std::memory_order_relaxed);
|
|
|
|
ProcessorInterface::Fifo_CPUEnd = fifo.CPEnd.load(std::memory_order_relaxed);
|
2016-06-24 10:43:46 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// If the game is running close to overflowing, make the exception checking more frequent.
|
2021-05-13 19:30:30 +02:00
|
|
|
if (fifo.bFF_HiWatermark.load(std::memory_order_relaxed) != 0)
|
2016-06-24 10:43:46 +02:00
|
|
|
CoreTiming::ForceExceptionCheck(0);
|
|
|
|
|
2021-05-13 18:44:59 +02:00
|
|
|
fifo.CPReadWriteDistance.fetch_add(GATHER_PIPE_SIZE, std::memory_order_seq_cst);
|
2016-06-24 10:43:46 +02:00
|
|
|
|
|
|
|
Fifo::RunGpu();
|
|
|
|
|
2021-05-13 18:44:59 +02:00
|
|
|
ASSERT_MSG(COMMANDPROCESSOR,
|
|
|
|
fifo.CPReadWriteDistance.load(std::memory_order_relaxed) <=
|
|
|
|
fifo.CPEnd.load(std::memory_order_relaxed) -
|
|
|
|
fifo.CPBase.load(std::memory_order_relaxed),
|
2018-03-14 20:34:35 -04:00
|
|
|
"FIFO is overflowed by GatherPipe !\nCPU thread is too fast!");
|
2016-06-24 10:43:46 +02:00
|
|
|
|
|
|
|
// check if we are in sync
|
2021-05-13 18:44:59 +02:00
|
|
|
ASSERT_MSG(COMMANDPROCESSOR,
|
|
|
|
fifo.CPWritePointer.load(std::memory_order_relaxed) ==
|
|
|
|
ProcessorInterface::Fifo_CPUWritePointer,
|
2018-03-14 20:34:35 -04:00
|
|
|
"FIFOs linked but out of sync");
|
2021-05-13 18:44:59 +02:00
|
|
|
ASSERT_MSG(COMMANDPROCESSOR,
|
|
|
|
fifo.CPBase.load(std::memory_order_relaxed) == ProcessorInterface::Fifo_CPUBase,
|
2018-03-14 20:34:35 -04:00
|
|
|
"FIFOs linked but out of sync");
|
2021-05-13 18:44:59 +02:00
|
|
|
ASSERT_MSG(COMMANDPROCESSOR,
|
|
|
|
fifo.CPEnd.load(std::memory_order_relaxed) == ProcessorInterface::Fifo_CPUEnd,
|
2018-03-14 20:34:35 -04:00
|
|
|
"FIFOs linked but out of sync");
|
2010-06-09 01:37:08 +00:00
|
|
|
}
|
|
|
|
|
2010-12-11 12:42:55 +00:00
|
|
|
void UpdateInterrupts(u64 userdata)
|
|
|
|
{
|
2016-06-24 10:43:46 +02:00
|
|
|
if (userdata)
|
|
|
|
{
|
2016-08-05 16:04:39 +02:00
|
|
|
s_interrupt_set.Set();
|
2020-11-13 22:33:26 -05:00
|
|
|
DEBUG_LOG_FMT(COMMANDPROCESSOR, "Interrupt set");
|
2016-06-24 10:43:46 +02:00
|
|
|
ProcessorInterface::SetInterrupt(INT_CAUSE_CP, true);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-08-05 16:04:39 +02:00
|
|
|
s_interrupt_set.Clear();
|
2020-11-13 22:33:26 -05:00
|
|
|
DEBUG_LOG_FMT(COMMANDPROCESSOR, "Interrupt cleared");
|
2016-06-24 10:43:46 +02:00
|
|
|
ProcessorInterface::SetInterrupt(INT_CAUSE_CP, false);
|
|
|
|
}
|
|
|
|
CoreTiming::ForceExceptionCheck(0);
|
2016-08-05 16:04:39 +02:00
|
|
|
s_interrupt_waiting.Clear();
|
2016-06-24 10:43:46 +02:00
|
|
|
Fifo::RunGpu();
|
2010-12-11 12:42:55 +00:00
|
|
|
}
|
|
|
|
|
2011-02-14 02:18:03 +00:00
|
|
|
void UpdateInterruptsFromVideoBackend(u64 userdata)
|
2010-06-09 01:37:08 +00:00
|
|
|
{
|
2016-06-24 10:43:46 +02:00
|
|
|
if (!Fifo::UseDeterministicGPUThread())
|
2016-07-07 15:52:08 +02:00
|
|
|
CoreTiming::ScheduleEvent(0, et_UpdateInterrupts, userdata, CoreTiming::FromThread::NON_CPU);
|
2010-06-09 01:37:08 +00:00
|
|
|
}
|
|
|
|
|
2015-05-27 03:08:48 -04:00
|
|
|
bool IsInterruptWaiting()
|
|
|
|
{
|
2016-08-05 16:04:39 +02:00
|
|
|
return s_interrupt_waiting.IsSet();
|
2015-05-27 03:08:48 -04:00
|
|
|
}
|
|
|
|
|
2014-08-24 16:27:32 -04:00
|
|
|
void SetCPStatusFromGPU()
|
2010-12-11 12:42:55 +00:00
|
|
|
{
|
2016-06-24 10:43:46 +02:00
|
|
|
// breakpoint
|
2021-05-13 19:30:30 +02:00
|
|
|
const bool breakpoint = fifo.bFF_Breakpoint.load(std::memory_order_relaxed);
|
|
|
|
if (fifo.bFF_BPEnable.load(std::memory_order_relaxed) != 0)
|
2016-06-24 10:43:46 +02:00
|
|
|
{
|
2021-05-13 18:44:59 +02:00
|
|
|
if (fifo.CPBreakpoint.load(std::memory_order_relaxed) ==
|
|
|
|
fifo.CPReadPointer.load(std::memory_order_relaxed))
|
2016-06-24 10:43:46 +02:00
|
|
|
{
|
2021-05-13 19:30:30 +02:00
|
|
|
if (!breakpoint)
|
2016-06-24 10:43:46 +02:00
|
|
|
{
|
2021-05-13 18:44:59 +02:00
|
|
|
DEBUG_LOG_FMT(COMMANDPROCESSOR, "Hit breakpoint at {}",
|
|
|
|
fifo.CPReadPointer.load(std::memory_order_relaxed));
|
2021-05-13 19:30:30 +02:00
|
|
|
fifo.bFF_Breakpoint.store(1, std::memory_order_relaxed);
|
2016-06-24 10:43:46 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2021-05-13 19:30:30 +02:00
|
|
|
if (breakpoint)
|
2021-05-13 18:44:59 +02:00
|
|
|
{
|
|
|
|
DEBUG_LOG_FMT(COMMANDPROCESSOR, "Cleared breakpoint at {}",
|
|
|
|
fifo.CPReadPointer.load(std::memory_order_relaxed));
|
2021-05-13 19:30:30 +02:00
|
|
|
fifo.bFF_Breakpoint.store(0, std::memory_order_relaxed);
|
2021-05-13 18:44:59 +02:00
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2021-05-13 19:30:30 +02:00
|
|
|
if (breakpoint)
|
2021-05-13 18:44:59 +02:00
|
|
|
{
|
|
|
|
DEBUG_LOG_FMT(COMMANDPROCESSOR, "Cleared breakpoint at {}",
|
|
|
|
fifo.CPReadPointer.load(std::memory_order_relaxed));
|
2021-05-13 19:30:30 +02:00
|
|
|
fifo.bFF_Breakpoint = false;
|
2021-05-13 18:44:59 +02:00
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// overflow & underflow check
|
2021-05-13 19:30:30 +02:00
|
|
|
fifo.bFF_HiWatermark.store(
|
|
|
|
(fifo.CPReadWriteDistance.load(std::memory_order_relaxed) > fifo.CPHiWatermark),
|
|
|
|
std::memory_order_relaxed);
|
|
|
|
fifo.bFF_LoWatermark.store(
|
|
|
|
(fifo.CPReadWriteDistance.load(std::memory_order_relaxed) < fifo.CPLoWatermark),
|
|
|
|
std::memory_order_relaxed);
|
|
|
|
|
|
|
|
bool bpInt = fifo.bFF_Breakpoint.load(std::memory_order_relaxed) &&
|
|
|
|
fifo.bFF_BPInt.load(std::memory_order_relaxed);
|
|
|
|
bool ovfInt = fifo.bFF_HiWatermark.load(std::memory_order_relaxed) &&
|
|
|
|
fifo.bFF_HiWatermarkInt.load(std::memory_order_relaxed);
|
|
|
|
bool undfInt = fifo.bFF_LoWatermark.load(std::memory_order_relaxed) &&
|
|
|
|
fifo.bFF_LoWatermarkInt.load(std::memory_order_relaxed);
|
2016-06-24 10:43:46 +02:00
|
|
|
|
|
|
|
bool interrupt = (bpInt || ovfInt || undfInt) && m_CPCtrlReg.GPReadEnable;
|
|
|
|
|
2016-08-05 16:04:39 +02:00
|
|
|
if (interrupt != s_interrupt_set.IsSet() && !s_interrupt_waiting.IsSet())
|
2016-06-24 10:43:46 +02:00
|
|
|
{
|
|
|
|
u64 userdata = interrupt ? 1 : 0;
|
|
|
|
if (IsOnThread())
|
|
|
|
{
|
|
|
|
if (!interrupt || bpInt || undfInt || ovfInt)
|
|
|
|
{
|
|
|
|
// Schedule the interrupt asynchronously
|
2016-08-05 16:04:39 +02:00
|
|
|
s_interrupt_waiting.Set();
|
2016-06-24 10:43:46 +02:00
|
|
|
CommandProcessor::UpdateInterruptsFromVideoBackend(userdata);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
CommandProcessor::UpdateInterrupts(userdata);
|
|
|
|
}
|
|
|
|
}
|
2014-08-24 16:27:32 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void SetCPStatusFromCPU()
|
|
|
|
{
|
2016-06-24 10:43:46 +02:00
|
|
|
// overflow & underflow check
|
2021-05-13 19:30:30 +02:00
|
|
|
fifo.bFF_HiWatermark.store(
|
|
|
|
(fifo.CPReadWriteDistance.load(std::memory_order_relaxed) > fifo.CPHiWatermark),
|
|
|
|
std::memory_order_relaxed);
|
|
|
|
fifo.bFF_LoWatermark.store(
|
|
|
|
(fifo.CPReadWriteDistance.load(std::memory_order_relaxed) < fifo.CPLoWatermark),
|
|
|
|
std::memory_order_relaxed);
|
|
|
|
|
|
|
|
bool bpInt = fifo.bFF_Breakpoint.load(std::memory_order_relaxed) &&
|
|
|
|
fifo.bFF_BPInt.load(std::memory_order_relaxed);
|
|
|
|
bool ovfInt = fifo.bFF_HiWatermark.load(std::memory_order_relaxed) &&
|
|
|
|
fifo.bFF_HiWatermarkInt.load(std::memory_order_relaxed);
|
|
|
|
bool undfInt = fifo.bFF_LoWatermark.load(std::memory_order_relaxed) &&
|
|
|
|
fifo.bFF_LoWatermarkInt.load(std::memory_order_relaxed);
|
2016-06-24 10:43:46 +02:00
|
|
|
|
|
|
|
bool interrupt = (bpInt || ovfInt || undfInt) && m_CPCtrlReg.GPReadEnable;
|
|
|
|
|
2016-08-05 16:04:39 +02:00
|
|
|
if (interrupt != s_interrupt_set.IsSet() && !s_interrupt_waiting.IsSet())
|
2016-06-24 10:43:46 +02:00
|
|
|
{
|
|
|
|
u64 userdata = interrupt ? 1 : 0;
|
|
|
|
if (IsOnThread())
|
|
|
|
{
|
|
|
|
if (!interrupt || bpInt || undfInt || ovfInt)
|
|
|
|
{
|
2016-08-05 16:04:39 +02:00
|
|
|
s_interrupt_set.Set(interrupt);
|
2020-11-13 22:33:26 -05:00
|
|
|
DEBUG_LOG_FMT(COMMANDPROCESSOR, "Interrupt set");
|
2016-06-24 10:43:46 +02:00
|
|
|
ProcessorInterface::SetInterrupt(INT_CAUSE_CP, interrupt);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
CommandProcessor::UpdateInterrupts(userdata);
|
|
|
|
}
|
|
|
|
}
|
2010-12-11 12:42:55 +00:00
|
|
|
}
|
|
|
|
|
Big Fifo Commit Part2: Now the fifo is more stable than my first commit, so is time...
- ReImplementing Single Core Mode like Dual Core Mode Style.
- Stage 1: My goal is, we have the Fifo, CommandProccessor code the more clear, maintenible and documented possible. When I quit dolphin I want any developer can continue with the work only reading the code.
* Big Refactoring: A lot of functions was changed the names, and modularized.
Now the FifoLoop and CatchUpGPU does not exist, was replaced by RunGpu() and RunGpuLoop().
The general idea is modeling the code like the real HW. The fifo is only a buffer where the Write Gather Pipe write the commands and from the Graphic Processor read these.
* Big Clean UP a lot of obsolete code and comments was deleted, like DcFakeWachDog, "Fifo very soon hack", etc.
In the stage 2, I will refactoring more code doing emphasis in the division of CommandProcessor, Fifo, Gpu Emulation. Beside I will comment all functions and variables in the code (Don't worry I will ask for English help for this part ;) )
Please test a lot SC mode and DC mode :)
Thank you so much for testing always and the patience. I don't like broke your favorite game but... you must believe me this part is very sensible, I only try to contribute for have a better and stable dolphin emulator.
git-svn-id: https://dolphin-emu.googlecode.com/svn/trunk@7185 8ced0084-cf51-0410-be5f-012b33b47a6e
2011-02-17 04:25:21 +00:00
|
|
|
void SetCpStatusRegister()
|
|
|
|
{
|
2016-06-24 10:43:46 +02:00
|
|
|
// Here always there is one fifo attached to the GPU
|
2021-05-13 19:30:30 +02:00
|
|
|
m_CPStatusReg.Breakpoint = fifo.bFF_Breakpoint.load(std::memory_order_relaxed);
|
2021-05-13 18:44:59 +02:00
|
|
|
m_CPStatusReg.ReadIdle = !fifo.CPReadWriteDistance.load(std::memory_order_relaxed) ||
|
|
|
|
(fifo.CPReadPointer.load(std::memory_order_relaxed) ==
|
|
|
|
fifo.CPWritePointer.load(std::memory_order_relaxed));
|
|
|
|
m_CPStatusReg.CommandIdle = !fifo.CPReadWriteDistance.load(std::memory_order_relaxed) ||
|
2021-05-13 19:30:30 +02:00
|
|
|
Fifo::AtBreakpoint() ||
|
|
|
|
!fifo.bFF_GPReadEnable.load(std::memory_order_relaxed);
|
|
|
|
m_CPStatusReg.UnderflowLoWatermark = fifo.bFF_LoWatermark.load(std::memory_order_relaxed);
|
|
|
|
m_CPStatusReg.OverflowHiWatermark = fifo.bFF_HiWatermark.load(std::memory_order_relaxed);
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2020-11-13 22:33:26 -05:00
|
|
|
DEBUG_LOG_FMT(COMMANDPROCESSOR, "\t Read from STATUS_REGISTER : {:04x}", m_CPStatusReg.Hex);
|
|
|
|
DEBUG_LOG_FMT(
|
|
|
|
COMMANDPROCESSOR, "(r) status: iBP {} | fReadIdle {} | fCmdIdle {} | iOvF {} | iUndF {}",
|
2016-06-24 10:43:46 +02:00
|
|
|
m_CPStatusReg.Breakpoint ? "ON" : "OFF", m_CPStatusReg.ReadIdle ? "ON" : "OFF",
|
|
|
|
m_CPStatusReg.CommandIdle ? "ON" : "OFF", m_CPStatusReg.OverflowHiWatermark ? "ON" : "OFF",
|
|
|
|
m_CPStatusReg.UnderflowLoWatermark ? "ON" : "OFF");
|
Big Fifo Commit Part2: Now the fifo is more stable than my first commit, so is time...
- ReImplementing Single Core Mode like Dual Core Mode Style.
- Stage 1: My goal is, we have the Fifo, CommandProccessor code the more clear, maintenible and documented possible. When I quit dolphin I want any developer can continue with the work only reading the code.
* Big Refactoring: A lot of functions was changed the names, and modularized.
Now the FifoLoop and CatchUpGPU does not exist, was replaced by RunGpu() and RunGpuLoop().
The general idea is modeling the code like the real HW. The fifo is only a buffer where the Write Gather Pipe write the commands and from the Graphic Processor read these.
* Big Clean UP a lot of obsolete code and comments was deleted, like DcFakeWachDog, "Fifo very soon hack", etc.
In the stage 2, I will refactoring more code doing emphasis in the division of CommandProcessor, Fifo, Gpu Emulation. Beside I will comment all functions and variables in the code (Don't worry I will ask for English help for this part ;) )
Please test a lot SC mode and DC mode :)
Thank you so much for testing always and the patience. I don't like broke your favorite game but... you must believe me this part is very sensible, I only try to contribute for have a better and stable dolphin emulator.
git-svn-id: https://dolphin-emu.googlecode.com/svn/trunk@7185 8ced0084-cf51-0410-be5f-012b33b47a6e
2011-02-17 04:25:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void SetCpControlRegister()
|
|
|
|
{
|
2021-05-13 19:30:30 +02:00
|
|
|
fifo.bFF_BPInt.store(m_CPCtrlReg.BPInt, std::memory_order_relaxed);
|
|
|
|
fifo.bFF_BPEnable.store(m_CPCtrlReg.BPEnable, std::memory_order_relaxed);
|
|
|
|
fifo.bFF_HiWatermarkInt.store(m_CPCtrlReg.FifoOverflowIntEnable, std::memory_order_relaxed);
|
|
|
|
fifo.bFF_LoWatermarkInt.store(m_CPCtrlReg.FifoUnderflowIntEnable, std::memory_order_relaxed);
|
|
|
|
fifo.bFF_GPLinkEnable.store(m_CPCtrlReg.GPLinkEnable, std::memory_order_relaxed);
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2021-05-13 19:30:30 +02:00
|
|
|
if (fifo.bFF_GPReadEnable.load(std::memory_order_relaxed) && !m_CPCtrlReg.GPReadEnable)
|
2016-06-24 10:43:46 +02:00
|
|
|
{
|
2021-05-13 19:30:30 +02:00
|
|
|
fifo.bFF_GPReadEnable.store(m_CPCtrlReg.GPReadEnable, std::memory_order_relaxed);
|
2016-06-24 10:43:46 +02:00
|
|
|
Fifo::FlushGpu();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fifo.bFF_GPReadEnable = m_CPCtrlReg.GPReadEnable;
|
|
|
|
}
|
|
|
|
|
2020-11-13 22:33:26 -05:00
|
|
|
DEBUG_LOG_FMT(COMMANDPROCESSOR, "\t GPREAD {} | BP {} | Int {} | OvF {} | UndF {} | LINK {}",
|
2021-05-13 19:30:30 +02:00
|
|
|
fifo.bFF_GPReadEnable.load(std::memory_order_relaxed) ? "ON" : "OFF",
|
|
|
|
fifo.bFF_BPEnable.load(std::memory_order_relaxed) ? "ON" : "OFF",
|
|
|
|
fifo.bFF_BPInt.load(std::memory_order_relaxed) ? "ON" : "OFF",
|
|
|
|
m_CPCtrlReg.FifoOverflowIntEnable ? "ON" : "OFF",
|
2020-11-13 22:33:26 -05:00
|
|
|
m_CPCtrlReg.FifoUnderflowIntEnable ? "ON" : "OFF",
|
|
|
|
m_CPCtrlReg.GPLinkEnable ? "ON" : "OFF");
|
Big Fifo Commit Part2: Now the fifo is more stable than my first commit, so is time...
- ReImplementing Single Core Mode like Dual Core Mode Style.
- Stage 1: My goal is, we have the Fifo, CommandProccessor code the more clear, maintenible and documented possible. When I quit dolphin I want any developer can continue with the work only reading the code.
* Big Refactoring: A lot of functions was changed the names, and modularized.
Now the FifoLoop and CatchUpGPU does not exist, was replaced by RunGpu() and RunGpuLoop().
The general idea is modeling the code like the real HW. The fifo is only a buffer where the Write Gather Pipe write the commands and from the Graphic Processor read these.
* Big Clean UP a lot of obsolete code and comments was deleted, like DcFakeWachDog, "Fifo very soon hack", etc.
In the stage 2, I will refactoring more code doing emphasis in the division of CommandProcessor, Fifo, Gpu Emulation. Beside I will comment all functions and variables in the code (Don't worry I will ask for English help for this part ;) )
Please test a lot SC mode and DC mode :)
Thank you so much for testing always and the patience. I don't like broke your favorite game but... you must believe me this part is very sensible, I only try to contribute for have a better and stable dolphin emulator.
git-svn-id: https://dolphin-emu.googlecode.com/svn/trunk@7185 8ced0084-cf51-0410-be5f-012b33b47a6e
2011-02-17 04:25:21 +00:00
|
|
|
}
|
|
|
|
|
2014-06-01 01:56:09 -07:00
|
|
|
// NOTE: We intentionally don't emulate this function at the moment.
|
|
|
|
// We don't emulate proper GP timing anyway at the moment, so it would just slow down emulation.
|
Big Fifo Commit Part2: Now the fifo is more stable than my first commit, so is time...
- ReImplementing Single Core Mode like Dual Core Mode Style.
- Stage 1: My goal is, we have the Fifo, CommandProccessor code the more clear, maintenible and documented possible. When I quit dolphin I want any developer can continue with the work only reading the code.
* Big Refactoring: A lot of functions was changed the names, and modularized.
Now the FifoLoop and CatchUpGPU does not exist, was replaced by RunGpu() and RunGpuLoop().
The general idea is modeling the code like the real HW. The fifo is only a buffer where the Write Gather Pipe write the commands and from the Graphic Processor read these.
* Big Clean UP a lot of obsolete code and comments was deleted, like DcFakeWachDog, "Fifo very soon hack", etc.
In the stage 2, I will refactoring more code doing emphasis in the division of CommandProcessor, Fifo, Gpu Emulation. Beside I will comment all functions and variables in the code (Don't worry I will ask for English help for this part ;) )
Please test a lot SC mode and DC mode :)
Thank you so much for testing always and the patience. I don't like broke your favorite game but... you must believe me this part is very sensible, I only try to contribute for have a better and stable dolphin emulator.
git-svn-id: https://dolphin-emu.googlecode.com/svn/trunk@7185 8ced0084-cf51-0410-be5f-012b33b47a6e
2011-02-17 04:25:21 +00:00
|
|
|
void SetCpClearRegister()
|
|
|
|
{
|
|
|
|
}
|
2010-12-11 12:42:55 +00:00
|
|
|
|
2021-04-22 20:57:56 -07:00
|
|
|
void HandleUnknownOpcode(u8 cmd_byte, const u8* buffer, bool preprocess)
|
2017-01-27 09:44:31 +01:00
|
|
|
{
|
|
|
|
// TODO(Omega): Maybe dump FIFO to file on this error
|
2021-04-22 20:57:56 -07:00
|
|
|
PanicAlertFmtT("GFX FIFO: Unknown Opcode ({0:#04x} @ {1}, preprocess={2}).\n"
|
2020-11-13 22:33:26 -05:00
|
|
|
"This means one of the following:\n"
|
|
|
|
"* The emulated GPU got desynced, disabling dual core can help\n"
|
|
|
|
"* Command stream corrupted by some spurious memory bug\n"
|
|
|
|
"* This really is an unknown opcode (unlikely)\n"
|
|
|
|
"* Some other sort of bug\n\n"
|
|
|
|
"Further errors will be sent to the Video Backend log and\n"
|
|
|
|
"Dolphin will now likely crash or hang. Enjoy.",
|
2021-04-22 20:57:56 -07:00
|
|
|
cmd_byte, fmt::ptr(buffer), preprocess);
|
2017-01-27 09:44:31 +01:00
|
|
|
|
|
|
|
{
|
2021-05-13 19:30:30 +02:00
|
|
|
PanicAlertFmt("Illegal command {:02x}\n"
|
|
|
|
"CPBase: {:#010x}\n"
|
|
|
|
"CPEnd: {:#010x}\n"
|
|
|
|
"CPHiWatermark: {:#010x}\n"
|
|
|
|
"CPLoWatermark: {:#010x}\n"
|
|
|
|
"CPReadWriteDistance: {:#010x}\n"
|
|
|
|
"CPWritePointer: {:#010x}\n"
|
|
|
|
"CPReadPointer: {:#010x}\n"
|
|
|
|
"CPBreakpoint: {:#010x}\n"
|
|
|
|
"bFF_GPReadEnable: {}\n"
|
|
|
|
"bFF_BPEnable: {}\n"
|
|
|
|
"bFF_BPInt: {}\n"
|
|
|
|
"bFF_Breakpoint: {}\n"
|
|
|
|
"bFF_GPLinkEnable: {}\n"
|
|
|
|
"bFF_HiWatermarkInt: {}\n"
|
|
|
|
"bFF_LoWatermarkInt: {}\n",
|
|
|
|
cmd_byte, fifo.CPBase.load(std::memory_order_relaxed),
|
|
|
|
fifo.CPEnd.load(std::memory_order_relaxed), fifo.CPHiWatermark,
|
|
|
|
fifo.CPLoWatermark, fifo.CPReadWriteDistance.load(std::memory_order_relaxed),
|
|
|
|
fifo.CPWritePointer.load(std::memory_order_relaxed),
|
|
|
|
fifo.CPReadPointer.load(std::memory_order_relaxed),
|
|
|
|
fifo.CPBreakpoint.load(std::memory_order_relaxed),
|
|
|
|
fifo.bFF_GPReadEnable.load(std::memory_order_relaxed) ? "true" : "false",
|
|
|
|
fifo.bFF_BPEnable.load(std::memory_order_relaxed) ? "true" : "false",
|
|
|
|
fifo.bFF_BPInt.load(std::memory_order_relaxed) ? "true" : "false",
|
|
|
|
fifo.bFF_Breakpoint.load(std::memory_order_relaxed) ? "true" : "false",
|
|
|
|
fifo.bFF_GPLinkEnable.load(std::memory_order_relaxed) ? "true" : "false",
|
|
|
|
fifo.bFF_HiWatermarkInt.load(std::memory_order_relaxed) ? "true" : "false",
|
|
|
|
fifo.bFF_LoWatermarkInt.load(std::memory_order_relaxed) ? "true" : "false");
|
2017-01-27 09:44:31 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-06-07 12:55:52 +04:00
|
|
|
} // namespace CommandProcessor
|