MMIO: Port the VideoCommon PE MMIOs to the new interface.
This commit is contained in:
parent
4129b30494
commit
5b5dfb384e
|
@ -311,6 +311,7 @@ void InitHWMemFuncsWii()
|
||||||
void InitMMIO(MMIO::Mapping* mmio)
|
void InitMMIO(MMIO::Mapping* mmio)
|
||||||
{
|
{
|
||||||
g_video_backend->RegisterCPMMIO(mmio, 0xCC000000);
|
g_video_backend->RegisterCPMMIO(mmio, 0xCC000000);
|
||||||
|
g_video_backend->RegisterPEMMIO(mmio, 0xCC001000);
|
||||||
VideoInterface::RegisterMMIO(mmio, 0xCC002000);
|
VideoInterface::RegisterMMIO(mmio, 0xCC002000);
|
||||||
ProcessorInterface::RegisterMMIO(mmio, 0xCC003000);
|
ProcessorInterface::RegisterMMIO(mmio, 0xCC003000);
|
||||||
MemoryInterface::RegisterMMIO(mmio, 0xCC004000);
|
MemoryInterface::RegisterMMIO(mmio, 0xCC004000);
|
||||||
|
|
|
@ -16,6 +16,7 @@
|
||||||
#include "PixelEngine.h"
|
#include "PixelEngine.h"
|
||||||
#include "RenderBase.h"
|
#include "RenderBase.h"
|
||||||
#include "CommandProcessor.h"
|
#include "CommandProcessor.h"
|
||||||
|
#include "HW/MMIO.h"
|
||||||
#include "HW/ProcessorInterface.h"
|
#include "HW/ProcessorInterface.h"
|
||||||
#include "State.h"
|
#include "State.h"
|
||||||
|
|
||||||
|
@ -133,11 +134,6 @@ void DoState(PointerWrap &p)
|
||||||
p.Do(bbox_active);
|
p.Do(bbox_active);
|
||||||
}
|
}
|
||||||
|
|
||||||
void RegisterMMIO(MMIO::Mapping* mmio, u32 base)
|
|
||||||
{
|
|
||||||
// TODO
|
|
||||||
}
|
|
||||||
|
|
||||||
void UpdateInterrupts();
|
void UpdateInterrupts();
|
||||||
void UpdateTokenInterrupt(bool active);
|
void UpdateTokenInterrupt(bool active);
|
||||||
void UpdateFinishInterrupt(bool active);
|
void UpdateFinishInterrupt(bool active);
|
||||||
|
@ -169,145 +165,61 @@ void Init()
|
||||||
bbox_active = false;
|
bbox_active = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
void Read16(u16& _uReturnValue, const u32 _iAddress)
|
void RegisterMMIO(MMIO::Mapping* mmio, u32 base)
|
||||||
{
|
{
|
||||||
DEBUG_LOG(PIXELENGINE, "(r16) 0x%08x", _iAddress);
|
// Directly mapped registers.
|
||||||
switch (_iAddress & 0xFFF)
|
struct {
|
||||||
|
u32 addr;
|
||||||
|
u16* ptr;
|
||||||
|
} directly_mapped_vars[] = {
|
||||||
|
{ PE_ZCONF, &m_ZConf.Hex },
|
||||||
|
{ PE_ALPHACONF, &m_AlphaConf.Hex },
|
||||||
|
{ PE_DSTALPHACONF, &m_DstAlphaConf.Hex },
|
||||||
|
{ PE_ALPHAMODE, &m_AlphaModeConf.Hex },
|
||||||
|
{ PE_ALPHAREAD, &m_AlphaRead.Hex },
|
||||||
|
};
|
||||||
|
for (auto& mapped_var : directly_mapped_vars)
|
||||||
{
|
{
|
||||||
// CPU Direct Access EFB Raster State Config
|
mmio->Register(base | mapped_var.addr,
|
||||||
case PE_ZCONF:
|
MMIO::DirectRead<u16>(mapped_var.ptr),
|
||||||
_uReturnValue = m_ZConf.Hex;
|
MMIO::DirectWrite<u16>(mapped_var.ptr)
|
||||||
INFO_LOG(PIXELENGINE, "(r16) ZCONF");
|
);
|
||||||
break;
|
|
||||||
case PE_ALPHACONF:
|
|
||||||
// Most games read this early. no idea why.
|
|
||||||
_uReturnValue = m_AlphaConf.Hex;
|
|
||||||
INFO_LOG(PIXELENGINE, "(r16) ALPHACONF");
|
|
||||||
break;
|
|
||||||
case PE_DSTALPHACONF:
|
|
||||||
_uReturnValue = m_DstAlphaConf.Hex;
|
|
||||||
INFO_LOG(PIXELENGINE, "(r16) DSTALPHACONF");
|
|
||||||
break;
|
|
||||||
case PE_ALPHAMODE:
|
|
||||||
_uReturnValue = m_AlphaModeConf.Hex;
|
|
||||||
INFO_LOG(PIXELENGINE, "(r16) ALPHAMODE");
|
|
||||||
break;
|
|
||||||
case PE_ALPHAREAD:
|
|
||||||
_uReturnValue = m_AlphaRead.Hex;
|
|
||||||
WARN_LOG(PIXELENGINE, "(r16) ALPHAREAD");
|
|
||||||
break;
|
|
||||||
|
|
||||||
case PE_CTRL_REGISTER:
|
|
||||||
_uReturnValue = m_Control.Hex;
|
|
||||||
INFO_LOG(PIXELENGINE, "(r16) CTRL_REGISTER : %04x", _uReturnValue);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case PE_TOKEN_REG:
|
|
||||||
_uReturnValue = Common::AtomicLoad(*(volatile u32*)&CommandProcessor::fifo.PEToken);
|
|
||||||
INFO_LOG(PIXELENGINE, "(r16) TOKEN_REG : %04x", _uReturnValue);
|
|
||||||
break;
|
|
||||||
|
|
||||||
// BBox
|
|
||||||
case PE_BBOX_LEFT:
|
|
||||||
case PE_BBOX_RIGHT:
|
|
||||||
case PE_BBOX_TOP:
|
|
||||||
case PE_BBOX_BOTTOM:
|
|
||||||
_uReturnValue = bbox[(_iAddress >> 1) & 3];
|
|
||||||
bbox_active = false;
|
|
||||||
break;
|
|
||||||
|
|
||||||
// NOTE(neobrain): only PE_PERF_ZCOMP_OUTPUT is implemented in D3D11, but the other values shouldn't be contradictionary to the value of that register (i.e. INPUT registers should always be greater or equal to their corresponding OUTPUT registers).
|
|
||||||
case PE_PERF_ZCOMP_INPUT_ZCOMPLOC_L:
|
|
||||||
_uReturnValue = g_video_backend->Video_GetQueryResult(PQ_ZCOMP_INPUT_ZCOMPLOC) & 0xFFFF;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case PE_PERF_ZCOMP_INPUT_ZCOMPLOC_H:
|
|
||||||
_uReturnValue = g_video_backend->Video_GetQueryResult(PQ_ZCOMP_INPUT_ZCOMPLOC) >> 16;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case PE_PERF_ZCOMP_OUTPUT_ZCOMPLOC_L:
|
|
||||||
_uReturnValue = g_video_backend->Video_GetQueryResult(PQ_ZCOMP_OUTPUT_ZCOMPLOC) & 0xFFFF;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case PE_PERF_ZCOMP_OUTPUT_ZCOMPLOC_H:
|
|
||||||
_uReturnValue = g_video_backend->Video_GetQueryResult(PQ_ZCOMP_OUTPUT_ZCOMPLOC) >> 16;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case PE_PERF_ZCOMP_INPUT_L:
|
|
||||||
_uReturnValue = g_video_backend->Video_GetQueryResult(PQ_ZCOMP_INPUT) & 0xFFFF;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case PE_PERF_ZCOMP_INPUT_H:
|
|
||||||
_uReturnValue = g_video_backend->Video_GetQueryResult(PQ_ZCOMP_INPUT) >> 16;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case PE_PERF_ZCOMP_OUTPUT_L:
|
|
||||||
_uReturnValue = g_video_backend->Video_GetQueryResult(PQ_ZCOMP_OUTPUT) & 0xFFFF;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case PE_PERF_ZCOMP_OUTPUT_H:
|
|
||||||
_uReturnValue = g_video_backend->Video_GetQueryResult(PQ_ZCOMP_OUTPUT) >> 16;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case PE_PERF_BLEND_INPUT_L:
|
|
||||||
// Super Mario Sunshine uses this register in episode 6 of Sirena Beach:
|
|
||||||
// The amount of remaining goop is determined by checking how many pixels reach the blending stage.
|
|
||||||
// Once this register falls below a particular value (around 0x90), the game regards the challenge finished.
|
|
||||||
// In very old builds, Dolphin only returned 0. That caused the challenge to be immediately finished without any goop being cleaned (the timer just didn't even start counting from 3:00:00).
|
|
||||||
// Later builds returned 1 for the high register. That caused the timer to actually count down, but made the challenge unbeatable because the game always thought you didn't clear any goop at all.
|
|
||||||
// Note that currently this functionality is only implemented in the D3D11 backend.
|
|
||||||
_uReturnValue = g_video_backend->Video_GetQueryResult(PQ_BLEND_INPUT) & 0xFFFF;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case PE_PERF_BLEND_INPUT_H:
|
|
||||||
_uReturnValue = g_video_backend->Video_GetQueryResult(PQ_BLEND_INPUT) >> 16;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case PE_PERF_EFB_COPY_CLOCKS_L:
|
|
||||||
_uReturnValue = g_video_backend->Video_GetQueryResult(PQ_EFB_COPY_CLOCKS) & 0xFFFF;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case PE_PERF_EFB_COPY_CLOCKS_H:
|
|
||||||
_uReturnValue = g_video_backend->Video_GetQueryResult(PQ_EFB_COPY_CLOCKS) >> 16;
|
|
||||||
break;
|
|
||||||
|
|
||||||
default:
|
|
||||||
INFO_LOG(PIXELENGINE, "(r16) unknown @ %08x", _iAddress);
|
|
||||||
_uReturnValue = 1;
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
// Performance queries registers: read only, need to call the video backend
|
||||||
|
// to get the results.
|
||||||
void Write16(const u16 _iValue, const u32 _iAddress)
|
struct {
|
||||||
{
|
u32 addr;
|
||||||
switch (_iAddress & 0xFFF)
|
PerfQueryType pqtype;
|
||||||
|
} pq_regs[] = {
|
||||||
|
{ PE_PERF_ZCOMP_INPUT_ZCOMPLOC_L, PQ_ZCOMP_INPUT_ZCOMPLOC },
|
||||||
|
{ PE_PERF_ZCOMP_OUTPUT_ZCOMPLOC_L, PQ_ZCOMP_OUTPUT_ZCOMPLOC },
|
||||||
|
{ PE_PERF_ZCOMP_INPUT_L, PQ_ZCOMP_INPUT },
|
||||||
|
{ PE_PERF_ZCOMP_OUTPUT_L, PQ_ZCOMP_OUTPUT },
|
||||||
|
{ PE_PERF_BLEND_INPUT_L, PQ_BLEND_INPUT },
|
||||||
|
{ PE_PERF_EFB_COPY_CLOCKS_L, PQ_EFB_COPY_CLOCKS },
|
||||||
|
};
|
||||||
|
for (auto& pq_reg : pq_regs)
|
||||||
{
|
{
|
||||||
// CPU Direct Access EFB Raster State Config
|
mmio->Register(base | pq_reg.addr,
|
||||||
case PE_ZCONF:
|
MMIO::ComplexRead<u16>([pq_reg](u32) {
|
||||||
m_ZConf.Hex = _iValue;
|
return g_video_backend->Video_GetQueryResult(pq_reg.pqtype) & 0xFFFF;
|
||||||
INFO_LOG(PIXELENGINE, "(w16) ZCONF: %02x", _iValue);
|
}),
|
||||||
break;
|
MMIO::InvalidWrite<u16>()
|
||||||
case PE_ALPHACONF:
|
);
|
||||||
m_AlphaConf.Hex = _iValue;
|
mmio->Register(base | (pq_reg.addr + 2),
|
||||||
INFO_LOG(PIXELENGINE, "(w16) ALPHACONF: %02x", _iValue);
|
MMIO::ComplexRead<u16>([pq_reg](u32) {
|
||||||
break;
|
return g_video_backend->Video_GetQueryResult(pq_reg.pqtype) >> 16;
|
||||||
case PE_DSTALPHACONF:
|
}),
|
||||||
m_DstAlphaConf.Hex = _iValue;
|
MMIO::InvalidWrite<u16>()
|
||||||
INFO_LOG(PIXELENGINE, "(w16) DSTALPHACONF: %02x", _iValue);
|
);
|
||||||
break;
|
}
|
||||||
case PE_ALPHAMODE:
|
|
||||||
m_AlphaModeConf.Hex = _iValue;
|
|
||||||
INFO_LOG(PIXELENGINE, "(w16) ALPHAMODE: %02x", _iValue);
|
|
||||||
break;
|
|
||||||
case PE_ALPHAREAD:
|
|
||||||
m_AlphaRead.Hex = _iValue;
|
|
||||||
INFO_LOG(PIXELENGINE, "(w16) ALPHAREAD: %02x", _iValue);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case PE_CTRL_REGISTER:
|
// Control register
|
||||||
{
|
mmio->Register(base | PE_CTRL_REGISTER,
|
||||||
UPECtrlReg tmpCtrl(_iValue);
|
MMIO::DirectRead<u16>(&m_Control.Hex),
|
||||||
|
MMIO::ComplexWrite<u16>([](u32, u16 val) {
|
||||||
|
UPECtrlReg tmpCtrl(val);
|
||||||
|
|
||||||
if (tmpCtrl.PEToken) g_bSignalTokenInterrupt = 0;
|
if (tmpCtrl.PEToken) g_bSignalTokenInterrupt = 0;
|
||||||
if (tmpCtrl.PEFinish) g_bSignalFinishInterrupt = 0;
|
if (tmpCtrl.PEFinish) g_bSignalFinishInterrupt = 0;
|
||||||
|
@ -317,27 +229,46 @@ void Write16(const u16 _iValue, const u32 _iAddress)
|
||||||
m_Control.PEToken = 0; // this flag is write only
|
m_Control.PEToken = 0; // this flag is write only
|
||||||
m_Control.PEFinish = 0; // this flag is write only
|
m_Control.PEFinish = 0; // this flag is write only
|
||||||
|
|
||||||
DEBUG_LOG(PIXELENGINE, "(w16) CTRL_REGISTER: 0x%04x", _iValue);
|
DEBUG_LOG(PIXELENGINE, "(w16) CTRL_REGISTER: 0x%04x", val);
|
||||||
UpdateInterrupts();
|
UpdateInterrupts();
|
||||||
}
|
})
|
||||||
break;
|
);
|
||||||
|
|
||||||
case PE_TOKEN_REG:
|
// Token register, readonly.
|
||||||
PanicAlert("(w16) WTF? PowerPC program wrote token: %i", _iValue);
|
mmio->Register(base | PE_TOKEN_REG,
|
||||||
//only the gx pipeline is supposed to be able to write here
|
MMIO::DirectRead<u16>(&CommandProcessor::fifo.PEToken),
|
||||||
//g_token = _iValue;
|
MMIO::InvalidWrite<u16>()
|
||||||
break;
|
);
|
||||||
|
|
||||||
default:
|
// BBOX registers, readonly and need to update a flag.
|
||||||
WARN_LOG(PIXELENGINE, "(w16) unknown %04x @ %08x", _iValue, _iAddress);
|
for (int i = 0; i < 4; ++i)
|
||||||
break;
|
{
|
||||||
|
mmio->Register(base | (PE_BBOX_LEFT + 2 * i),
|
||||||
|
MMIO::ComplexRead<u16>([i](u32) {
|
||||||
|
bbox_active = false;
|
||||||
|
return bbox[i];
|
||||||
|
}),
|
||||||
|
MMIO::InvalidWrite<u16>()
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void Read16(u16& _uReturnValue, const u32 _iAddress)
|
||||||
|
{
|
||||||
|
// HACK: Remove this function when the new MMIO interface is used.
|
||||||
|
Memory::mmio_mapping->Read(_iAddress, _uReturnValue);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Write16(const u16 _iValue, const u32 _iAddress)
|
||||||
|
{
|
||||||
|
// HACK: Remove this function when the new MMIO interface is used.
|
||||||
|
Memory::mmio_mapping->Write(_iAddress, _iValue);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Write32(const u32 _iValue, const u32 _iAddress)
|
void Write32(const u32 _iValue, const u32 _iAddress)
|
||||||
{
|
{
|
||||||
WARN_LOG(PIXELENGINE, "(w32) 0x%08x @ 0x%08x IGNORING...",_iValue,_iAddress);
|
// HACK: Remove this function when the new MMIO interface is used.
|
||||||
|
Memory::mmio_mapping->Write(_iAddress, _iValue);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool AllowIdleSkipping()
|
bool AllowIdleSkipping()
|
||||||
|
|
Loading…
Reference in New Issue