2013-06-24 14:03:25 +00:00
|
|
|
// Copyright (c) 2012- PPSSPP Project.
|
|
|
|
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License as published by
|
|
|
|
// the Free Software Foundation, version 2.0 or later versions.
|
|
|
|
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License 2.0 for more details.
|
|
|
|
|
|
|
|
// A copy of the GPL 2.0 should have been included with the program.
|
|
|
|
// If not, see http://www.gnu.org/licenses/
|
|
|
|
|
|
|
|
// Official git repository and contact information can be found at
|
|
|
|
// https://github.com/hrydgard/ppsspp and http://www.ppsspp.org/.
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
2022-02-12 23:27:18 +00:00
|
|
|
#include <cstdint>
|
2013-09-23 02:03:31 +00:00
|
|
|
#include "GPU/GPUCommon.h"
|
|
|
|
#include "GPU/Common/GPUDebugInterface.h"
|
2020-10-04 21:24:14 +00:00
|
|
|
#include "Common/GPU/thin3d.h"
|
2013-06-24 14:03:25 +00:00
|
|
|
|
2017-01-23 15:57:16 +00:00
|
|
|
struct FormatBuffer {
|
2017-05-24 08:20:10 +00:00
|
|
|
FormatBuffer() { data = nullptr; }
|
2013-09-14 18:36:56 +00:00
|
|
|
union {
|
|
|
|
u8 *data;
|
|
|
|
u16 *as16;
|
|
|
|
u32 *as32;
|
|
|
|
};
|
|
|
|
|
|
|
|
inline void Set16(int x, int y, int stride, u16 v) {
|
|
|
|
as16[x + y * stride] = v;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void Set32(int x, int y, int stride, u32 v) {
|
|
|
|
as32[x + y * stride] = v;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline u16 Get16(int x, int y, int stride) {
|
|
|
|
return as16[x + y * stride];
|
|
|
|
}
|
|
|
|
|
|
|
|
inline u32 Get32(int x, int y, int stride) {
|
|
|
|
return as32[x + y * stride];
|
|
|
|
}
|
2019-10-25 21:02:53 +00:00
|
|
|
|
|
|
|
inline u16 *Get16Ptr(int x, int y, int stride) {
|
|
|
|
return &as16[x + y * stride];
|
|
|
|
}
|
2021-11-15 14:09:12 +00:00
|
|
|
|
|
|
|
inline u32 *Get32Ptr(int x, int y, int stride) {
|
|
|
|
return &as32[x + y * stride];
|
|
|
|
}
|
2017-01-23 15:57:16 +00:00
|
|
|
};
|
2013-09-14 18:36:56 +00:00
|
|
|
|
2022-01-23 06:41:41 +00:00
|
|
|
enum class SoftDirty : uint64_t {
|
|
|
|
NONE = 0,
|
|
|
|
|
|
|
|
PIXEL_BASIC = 1ULL << 0,
|
|
|
|
PIXEL_STENCIL = 1ULL << 1,
|
|
|
|
PIXEL_ALPHA = 1ULL << 2,
|
|
|
|
PIXEL_DITHER = 1ULL << 3,
|
|
|
|
PIXEL_WRITEMASK = 1ULL << 4,
|
|
|
|
PIXEL_CACHED = 1ULL << 5,
|
2022-09-23 03:21:44 +00:00
|
|
|
PIXEL_ALL = 0b111111ULL << 0,
|
2022-01-23 06:41:41 +00:00
|
|
|
|
|
|
|
SAMPLER_BASIC = 1ULL << 6,
|
|
|
|
SAMPLER_TEXLIST = 1ULL << 7,
|
2022-01-23 15:09:25 +00:00
|
|
|
SAMPLER_CLUT = 1ULL << 8,
|
2022-09-23 03:21:44 +00:00
|
|
|
SAMPLER_ALL = 0b111ULL << 6,
|
2022-01-23 15:09:25 +00:00
|
|
|
|
|
|
|
RAST_BASIC = 1ULL << 9,
|
|
|
|
RAST_TEX = 1ULL << 10,
|
|
|
|
RAST_OFFSET = 1ULL << 11,
|
2022-09-23 03:21:44 +00:00
|
|
|
RAST_ALL = 0b111ULL << 9,
|
2022-01-23 15:09:25 +00:00
|
|
|
|
|
|
|
LIGHT_BASIC = 1ULL << 12,
|
|
|
|
LIGHT_MATERIAL = 1ULL << 13,
|
|
|
|
LIGHT_0 = 1ULL << 14,
|
|
|
|
LIGHT_1 = 1ULL << 15,
|
|
|
|
LIGHT_2 = 1ULL << 16,
|
|
|
|
LIGHT_3 = 1ULL << 17,
|
2022-09-23 03:21:44 +00:00
|
|
|
LIGHT_ALL = 0b111111ULL << 12,
|
2022-01-23 15:09:25 +00:00
|
|
|
|
|
|
|
TRANSFORM_BASIC = 1ULL << 18,
|
|
|
|
TRANSFORM_MATRIX = 1ULL << 19,
|
2022-05-08 21:24:21 +00:00
|
|
|
TRANSFORM_VIEWPORT = 1ULL << 20,
|
2022-01-23 15:09:25 +00:00
|
|
|
TRANSFORM_FOG = 1ULL << 21,
|
2022-09-23 03:21:44 +00:00
|
|
|
TRANSFORM_ALL = 0b1111ULL << 18,
|
2022-01-23 15:09:25 +00:00
|
|
|
|
|
|
|
BINNER_RANGE = 1ULL << 22,
|
|
|
|
BINNER_OVERLAP = 1ULL << 23,
|
2022-01-23 06:41:41 +00:00
|
|
|
};
|
|
|
|
static inline SoftDirty operator |(const SoftDirty &lhs, const SoftDirty &rhs) {
|
|
|
|
return SoftDirty((uint64_t)lhs | (uint64_t)rhs);
|
|
|
|
}
|
|
|
|
static inline SoftDirty &operator |=(SoftDirty &lhs, const SoftDirty &rhs) {
|
|
|
|
lhs = lhs | rhs;
|
|
|
|
return lhs;
|
|
|
|
}
|
|
|
|
static inline bool operator &(const SoftDirty &lhs, const SoftDirty &rhs) {
|
|
|
|
return ((uint64_t)lhs & (uint64_t)rhs) != 0;
|
|
|
|
}
|
|
|
|
static inline SoftDirty &operator &=(SoftDirty &lhs, const SoftDirty &rhs) {
|
|
|
|
lhs = SoftDirty((uint64_t)lhs & (uint64_t)rhs);
|
|
|
|
return lhs;
|
|
|
|
}
|
|
|
|
static inline SoftDirty operator ~(const SoftDirty &v) {
|
|
|
|
return SoftDirty(~(uint64_t)v);
|
|
|
|
}
|
|
|
|
|
2020-05-10 05:06:22 +00:00
|
|
|
class PresentationCommon;
|
2017-05-21 03:19:08 +00:00
|
|
|
class SoftwareDrawEngine;
|
2013-06-24 14:03:25 +00:00
|
|
|
|
2022-02-12 23:27:18 +00:00
|
|
|
enum class SoftGPUVRAMDirty : uint8_t {
|
|
|
|
CLEAR = 0,
|
|
|
|
DIRTY = 1,
|
|
|
|
REALLY_DIRTY = 2,
|
|
|
|
};
|
|
|
|
|
|
|
|
ENUM_CLASS_BITOPS(SoftGPUVRAMDirty);
|
|
|
|
|
2015-07-25 12:22:43 +00:00
|
|
|
class SoftGPU : public GPUCommon {
|
2013-06-24 14:03:25 +00:00
|
|
|
public:
|
2019-09-29 04:36:03 +00:00
|
|
|
SoftGPU(GraphicsContext *gfxCtx, Draw::DrawContext *draw);
|
2013-06-24 14:03:25 +00:00
|
|
|
~SoftGPU();
|
2018-02-26 10:35:37 +00:00
|
|
|
|
2022-09-20 08:02:15 +00:00
|
|
|
u32 CheckGPUFeatures() const override { return 0; }
|
2023-01-01 14:59:14 +00:00
|
|
|
bool IsStarted() override;
|
2015-07-25 12:22:43 +00:00
|
|
|
void ExecuteOp(u32 op, u32 diff) override;
|
2022-01-16 06:19:44 +00:00
|
|
|
void FinishDeferred() override;
|
2022-09-17 23:28:15 +00:00
|
|
|
int ListSync(int listid, int mode) override;
|
|
|
|
u32 DrawSync(int mode) override;
|
2023-02-25 15:20:34 +00:00
|
|
|
void UpdateCmdInfo() override {}
|
2015-07-25 12:22:43 +00:00
|
|
|
|
|
|
|
void SetDisplayFramebuffer(u32 framebuf, u32 stride, GEBufferFormat format) override;
|
2020-03-01 21:55:28 +00:00
|
|
|
void CopyDisplayToOutput(bool reallyDirty) override;
|
2016-03-31 08:17:02 +00:00
|
|
|
void GetStats(char *buffer, size_t bufsize) override;
|
2023-12-02 12:56:18 +00:00
|
|
|
std::vector<const VirtualFramebuffer *> GetFramebufferList() const override { return std::vector<const VirtualFramebuffer *>(); }
|
2015-07-25 12:22:43 +00:00
|
|
|
void InvalidateCache(u32 addr, int size, GPUInvalidationType type) override;
|
2022-10-09 20:49:41 +00:00
|
|
|
void PerformWriteFormattedFromMemory(u32 addr, int size, int width, GEBufferFormat format) override;
|
2022-10-04 03:17:25 +00:00
|
|
|
bool PerformMemoryCopy(u32 dest, u32 src, int size, GPUCopyFlag flags = GPUCopyFlag::NONE) override;
|
2015-07-25 12:22:43 +00:00
|
|
|
bool PerformMemorySet(u32 dest, u8 v, int size) override;
|
2022-10-09 20:49:41 +00:00
|
|
|
bool PerformReadbackToMemory(u32 dest, int size) override;
|
|
|
|
bool PerformWriteColorFromMemory(u32 dest, int size) override;
|
|
|
|
bool PerformWriteStencilFromMemory(u32 dest, int size, WriteStencil flags) override;
|
2015-07-25 12:22:43 +00:00
|
|
|
|
2016-01-18 01:56:19 +00:00
|
|
|
void DeviceLost() override;
|
2023-02-25 22:04:27 +00:00
|
|
|
void DeviceRestore(Draw::DrawContext *draw) override;
|
2015-07-25 12:22:43 +00:00
|
|
|
|
2022-11-21 14:14:20 +00:00
|
|
|
void NotifyRenderResized() override;
|
|
|
|
void NotifyDisplayResized() override;
|
|
|
|
|
2023-05-26 08:28:10 +00:00
|
|
|
void CheckDisplayResized() override;
|
2023-05-25 21:59:17 +00:00
|
|
|
void CheckConfigChanged() override;
|
|
|
|
|
2015-07-25 12:22:43 +00:00
|
|
|
void GetReportingInfo(std::string &primaryInfo, std::string &fullInfo) override {
|
2013-08-17 16:48:06 +00:00
|
|
|
primaryInfo = "Software";
|
|
|
|
fullInfo = "Software";
|
2013-06-24 14:03:25 +00:00
|
|
|
}
|
|
|
|
|
2015-07-25 12:22:43 +00:00
|
|
|
bool FramebufferDirty() override;
|
2022-02-12 23:27:18 +00:00
|
|
|
bool FramebufferReallyDirty() override;
|
2013-10-27 23:50:03 +00:00
|
|
|
|
2016-09-25 23:31:38 +00:00
|
|
|
bool GetCurrentFramebuffer(GPUDebugBuffer &buffer, GPUDebugFramebufferType type, int maxRes = -1) override;
|
2017-06-02 03:40:45 +00:00
|
|
|
bool GetOutputFramebuffer(GPUDebugBuffer &buffer) override;
|
2015-07-25 12:22:43 +00:00
|
|
|
bool GetCurrentDepthbuffer(GPUDebugBuffer &buffer) override;
|
|
|
|
bool GetCurrentStencilbuffer(GPUDebugBuffer &buffer) override;
|
2022-10-11 05:35:42 +00:00
|
|
|
bool GetCurrentTexture(GPUDebugBuffer &buffer, int level, bool *isFramebuffer) override;
|
2016-01-10 17:25:19 +00:00
|
|
|
bool GetCurrentClut(GPUDebugBuffer &buffer) override;
|
2015-09-17 18:29:37 +00:00
|
|
|
bool GetCurrentSimpleVertices(int count, std::vector<GPUDebugVertex> &vertices, std::vector<u16> &indices) override;
|
2013-09-23 02:03:31 +00:00
|
|
|
|
2017-05-11 03:36:09 +00:00
|
|
|
bool DescribeCodePtr(const u8 *ptr, std::string &name) override;
|
|
|
|
|
2022-01-23 06:41:41 +00:00
|
|
|
void Execute_BlockTransferStart(u32 op, u32 diff);
|
|
|
|
void Execute_Prim(u32 op, u32 diff);
|
|
|
|
void Execute_Bezier(u32 op, u32 diff);
|
|
|
|
void Execute_Spline(u32 op, u32 diff);
|
|
|
|
void Execute_LoadClut(u32 op, u32 diff);
|
|
|
|
void Execute_FramebufPtr(u32 op, u32 diff);
|
|
|
|
void Execute_FramebufFormat(u32 op, u32 diff);
|
|
|
|
void Execute_ZbufPtr(u32 op, u32 diff);
|
|
|
|
void Execute_VertexType(u32 op, u32 diff);
|
|
|
|
|
|
|
|
// Overridden to change flushing behavior.
|
|
|
|
void Execute_Call(u32 op, u32 diff);
|
|
|
|
|
2023-07-30 16:35:18 +00:00
|
|
|
// Overridden for a dirty flag change.
|
|
|
|
void Execute_BoundingBox(u32 op, u32 diff);
|
|
|
|
|
2022-09-28 05:29:55 +00:00
|
|
|
void Execute_WorldMtxNum(u32 op, u32 diff);
|
|
|
|
void Execute_ViewMtxNum(u32 op, u32 diff);
|
|
|
|
void Execute_ProjMtxNum(u32 op, u32 diff);
|
|
|
|
void Execute_TgenMtxNum(u32 op, u32 diff);
|
|
|
|
void Execute_BoneMtxNum(u32 op, u32 diff);
|
|
|
|
|
2022-01-23 06:41:41 +00:00
|
|
|
void Execute_WorldMtxData(u32 op, u32 diff);
|
|
|
|
void Execute_ViewMtxData(u32 op, u32 diff);
|
|
|
|
void Execute_ProjMtxData(u32 op, u32 diff);
|
|
|
|
void Execute_TgenMtxData(u32 op, u32 diff);
|
|
|
|
void Execute_BoneMtxData(u32 op, u32 diff);
|
|
|
|
|
2022-09-28 05:29:55 +00:00
|
|
|
bool GetMatrix24(GEMatrixType type, u32_le *result, u32 cmdbits) override;
|
2022-09-30 04:36:53 +00:00
|
|
|
void ResetMatrices() override;
|
2022-09-28 05:29:55 +00:00
|
|
|
|
2022-09-18 03:15:40 +00:00
|
|
|
void Execute_ImmVertexAlphaPrim(u32 op, u32 diff);
|
|
|
|
|
2022-01-23 06:41:41 +00:00
|
|
|
typedef void (SoftGPU::*CmdFunc)(u32 op, u32 diff);
|
|
|
|
|
2013-06-24 14:03:25 +00:00
|
|
|
protected:
|
2015-07-25 12:22:43 +00:00
|
|
|
void FastRunLoop(DisplayList &list) override;
|
2013-12-17 08:02:54 +00:00
|
|
|
void CopyToCurrentFboFromDisplayRam(int srcwidth, int srcheight);
|
2022-09-20 20:43:19 +00:00
|
|
|
void ConvertTextureDescFrom16(Draw::TextureDesc &desc, int srcwidth, int srcheight, const uint16_t *overrideData = nullptr);
|
2022-11-20 11:57:32 +00:00
|
|
|
|
|
|
|
void BuildReportingInfo() override {}
|
2013-08-17 16:52:35 +00:00
|
|
|
|
|
|
|
private:
|
2022-02-12 23:27:18 +00:00
|
|
|
void MarkDirty(uint32_t addr, uint32_t stride, uint32_t height, GEBufferFormat fmt, SoftGPUVRAMDirty value);
|
|
|
|
void MarkDirty(uint32_t addr, uint32_t bytes, SoftGPUVRAMDirty value);
|
|
|
|
bool ClearDirty(uint32_t addr, uint32_t stride, uint32_t height, GEBufferFormat fmt, SoftGPUVRAMDirty value);
|
|
|
|
bool ClearDirty(uint32_t addr, uint32_t bytes, SoftGPUVRAMDirty value);
|
|
|
|
|
|
|
|
uint8_t vramDirty_[2048];
|
|
|
|
uint32_t lastDirtyAddr_ = 0;
|
|
|
|
uint32_t lastDirtySize_ = 0;
|
|
|
|
SoftGPUVRAMDirty lastDirtyValue_ = SoftGPUVRAMDirty::CLEAR;
|
|
|
|
|
2013-12-15 16:55:19 +00:00
|
|
|
u32 displayFramebuf_;
|
|
|
|
u32 displayStride_;
|
|
|
|
GEBufferFormat displayFormat_;
|
2022-01-23 06:41:41 +00:00
|
|
|
SoftDirty dirtyFlags_ = SoftDirty(-1);
|
2016-02-13 18:53:28 +00:00
|
|
|
|
2020-05-10 05:06:22 +00:00
|
|
|
PresentationCommon *presentation_ = nullptr;
|
2017-05-21 03:19:08 +00:00
|
|
|
SoftwareDrawEngine *drawEngine_ = nullptr;
|
|
|
|
|
2020-05-10 05:06:22 +00:00
|
|
|
Draw::Texture *fbTex = nullptr;
|
2019-12-24 19:05:15 +00:00
|
|
|
std::vector<u32> fbTexBuffer_;
|
2013-06-24 14:03:25 +00:00
|
|
|
};
|
2017-05-24 08:20:10 +00:00
|
|
|
|
|
|
|
// TODO: These shouldn't be global.
|
2022-08-21 00:36:15 +00:00
|
|
|
extern uint8_t clut[1024];
|
2017-05-24 08:20:10 +00:00
|
|
|
extern FormatBuffer fb;
|
|
|
|
extern FormatBuffer depthbuf;
|
2020-05-24 14:51:37 +00:00
|
|
|
|
|
|
|
// Type for the DarkStalkers stretch replacement.
|
|
|
|
enum class DSStretch {
|
|
|
|
Off = 0,
|
|
|
|
Normal,
|
|
|
|
Wide,
|
|
|
|
};
|