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
|
|
|
|
2014-02-10 13:54:46 -05:00
|
|
|
#pragma once
|
2010-06-09 01:37:08 +00:00
|
|
|
|
2022-12-10 16:35:07 +01:00
|
|
|
#include <mutex>
|
|
|
|
|
|
|
|
#include "Common/BitField.h"
|
2014-02-17 05:18:15 -05:00
|
|
|
#include "Common/CommonTypes.h"
|
2022-01-29 13:28:17 -08:00
|
|
|
|
2009-10-10 21:19:39 +00:00
|
|
|
class PointerWrap;
|
2022-01-29 13:28:17 -08:00
|
|
|
|
2022-12-10 16:35:07 +01:00
|
|
|
namespace Core
|
|
|
|
{
|
|
|
|
class System;
|
|
|
|
}
|
|
|
|
namespace CoreTiming
|
|
|
|
{
|
|
|
|
struct EventType;
|
|
|
|
}
|
2014-02-02 14:16:43 +01:00
|
|
|
namespace MMIO
|
|
|
|
{
|
|
|
|
class Mapping;
|
|
|
|
}
|
2010-06-09 01:37:08 +00:00
|
|
|
|
2017-02-28 06:57:31 -05:00
|
|
|
namespace PixelEngine
|
|
|
|
{
|
2009-10-10 21:19:39 +00:00
|
|
|
// internal hardware addresses
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
PE_ZCONF = 0x00, // Z Config
|
|
|
|
PE_ALPHACONF = 0x02, // Alpha Config
|
|
|
|
PE_DSTALPHACONF = 0x04, // Destination Alpha Config
|
|
|
|
PE_ALPHAMODE = 0x06, // Alpha Mode Config
|
|
|
|
PE_ALPHAREAD = 0x08, // Alpha Read
|
|
|
|
PE_CTRL_REGISTER = 0x0a, // Control
|
|
|
|
PE_TOKEN_REG = 0x0e, // Token
|
2014-09-14 17:52:51 +01:00
|
|
|
PE_BBOX_LEFT = 0x10, // Bounding Box Left Pixel
|
|
|
|
PE_BBOX_RIGHT = 0x12, // Bounding Box Right Pixel
|
|
|
|
PE_BBOX_TOP = 0x14, // Bounding Box Top Pixel
|
|
|
|
PE_BBOX_BOTTOM = 0x16, // Bounding Box Bottom Pixel
|
2010-06-09 01:37:08 +00:00
|
|
|
|
2012-05-29 13:54:20 +02:00
|
|
|
// NOTE: Order not verified
|
|
|
|
// These indicate the number of quads that are being used as input/output for each particular
|
|
|
|
// stage
|
|
|
|
PE_PERF_ZCOMP_INPUT_ZCOMPLOC_L = 0x18,
|
|
|
|
PE_PERF_ZCOMP_INPUT_ZCOMPLOC_H = 0x1a,
|
|
|
|
PE_PERF_ZCOMP_OUTPUT_ZCOMPLOC_L = 0x1c,
|
|
|
|
PE_PERF_ZCOMP_OUTPUT_ZCOMPLOC_H = 0x1e,
|
|
|
|
PE_PERF_ZCOMP_INPUT_L = 0x20,
|
|
|
|
PE_PERF_ZCOMP_INPUT_H = 0x22,
|
|
|
|
PE_PERF_ZCOMP_OUTPUT_L = 0x24,
|
|
|
|
PE_PERF_ZCOMP_OUTPUT_H = 0x26,
|
|
|
|
PE_PERF_BLEND_INPUT_L = 0x28,
|
|
|
|
PE_PERF_BLEND_INPUT_H = 0x2a,
|
|
|
|
PE_PERF_EFB_COPY_CLOCKS_L = 0x2c,
|
|
|
|
PE_PERF_EFB_COPY_CLOCKS_H = 0x2e,
|
2009-10-10 21:19:39 +00:00
|
|
|
};
|
2010-06-09 01:37:08 +00:00
|
|
|
|
2010-10-24 19:52:52 +00:00
|
|
|
// ReadMode specifies the returned alpha channel for EFB peeks
|
2022-01-29 13:28:17 -08:00
|
|
|
enum class AlphaReadMode : u16
|
2017-01-04 12:45:40 +01:00
|
|
|
{
|
2022-01-29 13:28:17 -08:00
|
|
|
Read00 = 0, // Always read 0x00
|
|
|
|
ReadFF = 1, // Always read 0xFF
|
|
|
|
ReadNone = 2, // Always read the real alpha value
|
2010-10-24 19:52:52 +00:00
|
|
|
};
|
|
|
|
|
2022-12-10 16:35:07 +01:00
|
|
|
// Note: These enums are (assumed to be) identical to the one in BPMemory, but the base type is set
|
|
|
|
// to u16 instead of u32 for BitField
|
|
|
|
enum class CompareMode : u16
|
|
|
|
{
|
|
|
|
Never = 0,
|
|
|
|
Less = 1,
|
|
|
|
Equal = 2,
|
|
|
|
LEqual = 3,
|
|
|
|
Greater = 4,
|
|
|
|
NEqual = 5,
|
|
|
|
GEqual = 6,
|
|
|
|
Always = 7
|
|
|
|
};
|
|
|
|
|
|
|
|
union UPEZConfReg
|
|
|
|
{
|
|
|
|
u16 hex = 0;
|
|
|
|
BitField<0, 1, bool, u16> z_comparator_enable;
|
|
|
|
BitField<1, 3, CompareMode, u16> function;
|
|
|
|
BitField<4, 1, bool, u16> z_update_enable;
|
|
|
|
};
|
|
|
|
|
|
|
|
enum class SrcBlendFactor : u16
|
|
|
|
{
|
|
|
|
Zero = 0,
|
|
|
|
One = 1,
|
|
|
|
DstClr = 2,
|
|
|
|
InvDstClr = 3,
|
|
|
|
SrcAlpha = 4,
|
|
|
|
InvSrcAlpha = 5,
|
|
|
|
DstAlpha = 6,
|
|
|
|
InvDstAlpha = 7
|
|
|
|
};
|
2010-06-09 01:37:08 +00:00
|
|
|
|
2022-12-10 16:35:07 +01:00
|
|
|
enum class DstBlendFactor : u16
|
|
|
|
{
|
|
|
|
Zero = 0,
|
|
|
|
One = 1,
|
|
|
|
SrcClr = 2,
|
|
|
|
InvSrcClr = 3,
|
|
|
|
SrcAlpha = 4,
|
|
|
|
InvSrcAlpha = 5,
|
|
|
|
DstAlpha = 6,
|
|
|
|
InvDstAlpha = 7
|
|
|
|
};
|
2014-02-02 14:16:43 +01:00
|
|
|
|
2022-12-10 16:35:07 +01:00
|
|
|
enum class LogicOp : u16
|
|
|
|
{
|
|
|
|
Clear = 0,
|
|
|
|
And = 1,
|
|
|
|
AndReverse = 2,
|
|
|
|
Copy = 3,
|
|
|
|
AndInverted = 4,
|
|
|
|
NoOp = 5,
|
|
|
|
Xor = 6,
|
|
|
|
Or = 7,
|
|
|
|
Nor = 8,
|
|
|
|
Equiv = 9,
|
|
|
|
Invert = 10,
|
|
|
|
OrReverse = 11,
|
|
|
|
CopyInverted = 12,
|
|
|
|
OrInverted = 13,
|
|
|
|
Nand = 14,
|
|
|
|
Set = 15
|
|
|
|
};
|
|
|
|
|
|
|
|
union UPEAlphaConfReg
|
|
|
|
{
|
|
|
|
u16 hex = 0;
|
|
|
|
BitField<0, 1, bool, u16> blend; // Set for GX_BM_BLEND or GX_BM_SUBTRACT
|
|
|
|
BitField<1, 1, bool, u16> logic; // Set for GX_BM_LOGIC
|
|
|
|
BitField<2, 1, bool, u16> dither;
|
|
|
|
BitField<3, 1, bool, u16> color_update_enable;
|
|
|
|
BitField<4, 1, bool, u16> alpha_update_enable;
|
|
|
|
BitField<5, 3, DstBlendFactor, u16> dst_factor;
|
|
|
|
BitField<8, 3, SrcBlendFactor, u16> src_factor;
|
|
|
|
BitField<11, 1, bool, u16> subtract; // Set for GX_BM_SUBTRACT
|
|
|
|
BitField<12, 4, LogicOp, u16> logic_op;
|
|
|
|
};
|
|
|
|
|
|
|
|
union UPEDstAlphaConfReg
|
|
|
|
{
|
|
|
|
u16 hex = 0;
|
|
|
|
BitField<0, 8, u8, u16> alpha;
|
|
|
|
BitField<8, 1, bool, u16> enable;
|
|
|
|
};
|
|
|
|
|
|
|
|
union UPEAlphaModeConfReg
|
|
|
|
{
|
|
|
|
u16 hex = 0;
|
|
|
|
BitField<0, 8, u8, u16> threshold;
|
|
|
|
// Yagcd and libogc use 8 bits for this, but the enum only needs 3
|
|
|
|
BitField<8, 3, CompareMode, u16> compare_mode;
|
|
|
|
};
|
|
|
|
|
|
|
|
union UPEAlphaReadReg
|
|
|
|
{
|
|
|
|
u16 hex = 0;
|
|
|
|
BitField<0, 2, AlphaReadMode, u16> read_mode;
|
|
|
|
};
|
|
|
|
|
|
|
|
// fifo Control Register
|
|
|
|
union UPECtrlReg
|
|
|
|
{
|
|
|
|
u16 hex = 0;
|
|
|
|
BitField<0, 1, bool, u16> pe_token_enable;
|
|
|
|
BitField<1, 1, bool, u16> pe_finish_enable;
|
|
|
|
BitField<2, 1, bool, u16> pe_token; // Write only
|
|
|
|
BitField<3, 1, bool, u16> pe_finish; // Write only
|
|
|
|
};
|
|
|
|
|
|
|
|
class PixelEngineManager
|
|
|
|
{
|
|
|
|
public:
|
2023-12-22 13:54:24 -05:00
|
|
|
explicit PixelEngineManager(Core::System& system);
|
|
|
|
|
|
|
|
void Init();
|
2022-12-10 16:35:07 +01:00
|
|
|
void DoState(PointerWrap& p);
|
|
|
|
|
|
|
|
void RegisterMMIO(MMIO::Mapping* mmio, u32 base);
|
|
|
|
|
|
|
|
// gfx backend support
|
2023-12-22 13:54:24 -05:00
|
|
|
void SetToken(const u16 token, const bool interrupt, int cycle_delay);
|
|
|
|
void SetFinish(int cycle_delay);
|
2022-12-10 16:35:07 +01:00
|
|
|
AlphaReadMode GetAlphaReadMode() const { return m_alpha_read.read_mode; }
|
|
|
|
|
|
|
|
private:
|
2023-12-22 13:54:24 -05:00
|
|
|
void RaiseEvent(int cycles_into_future);
|
2022-12-10 16:35:07 +01:00
|
|
|
void UpdateInterrupts();
|
2023-12-22 13:54:24 -05:00
|
|
|
void SetTokenFinish_OnMainThread(u64 userdata, s64 cycles_late);
|
2022-12-10 16:35:07 +01:00
|
|
|
|
|
|
|
static void SetTokenFinish_OnMainThread_Static(Core::System& system, u64 userdata,
|
|
|
|
s64 cycles_late);
|
|
|
|
|
|
|
|
// STATE_TO_SAVE
|
|
|
|
UPEZConfReg m_z_conf;
|
|
|
|
UPEAlphaConfReg m_alpha_conf;
|
|
|
|
UPEDstAlphaConfReg m_dst_alpha_conf;
|
|
|
|
UPEAlphaModeConfReg m_alpha_mode_conf;
|
|
|
|
UPEAlphaReadReg m_alpha_read;
|
|
|
|
UPECtrlReg m_control;
|
|
|
|
|
|
|
|
std::mutex m_token_finish_mutex;
|
|
|
|
u16 m_token = 0;
|
|
|
|
u16 m_token_pending = 0;
|
|
|
|
bool m_token_interrupt_pending = false;
|
|
|
|
bool m_finish_interrupt_pending = false;
|
|
|
|
bool m_event_raised = false;
|
|
|
|
|
|
|
|
bool m_signal_token_interrupt = false;
|
|
|
|
bool m_signal_finish_interrupt = false;
|
|
|
|
|
|
|
|
CoreTiming::EventType* m_event_type_set_token_finish = nullptr;
|
2023-12-22 13:54:24 -05:00
|
|
|
|
|
|
|
Core::System& m_system;
|
2022-12-10 16:35:07 +01:00
|
|
|
};
|
2010-06-09 01:37:08 +00:00
|
|
|
|
2021-06-07 12:55:52 +04:00
|
|
|
} // namespace PixelEngine
|