mirror of
https://github.com/dolphin-emu/dolphin.git
synced 2025-01-10 16:19:28 +01:00
Merge pull request #10448 from Pokechu22/prince-of-persia-rival-swords-unknown-opcode
Ignore unknown opcode for 0x3f
This commit is contained in:
commit
eed7d3b692
@ -29,7 +29,7 @@ namespace GPFifo
|
|||||||
// the same function could use both methods. Compile 2 different versions of each such block?
|
// the same function could use both methods. Compile 2 different versions of each such block?
|
||||||
|
|
||||||
// More room for the fastmodes
|
// More room for the fastmodes
|
||||||
alignas(32) static u8 s_gather_pipe[GATHER_PIPE_SIZE * 16];
|
alignas(GATHER_PIPE_SIZE) static u8 s_gather_pipe[GATHER_PIPE_EXTRA_SIZE];
|
||||||
|
|
||||||
static size_t GetGatherPipeCount()
|
static size_t GetGatherPipeCount()
|
||||||
{
|
{
|
||||||
|
@ -9,10 +9,8 @@ class PointerWrap;
|
|||||||
|
|
||||||
namespace GPFifo
|
namespace GPFifo
|
||||||
{
|
{
|
||||||
enum
|
constexpr u32 GATHER_PIPE_SIZE = 32;
|
||||||
{
|
constexpr u32 GATHER_PIPE_EXTRA_SIZE = GATHER_PIPE_SIZE * 16;
|
||||||
GATHER_PIPE_SIZE = 32
|
|
||||||
};
|
|
||||||
|
|
||||||
// Init
|
// Init
|
||||||
void Init();
|
void Init();
|
||||||
|
@ -1018,7 +1018,8 @@ bool Jit64::DoJit(u32 em_address, JitBlock* b, u32 nextPC)
|
|||||||
bool gatherPipeIntCheck = js.fifoWriteAddresses.find(op.address) != js.fifoWriteAddresses.end();
|
bool gatherPipeIntCheck = js.fifoWriteAddresses.find(op.address) != js.fifoWriteAddresses.end();
|
||||||
|
|
||||||
// Gather pipe writes using an immediate address are explicitly tracked.
|
// Gather pipe writes using an immediate address are explicitly tracked.
|
||||||
if (jo.optimizeGatherPipe && (js.fifoBytesSinceCheck >= 32 || js.mustCheckFifo))
|
if (jo.optimizeGatherPipe &&
|
||||||
|
(js.fifoBytesSinceCheck >= GPFifo::GATHER_PIPE_SIZE || js.mustCheckFifo))
|
||||||
{
|
{
|
||||||
js.fifoBytesSinceCheck = 0;
|
js.fifoBytesSinceCheck = 0;
|
||||||
js.mustCheckFifo = false;
|
js.mustCheckFifo = false;
|
||||||
|
@ -835,7 +835,8 @@ bool JitArm64::DoJit(u32 em_address, JitBlock* b, u32 nextPC)
|
|||||||
// Gather pipe writes using a non-immediate address are discovered by profiling.
|
// Gather pipe writes using a non-immediate address are discovered by profiling.
|
||||||
bool gatherPipeIntCheck = js.fifoWriteAddresses.find(op.address) != js.fifoWriteAddresses.end();
|
bool gatherPipeIntCheck = js.fifoWriteAddresses.find(op.address) != js.fifoWriteAddresses.end();
|
||||||
|
|
||||||
if (jo.optimizeGatherPipe && (js.fifoBytesSinceCheck >= 32 || js.mustCheckFifo))
|
if (jo.optimizeGatherPipe &&
|
||||||
|
(js.fifoBytesSinceCheck >= GPFifo::GATHER_PIPE_SIZE || js.mustCheckFifo))
|
||||||
{
|
{
|
||||||
js.fifoBytesSinceCheck = 0;
|
js.fifoBytesSinceCheck = 0;
|
||||||
js.mustCheckFifo = false;
|
js.mustCheckFifo = false;
|
||||||
|
@ -17,6 +17,7 @@
|
|||||||
#include "Core/HW/GPFifo.h"
|
#include "Core/HW/GPFifo.h"
|
||||||
#include "Core/HW/MMIO.h"
|
#include "Core/HW/MMIO.h"
|
||||||
#include "Core/HW/ProcessorInterface.h"
|
#include "Core/HW/ProcessorInterface.h"
|
||||||
|
#include "Core/PowerPC/PowerPC.h"
|
||||||
#include "Core/System.h"
|
#include "Core/System.h"
|
||||||
#include "VideoCommon/Fifo.h"
|
#include "VideoCommon/Fifo.h"
|
||||||
|
|
||||||
@ -382,7 +383,7 @@ void GatherPipeBursted()
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
fifo.CPWritePointer.fetch_add(GATHER_PIPE_SIZE, std::memory_order_relaxed);
|
fifo.CPWritePointer.fetch_add(GPFifo::GATHER_PIPE_SIZE, std::memory_order_relaxed);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (m_CPCtrlReg.GPReadEnable && m_CPCtrlReg.GPLinkEnable)
|
if (m_CPCtrlReg.GPReadEnable && m_CPCtrlReg.GPLinkEnable)
|
||||||
@ -396,7 +397,7 @@ void GatherPipeBursted()
|
|||||||
if (fifo.bFF_HiWatermark.load(std::memory_order_relaxed) != 0)
|
if (fifo.bFF_HiWatermark.load(std::memory_order_relaxed) != 0)
|
||||||
CoreTiming::ForceExceptionCheck(0);
|
CoreTiming::ForceExceptionCheck(0);
|
||||||
|
|
||||||
fifo.CPReadWriteDistance.fetch_add(GATHER_PIPE_SIZE, std::memory_order_seq_cst);
|
fifo.CPReadWriteDistance.fetch_add(GPFifo::GATHER_PIPE_SIZE, std::memory_order_seq_cst);
|
||||||
|
|
||||||
Fifo::RunGpu();
|
Fifo::RunGpu();
|
||||||
|
|
||||||
@ -615,12 +616,19 @@ void SetCpClearRegister()
|
|||||||
|
|
||||||
void HandleUnknownOpcode(u8 cmd_byte, const u8* buffer, bool preprocess)
|
void HandleUnknownOpcode(u8 cmd_byte, const u8* buffer, bool preprocess)
|
||||||
{
|
{
|
||||||
// Datel software uses 0x01 during startup, and Mario Party 5's Wiggler capsule
|
// Datel software uses 0x01 during startup, and Mario Party 5's Wiggler capsule accidentally uses
|
||||||
// accidentally uses 0x01-0x03 due to sending 4 more vertices than intended.
|
// 0x01-0x03 due to sending 4 more vertices than intended (see https://dolp.in/i8104).
|
||||||
// Hardware testing indicates that 0x01-0x07 do nothing, so to avoid annoying the user with
|
// Prince of Persia: Rival Swords sends 0x3f if the home menu is opened during the intro cutscene
|
||||||
|
// due to a game bug resulting in an incorrect vertex desc that results in the float value 1.0,
|
||||||
|
// encoded as 0x3f800000, being parsed as an opcode (see https://dolp.in/i9203).
|
||||||
|
//
|
||||||
|
// Hardware testing indicates that these opcodes do nothing, so to avoid annoying the user with
|
||||||
// spurious popups, we don't create a panic alert in those cases. Other unknown opcodes
|
// spurious popups, we don't create a panic alert in those cases. Other unknown opcodes
|
||||||
// (such as 0x18) seem to result in hangs.
|
// (such as 0x18) seem to result in actual hangs on real hardware, so the alert still is important
|
||||||
if (!s_is_fifo_error_seen && cmd_byte > 0x07)
|
// to keep around for unexpected cases.
|
||||||
|
const bool suppress_panic_alert = (cmd_byte <= 0x7) || (cmd_byte == 0x3f);
|
||||||
|
|
||||||
|
if (!s_is_fifo_error_seen && !suppress_panic_alert)
|
||||||
{
|
{
|
||||||
s_is_fifo_error_seen = true;
|
s_is_fifo_error_seen = true;
|
||||||
|
|
||||||
@ -634,26 +642,28 @@ void HandleUnknownOpcode(u8 cmd_byte, const u8* buffer, bool preprocess)
|
|||||||
"Further errors will be sent to the Video Backend log and\n"
|
"Further errors will be sent to the Video Backend log and\n"
|
||||||
"Dolphin will now likely crash or hang. Enjoy.",
|
"Dolphin will now likely crash or hang. Enjoy.",
|
||||||
cmd_byte, fmt::ptr(buffer), preprocess);
|
cmd_byte, fmt::ptr(buffer), preprocess);
|
||||||
|
}
|
||||||
|
|
||||||
PanicAlertFmt("Illegal command {:02x}\n"
|
// We always generate this log message, though we only generate the panic alerts once.
|
||||||
"CPBase: {:#010x}\n"
|
//
|
||||||
"CPEnd: {:#010x}\n"
|
// PC and LR are generally inaccurate in dual-core and are still misleading in single-core
|
||||||
"CPHiWatermark: {:#010x}\n"
|
// due to the gather pipe queueing data. Changing GATHER_PIPE_SIZE to 1 and
|
||||||
"CPLoWatermark: {:#010x}\n"
|
// GATHER_PIPE_EXTRA_SIZE to 16 * 32 in GPFifo.h, and using the cached interpreter CPU emulation
|
||||||
"CPReadWriteDistance: {:#010x}\n"
|
// engine, can result in more accurate information (though it is still a bit delayed).
|
||||||
"CPWritePointer: {:#010x}\n"
|
// PC and LR are meaningless when using the fifoplayer, and will generally not be helpful if the
|
||||||
"CPReadPointer: {:#010x}\n"
|
// unknown opcode is inside of a display list. Also note that the changes in GPFifo.h are not
|
||||||
"CPBreakpoint: {:#010x}\n"
|
// accurate and may introduce timing issues.
|
||||||
"bFF_GPReadEnable: {}\n"
|
ERROR_LOG_FMT(VIDEO,
|
||||||
"bFF_BPEnable: {}\n"
|
"FIFO: Unknown Opcode {:#04x} @ {}, preprocessing = {}, CPBase: {:#010x}, CPEnd: "
|
||||||
"bFF_BPInt: {}\n"
|
"{:#010x}, CPHiWatermark: {:#010x}, CPLoWatermark: {:#010x}, CPReadWriteDistance: "
|
||||||
"bFF_Breakpoint: {}\n"
|
"{:#010x}, CPWritePointer: {:#010x}, CPReadPointer: {:#010x}, CPBreakpoint: "
|
||||||
"bFF_GPLinkEnable: {}\n"
|
"{:#010x}, bFF_GPReadEnable: {}, bFF_BPEnable: {}, bFF_BPInt: {}, bFF_Breakpoint: "
|
||||||
"bFF_HiWatermarkInt: {}\n"
|
"{}, bFF_GPLinkEnable: {}, bFF_HiWatermarkInt: {}, bFF_LoWatermarkInt: {}, "
|
||||||
"bFF_LoWatermarkInt: {}\n",
|
"approximate PC: {:08x}, approximate LR: {:08x}",
|
||||||
cmd_byte, fifo.CPBase.load(std::memory_order_relaxed),
|
cmd_byte, fmt::ptr(buffer), preprocess ? "yes" : "no",
|
||||||
fifo.CPEnd.load(std::memory_order_relaxed), fifo.CPHiWatermark,
|
fifo.CPBase.load(std::memory_order_relaxed),
|
||||||
fifo.CPLoWatermark, fifo.CPReadWriteDistance.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.CPWritePointer.load(std::memory_order_relaxed),
|
||||||
fifo.CPReadPointer.load(std::memory_order_relaxed),
|
fifo.CPReadPointer.load(std::memory_order_relaxed),
|
||||||
fifo.CPBreakpoint.load(std::memory_order_relaxed),
|
fifo.CPBreakpoint.load(std::memory_order_relaxed),
|
||||||
@ -663,12 +673,7 @@ void HandleUnknownOpcode(u8 cmd_byte, const u8* buffer, bool preprocess)
|
|||||||
fifo.bFF_Breakpoint.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_GPLinkEnable.load(std::memory_order_relaxed) ? "true" : "false",
|
||||||
fifo.bFF_HiWatermarkInt.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");
|
fifo.bFF_LoWatermarkInt.load(std::memory_order_relaxed) ? "true" : "false", PC, LR);
|
||||||
}
|
|
||||||
|
|
||||||
// We always generate this log message, though we only generate the panic alerts once.
|
|
||||||
ERROR_LOG_FMT(VIDEO, "FIFO: Unknown Opcode ({:#04x} @ {}, preprocessing = {})", cmd_byte,
|
|
||||||
fmt::ptr(buffer), preprocess ? "yes" : "no");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace CommandProcessor
|
} // namespace CommandProcessor
|
||||||
|
@ -97,7 +97,6 @@ enum
|
|||||||
|
|
||||||
enum
|
enum
|
||||||
{
|
{
|
||||||
GATHER_PIPE_SIZE = 32,
|
|
||||||
INT_CAUSE_CP = 0x800
|
INT_CAUSE_CP = 0x800
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -17,6 +17,7 @@
|
|||||||
#include "Core/Config/MainSettings.h"
|
#include "Core/Config/MainSettings.h"
|
||||||
#include "Core/ConfigManager.h"
|
#include "Core/ConfigManager.h"
|
||||||
#include "Core/CoreTiming.h"
|
#include "Core/CoreTiming.h"
|
||||||
|
#include "Core/HW/GPFifo.h"
|
||||||
#include "Core/HW/Memmap.h"
|
#include "Core/HW/Memmap.h"
|
||||||
#include "Core/Host.h"
|
#include "Core/Host.h"
|
||||||
#include "Core/System.h"
|
#include "Core/System.h"
|
||||||
@ -249,13 +250,14 @@ void* PopFifoAuxBuffer(size_t size)
|
|||||||
// Description: RunGpuLoop() sends data through this function.
|
// Description: RunGpuLoop() sends data through this function.
|
||||||
static void ReadDataFromFifo(u32 readPtr)
|
static void ReadDataFromFifo(u32 readPtr)
|
||||||
{
|
{
|
||||||
constexpr size_t len = 32;
|
if (GPFifo::GATHER_PIPE_SIZE >
|
||||||
if (len > static_cast<size_t>(s_video_buffer + FIFO_SIZE - s_video_buffer_write_ptr))
|
static_cast<size_t>(s_video_buffer + FIFO_SIZE - s_video_buffer_write_ptr))
|
||||||
{
|
{
|
||||||
const size_t existing_len = s_video_buffer_write_ptr - s_video_buffer_read_ptr;
|
const size_t existing_len = s_video_buffer_write_ptr - s_video_buffer_read_ptr;
|
||||||
if (len > static_cast<size_t>(FIFO_SIZE - existing_len))
|
if (GPFifo::GATHER_PIPE_SIZE > static_cast<size_t>(FIFO_SIZE - existing_len))
|
||||||
{
|
{
|
||||||
PanicAlertFmt("FIFO out of bounds (existing {} + new {} > {})", existing_len, len, FIFO_SIZE);
|
PanicAlertFmt("FIFO out of bounds (existing {} + new {} > {})", existing_len,
|
||||||
|
GPFifo::GATHER_PIPE_SIZE, FIFO_SIZE);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
memmove(s_video_buffer, s_video_buffer_read_ptr, existing_len);
|
memmove(s_video_buffer, s_video_buffer_read_ptr, existing_len);
|
||||||
@ -263,16 +265,15 @@ static void ReadDataFromFifo(u32 readPtr)
|
|||||||
s_video_buffer_read_ptr = s_video_buffer;
|
s_video_buffer_read_ptr = s_video_buffer;
|
||||||
}
|
}
|
||||||
// Copy new video instructions to s_video_buffer for future use in rendering the new picture
|
// Copy new video instructions to s_video_buffer for future use in rendering the new picture
|
||||||
Memory::CopyFromEmu(s_video_buffer_write_ptr, readPtr, len);
|
Memory::CopyFromEmu(s_video_buffer_write_ptr, readPtr, GPFifo::GATHER_PIPE_SIZE);
|
||||||
s_video_buffer_write_ptr += len;
|
s_video_buffer_write_ptr += GPFifo::GATHER_PIPE_SIZE;
|
||||||
}
|
}
|
||||||
|
|
||||||
// The deterministic_gpu_thread version.
|
// The deterministic_gpu_thread version.
|
||||||
static void ReadDataFromFifoOnCPU(u32 readPtr)
|
static void ReadDataFromFifoOnCPU(u32 readPtr)
|
||||||
{
|
{
|
||||||
constexpr size_t len = 32;
|
|
||||||
u8* write_ptr = s_video_buffer_write_ptr;
|
u8* write_ptr = s_video_buffer_write_ptr;
|
||||||
if (len > static_cast<size_t>(s_video_buffer + FIFO_SIZE - write_ptr))
|
if (GPFifo::GATHER_PIPE_SIZE > static_cast<size_t>(s_video_buffer + FIFO_SIZE - write_ptr))
|
||||||
{
|
{
|
||||||
// We can't wrap around while the GPU is working on the data.
|
// We can't wrap around while the GPU is working on the data.
|
||||||
// This should be very rare due to the reset in SyncGPU.
|
// This should be very rare due to the reset in SyncGPU.
|
||||||
@ -290,17 +291,18 @@ static void ReadDataFromFifoOnCPU(u32 readPtr)
|
|||||||
}
|
}
|
||||||
write_ptr = s_video_buffer_write_ptr;
|
write_ptr = s_video_buffer_write_ptr;
|
||||||
const size_t existing_len = write_ptr - s_video_buffer_pp_read_ptr;
|
const size_t existing_len = write_ptr - s_video_buffer_pp_read_ptr;
|
||||||
if (len > static_cast<size_t>(FIFO_SIZE - existing_len))
|
if (GPFifo::GATHER_PIPE_SIZE > static_cast<size_t>(FIFO_SIZE - existing_len))
|
||||||
{
|
{
|
||||||
PanicAlertFmt("FIFO out of bounds (existing {} + new {} > {})", existing_len, len, FIFO_SIZE);
|
PanicAlertFmt("FIFO out of bounds (existing {} + new {} > {})", existing_len,
|
||||||
|
GPFifo::GATHER_PIPE_SIZE, FIFO_SIZE);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Memory::CopyFromEmu(s_video_buffer_write_ptr, readPtr, len);
|
Memory::CopyFromEmu(s_video_buffer_write_ptr, readPtr, GPFifo::GATHER_PIPE_SIZE);
|
||||||
s_video_buffer_pp_read_ptr = OpcodeDecoder::RunFifo<true>(
|
s_video_buffer_pp_read_ptr = OpcodeDecoder::RunFifo<true>(
|
||||||
DataReader(s_video_buffer_pp_read_ptr, write_ptr + len), nullptr);
|
DataReader(s_video_buffer_pp_read_ptr, write_ptr + GPFifo::GATHER_PIPE_SIZE), nullptr);
|
||||||
// This would have to be locked if the GPU thread didn't spin.
|
// This would have to be locked if the GPU thread didn't spin.
|
||||||
s_video_buffer_write_ptr = write_ptr + len;
|
s_video_buffer_write_ptr = write_ptr + GPFifo::GATHER_PIPE_SIZE;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ResetVideoBuffer()
|
void ResetVideoBuffer()
|
||||||
@ -362,20 +364,22 @@ void RunGpuLoop()
|
|||||||
if (readPtr == fifo.CPEnd.load(std::memory_order_relaxed))
|
if (readPtr == fifo.CPEnd.load(std::memory_order_relaxed))
|
||||||
readPtr = fifo.CPBase.load(std::memory_order_relaxed);
|
readPtr = fifo.CPBase.load(std::memory_order_relaxed);
|
||||||
else
|
else
|
||||||
readPtr += 32;
|
readPtr += GPFifo::GATHER_PIPE_SIZE;
|
||||||
|
|
||||||
ASSERT_MSG(COMMANDPROCESSOR,
|
const s32 distance =
|
||||||
(s32)fifo.CPReadWriteDistance.load(std::memory_order_relaxed) - 32 >= 0,
|
static_cast<s32>(fifo.CPReadWriteDistance.load(std::memory_order_relaxed)) -
|
||||||
|
GPFifo::GATHER_PIPE_SIZE;
|
||||||
|
ASSERT_MSG(COMMANDPROCESSOR, distance >= 0,
|
||||||
"Negative fifo.CPReadWriteDistance = {} in FIFO Loop !\nThat can produce "
|
"Negative fifo.CPReadWriteDistance = {} in FIFO Loop !\nThat can produce "
|
||||||
"instability in the game. Please report it.",
|
"instability in the game. Please report it.",
|
||||||
fifo.CPReadWriteDistance.load(std::memory_order_relaxed) - 32);
|
distance);
|
||||||
|
|
||||||
u8* write_ptr = s_video_buffer_write_ptr;
|
u8* write_ptr = s_video_buffer_write_ptr;
|
||||||
s_video_buffer_read_ptr = OpcodeDecoder::RunFifo(
|
s_video_buffer_read_ptr = OpcodeDecoder::RunFifo(
|
||||||
DataReader(s_video_buffer_read_ptr, write_ptr), &cyclesExecuted);
|
DataReader(s_video_buffer_read_ptr, write_ptr), &cyclesExecuted);
|
||||||
|
|
||||||
fifo.CPReadPointer.store(readPtr, std::memory_order_relaxed);
|
fifo.CPReadPointer.store(readPtr, std::memory_order_relaxed);
|
||||||
fifo.CPReadWriteDistance.fetch_sub(32, std::memory_order_seq_cst);
|
fifo.CPReadWriteDistance.fetch_sub(GPFifo::GATHER_PIPE_SIZE, std::memory_order_seq_cst);
|
||||||
if ((write_ptr - s_video_buffer_read_ptr) == 0)
|
if ((write_ptr - s_video_buffer_read_ptr) == 0)
|
||||||
{
|
{
|
||||||
fifo.SafeCPReadPointer.store(fifo.CPReadPointer.load(std::memory_order_relaxed),
|
fifo.SafeCPReadPointer.store(fifo.CPReadPointer.load(std::memory_order_relaxed),
|
||||||
@ -498,10 +502,10 @@ static int RunGpuOnCpu(int ticks)
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
fifo.CPReadPointer.fetch_add(32, std::memory_order_relaxed);
|
fifo.CPReadPointer.fetch_add(GPFifo::GATHER_PIPE_SIZE, std::memory_order_relaxed);
|
||||||
}
|
}
|
||||||
|
|
||||||
fifo.CPReadWriteDistance.fetch_sub(32, std::memory_order_relaxed);
|
fifo.CPReadWriteDistance.fetch_sub(GPFifo::GATHER_PIPE_SIZE, std::memory_order_relaxed);
|
||||||
}
|
}
|
||||||
|
|
||||||
CommandProcessor::SetCPStatusFromGPU();
|
CommandProcessor::SetCPStatusFromGPU();
|
||||||
|
Loading…
x
Reference in New Issue
Block a user