2014-09-09 15:12:42 +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
|
|
|
|
|
2014-09-13 22:40:55 +00:00
|
|
|
#include <set>
|
2014-09-09 15:12:42 +00:00
|
|
|
#include <vector>
|
2018-06-03 15:08:45 +00:00
|
|
|
#include <unordered_map>
|
2015-07-26 20:38:40 +00:00
|
|
|
|
2014-09-09 15:12:42 +00:00
|
|
|
#include "Common/CommonTypes.h"
|
|
|
|
#include "Core/MemMap.h"
|
2015-07-26 20:38:40 +00:00
|
|
|
#include "GPU/GPU.h"
|
2014-09-09 15:12:42 +00:00
|
|
|
#include "GPU/ge_constants.h"
|
2017-10-18 10:49:15 +00:00
|
|
|
#include "GPU/GPUInterface.h"
|
2017-02-06 23:24:38 +00:00
|
|
|
#include "thin3d/thin3d.h"
|
2014-09-09 15:12:42 +00:00
|
|
|
|
|
|
|
enum {
|
|
|
|
FB_USAGE_DISPLAYED_FRAMEBUFFER = 1,
|
|
|
|
FB_USAGE_RENDERTARGET = 2,
|
|
|
|
FB_USAGE_TEXTURE = 4,
|
2015-04-28 07:08:00 +00:00
|
|
|
FB_USAGE_CLUT = 8,
|
2017-04-09 22:10:07 +00:00
|
|
|
FB_USAGE_DOWNLOAD = 16,
|
|
|
|
FB_USAGE_DOWNLOAD_CLEAR = 32,
|
2014-09-09 15:12:42 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
enum {
|
|
|
|
FB_NON_BUFFERED_MODE = 0,
|
|
|
|
FB_BUFFERED_MODE = 1,
|
|
|
|
};
|
|
|
|
|
2017-02-04 17:46:12 +00:00
|
|
|
namespace Draw {
|
|
|
|
class Framebuffer;
|
2015-11-14 12:24:51 +00:00
|
|
|
}
|
2014-09-09 15:12:42 +00:00
|
|
|
|
2017-02-06 23:19:31 +00:00
|
|
|
struct CardboardSettings {
|
|
|
|
bool enabled;
|
|
|
|
float leftEyeXPosition;
|
|
|
|
float rightEyeXPosition;
|
|
|
|
float screenYPosition;
|
|
|
|
float screenWidth;
|
|
|
|
float screenHeight;
|
|
|
|
};
|
|
|
|
|
2016-03-17 10:56:43 +00:00
|
|
|
class VulkanFBO;
|
2016-01-03 22:09:37 +00:00
|
|
|
|
2017-02-15 11:09:51 +00:00
|
|
|
struct PostShaderUniforms {
|
|
|
|
float texelDelta[2]; float pixelDelta[2];
|
|
|
|
float time[4];
|
2017-11-01 22:35:06 +00:00
|
|
|
float video;
|
2017-02-15 11:09:51 +00:00
|
|
|
};
|
|
|
|
|
2014-09-09 15:12:42 +00:00
|
|
|
struct VirtualFramebuffer {
|
|
|
|
int last_frame_used;
|
|
|
|
int last_frame_attached;
|
|
|
|
int last_frame_render;
|
2015-03-14 21:58:32 +00:00
|
|
|
int last_frame_displayed;
|
2015-04-28 07:08:00 +00:00
|
|
|
int last_frame_clut;
|
2017-04-24 16:30:04 +00:00
|
|
|
int last_frame_failed;
|
2017-12-24 19:52:15 +00:00
|
|
|
int last_frame_depth_updated;
|
|
|
|
int last_frame_depth_render;
|
2016-01-05 06:21:33 +00:00
|
|
|
u32 clutUpdatedBytes;
|
2014-09-09 15:12:42 +00:00
|
|
|
bool memoryUpdated;
|
2016-06-05 02:36:30 +00:00
|
|
|
bool firstFrameSaved;
|
2014-09-09 15:12:42 +00:00
|
|
|
|
|
|
|
u32 fb_address;
|
|
|
|
u32 z_address;
|
|
|
|
int fb_stride;
|
|
|
|
int z_stride;
|
|
|
|
|
|
|
|
// There's also a top left of the drawing region, but meh...
|
|
|
|
|
|
|
|
// width/height: The detected size of the current framebuffer.
|
|
|
|
u16 width;
|
|
|
|
u16 height;
|
2018-06-18 00:40:37 +00:00
|
|
|
// renderWidth/renderHeight: The scaled size we render at. May be scaled to render at higher resolutions.
|
|
|
|
// The physical buffer may be larger than renderWidth/renderHeight.
|
2014-09-09 15:12:42 +00:00
|
|
|
u16 renderWidth;
|
|
|
|
u16 renderHeight;
|
2018-06-18 00:40:37 +00:00
|
|
|
// bufferWidth/bufferHeight: The pre-scaling size of the buffer itself. May only be bigger than width/height.
|
|
|
|
// Actual physical buffer is this size times the render resolution multiplier.
|
|
|
|
// The buffer may be used to render a width or height from 0 to these values without being recreated.
|
2014-09-09 15:12:42 +00:00
|
|
|
u16 bufferWidth;
|
|
|
|
u16 bufferHeight;
|
|
|
|
|
|
|
|
u16 usageFlags;
|
|
|
|
|
|
|
|
u16 newWidth;
|
|
|
|
u16 newHeight;
|
|
|
|
int lastFrameNewSize;
|
|
|
|
|
|
|
|
GEBufferFormat format; // virtual, right now they are all RGBA8888
|
2016-03-17 10:56:43 +00:00
|
|
|
|
2014-09-09 15:12:42 +00:00
|
|
|
// TODO: Handle fbo and colorDepth better.
|
|
|
|
u8 colorDepth;
|
2017-02-04 17:46:12 +00:00
|
|
|
Draw::Framebuffer *fbo;
|
2014-09-09 15:12:42 +00:00
|
|
|
|
|
|
|
u16 drawnWidth;
|
|
|
|
u16 drawnHeight;
|
|
|
|
GEBufferFormat drawnFormat;
|
2016-05-20 03:55:34 +00:00
|
|
|
u16 safeWidth;
|
|
|
|
u16 safeHeight;
|
2014-09-09 15:12:42 +00:00
|
|
|
|
|
|
|
bool dirtyAfterDisplay;
|
|
|
|
bool reallyDirtyAfterDisplay; // takes frame skipping into account
|
|
|
|
};
|
|
|
|
|
2015-08-05 09:51:24 +00:00
|
|
|
struct FramebufferHeuristicParams {
|
|
|
|
u32 fb_address;
|
|
|
|
int fb_stride;
|
|
|
|
u32 z_address;
|
|
|
|
int z_stride;
|
|
|
|
GEBufferFormat fmt;
|
|
|
|
bool isClearingDepth;
|
|
|
|
bool isWritingDepth;
|
|
|
|
bool isDrawing;
|
|
|
|
bool isModeThrough;
|
|
|
|
int viewportWidth;
|
|
|
|
int viewportHeight;
|
|
|
|
int regionWidth;
|
|
|
|
int regionHeight;
|
|
|
|
int scissorWidth;
|
|
|
|
int scissorHeight;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct GPUgstate;
|
|
|
|
extern GPUgstate gstate;
|
|
|
|
|
|
|
|
void GetFramebufferHeuristicInputs(FramebufferHeuristicParams *params, const GPUgstate &gstate);
|
|
|
|
|
2015-09-13 18:14:51 +00:00
|
|
|
enum BindFramebufferColorFlags {
|
|
|
|
BINDFBCOLOR_SKIP_COPY = 0,
|
|
|
|
BINDFBCOLOR_MAY_COPY = 1,
|
|
|
|
BINDFBCOLOR_MAY_COPY_WITH_UV = 3,
|
2015-09-13 02:43:02 +00:00
|
|
|
BINDFBCOLOR_APPLY_TEX_OFFSET = 4,
|
2015-09-13 18:14:51 +00:00
|
|
|
};
|
|
|
|
|
2017-06-01 06:24:56 +00:00
|
|
|
enum DrawTextureFlags {
|
|
|
|
DRAWTEX_NEAREST = 0,
|
|
|
|
DRAWTEX_LINEAR = 1,
|
|
|
|
DRAWTEX_KEEP_TEX = 2,
|
2018-02-10 16:18:18 +00:00
|
|
|
DRAWTEX_KEEP_STENCIL_ALPHA = 4,
|
2017-06-01 06:24:56 +00:00
|
|
|
};
|
|
|
|
|
2018-05-06 15:57:44 +00:00
|
|
|
enum class TempFBO {
|
|
|
|
DEPAL,
|
|
|
|
BLIT,
|
|
|
|
// For copies of framebuffers (e.g. shader blending.)
|
|
|
|
COPY,
|
|
|
|
};
|
|
|
|
|
2017-10-18 09:20:58 +00:00
|
|
|
inline Draw::DataFormat GEFormatToThin3D(int geFormat) {
|
|
|
|
switch (geFormat) {
|
|
|
|
case GE_FORMAT_4444:
|
|
|
|
return Draw::DataFormat::A4R4G4B4_UNORM_PACK16;
|
|
|
|
case GE_FORMAT_5551:
|
|
|
|
return Draw::DataFormat::A1R5G5B5_UNORM_PACK16;
|
|
|
|
case GE_FORMAT_565:
|
|
|
|
return Draw::DataFormat::R5G6B5_UNORM_PACK16;
|
|
|
|
case GE_FORMAT_8888:
|
|
|
|
return Draw::DataFormat::R8G8B8A8_UNORM;
|
|
|
|
default:
|
|
|
|
return Draw::DataFormat::UNDEFINED;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-02-05 18:51:50 +00:00
|
|
|
namespace Draw {
|
|
|
|
class DrawContext;
|
|
|
|
}
|
|
|
|
|
2017-02-14 11:42:35 +00:00
|
|
|
struct GPUDebugBuffer;
|
2017-02-06 11:02:30 +00:00
|
|
|
class TextureCacheCommon;
|
2017-02-15 17:32:44 +00:00
|
|
|
class ShaderManagerCommon;
|
2017-10-18 10:26:02 +00:00
|
|
|
class DrawEngineCommon;
|
2017-02-06 11:02:30 +00:00
|
|
|
|
2014-09-09 15:12:42 +00:00
|
|
|
class FramebufferManagerCommon {
|
|
|
|
public:
|
2017-02-05 18:51:50 +00:00
|
|
|
FramebufferManagerCommon(Draw::DrawContext *draw);
|
2014-09-09 15:12:42 +00:00
|
|
|
virtual ~FramebufferManagerCommon();
|
|
|
|
|
2014-09-13 23:47:23 +00:00
|
|
|
virtual void Init();
|
2017-10-25 18:28:12 +00:00
|
|
|
virtual void BeginFrame();
|
2014-09-13 21:44:18 +00:00
|
|
|
void SetDisplayFramebuffer(u32 framebuf, u32 stride, GEBufferFormat format);
|
2017-02-06 11:02:30 +00:00
|
|
|
void DestroyFramebuf(VirtualFramebuffer *v);
|
2014-09-10 05:56:54 +00:00
|
|
|
|
2017-05-31 11:38:22 +00:00
|
|
|
VirtualFramebuffer *DoSetRenderFrameBuffer(const FramebufferHeuristicParams ¶ms, u32 skipDrawReason);
|
2015-08-05 00:43:40 +00:00
|
|
|
VirtualFramebuffer *SetRenderFrameBuffer(bool framebufChanged, int skipDrawReason) {
|
2014-09-09 15:12:42 +00:00
|
|
|
// Inlining this part since it's so frequent.
|
2015-07-26 20:38:40 +00:00
|
|
|
if (!framebufChanged && currentRenderVfb_) {
|
2014-09-09 15:12:42 +00:00
|
|
|
currentRenderVfb_->last_frame_render = gpuStats.numFlips;
|
|
|
|
currentRenderVfb_->dirtyAfterDisplay = true;
|
2015-07-26 20:38:40 +00:00
|
|
|
if (!skipDrawReason)
|
2014-09-09 15:12:42 +00:00
|
|
|
currentRenderVfb_->reallyDirtyAfterDisplay = true;
|
2015-08-05 00:43:40 +00:00
|
|
|
return currentRenderVfb_;
|
2015-08-05 09:51:24 +00:00
|
|
|
} else {
|
|
|
|
// This is so that we will be able to drive DoSetRenderFramebuffer with inputs
|
|
|
|
// that come from elsewhere than gstate.
|
|
|
|
FramebufferHeuristicParams inputs;
|
|
|
|
GetFramebufferHeuristicInputs(&inputs, gstate);
|
2016-03-21 23:12:41 +00:00
|
|
|
VirtualFramebuffer *vfb = DoSetRenderFrameBuffer(inputs, skipDrawReason);
|
2017-10-25 18:28:12 +00:00
|
|
|
_dbg_assert_msg_(G3D, vfb, "DoSetRenderFramebuffer must return a valid framebuffer.");
|
|
|
|
_dbg_assert_msg_(G3D, currentRenderVfb_, "DoSetRenderFramebuffer must set a valid framebuffer.");
|
2016-03-21 23:12:41 +00:00
|
|
|
return vfb;
|
2014-09-09 15:12:42 +00:00
|
|
|
}
|
|
|
|
}
|
2017-12-12 13:07:52 +00:00
|
|
|
void RebindFramebuffer();
|
2017-10-18 10:49:15 +00:00
|
|
|
std::vector<FramebufferInfo> GetFramebufferList();
|
2014-09-09 15:12:42 +00:00
|
|
|
|
2017-02-15 22:24:25 +00:00
|
|
|
void CopyDisplayToOutput();
|
|
|
|
|
2015-08-05 10:13:14 +00:00
|
|
|
bool NotifyFramebufferCopy(u32 src, u32 dest, int size, bool isMemset, u32 skipDrawReason);
|
2016-01-17 20:52:40 +00:00
|
|
|
void NotifyVideoUpload(u32 addr, int size, int width, GEBufferFormat fmt);
|
2014-09-13 21:44:18 +00:00
|
|
|
void UpdateFromMemory(u32 addr, int size, bool safe);
|
2017-04-09 22:10:07 +00:00
|
|
|
void ApplyClearToMemory(int x1, int y1, int x2, int y2, u32 clearColor);
|
2014-09-13 21:44:18 +00:00
|
|
|
virtual bool NotifyStencilUpload(u32 addr, int size, bool skipZero = false) = 0;
|
2014-09-13 22:40:55 +00:00
|
|
|
// Returns true if it's sure this is a direct FBO->FBO transfer and it has already handle it.
|
|
|
|
// In that case we hardly need to actually copy the bytes in VRAM, they will be wrong anyway (unless
|
|
|
|
// read framebuffers is on, in which case this should always return false).
|
2015-08-05 10:13:14 +00:00
|
|
|
bool NotifyBlockTransferBefore(u32 dstBasePtr, int dstStride, int dstX, int dstY, u32 srcBasePtr, int srcStride, int srcX, int srcY, int w, int h, int bpp, u32 skipDrawReason);
|
|
|
|
void NotifyBlockTransferAfter(u32 dstBasePtr, int dstStride, int dstX, int dstY, u32 srcBasePtr, int srcStride, int srcX, int srcY, int w, int h, int bpp, u32 skipDrawReason);
|
2014-09-09 15:12:42 +00:00
|
|
|
|
2017-12-21 18:05:33 +00:00
|
|
|
void ReadFramebufferToMemory(VirtualFramebuffer *vfb, bool sync, int x, int y, int w, int h);
|
2017-10-18 10:34:01 +00:00
|
|
|
|
2017-12-21 14:38:55 +00:00
|
|
|
void DownloadFramebufferForClut(u32 fb_address, u32 loadBytes);
|
2017-02-15 22:11:46 +00:00
|
|
|
void DrawFramebufferToOutput(const u8 *srcPixels, GEBufferFormat srcPixelFormat, int srcStride, bool applyPostShader);
|
2014-09-13 21:44:18 +00:00
|
|
|
|
2017-02-15 17:32:44 +00:00
|
|
|
void DrawPixels(VirtualFramebuffer *vfb, int dstX, int dstY, const u8 *srcPixels, GEBufferFormat srcPixelFormat, int srcStride, int width, int height);
|
|
|
|
|
2014-09-13 21:44:18 +00:00
|
|
|
size_t NumVFBs() const { return vfbs_.size(); }
|
2014-09-09 15:12:42 +00:00
|
|
|
|
|
|
|
u32 PrevDisplayFramebufAddr() {
|
2018-11-11 09:54:28 +00:00
|
|
|
return prevDisplayFramebuf_ ? prevDisplayFramebuf_->fb_address : 0;
|
2014-09-09 15:12:42 +00:00
|
|
|
}
|
|
|
|
u32 DisplayFramebufAddr() {
|
2018-11-11 09:54:28 +00:00
|
|
|
return displayFramebuf_ ? displayFramebuf_->fb_address : 0;
|
2014-09-09 15:12:42 +00:00
|
|
|
}
|
|
|
|
|
2016-09-25 23:31:38 +00:00
|
|
|
u32 DisplayFramebufStride() {
|
|
|
|
return displayFramebuf_ ? displayStride_ : 0;
|
|
|
|
}
|
|
|
|
GEBufferFormat DisplayFramebufFormat() {
|
|
|
|
return displayFramebuf_ ? displayFormat_ : GE_FORMAT_INVALID;
|
|
|
|
}
|
|
|
|
|
2014-09-09 15:12:42 +00:00
|
|
|
bool MayIntersectFramebuffer(u32 start) {
|
|
|
|
// Clear the cache/kernel bits.
|
|
|
|
start = start & 0x3FFFFFFF;
|
|
|
|
// Most games only have two framebuffers at the start.
|
|
|
|
if (start >= framebufRangeEnd_ || start < PSP_GetVidMemBase()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-02-17 13:30:42 +00:00
|
|
|
VirtualFramebuffer *GetCurrentRenderVFB() const {
|
|
|
|
return currentRenderVfb_;
|
|
|
|
}
|
2014-09-10 05:09:41 +00:00
|
|
|
// TODO: Break out into some form of FBO manager
|
|
|
|
VirtualFramebuffer *GetVFBAt(u32 addr);
|
|
|
|
VirtualFramebuffer *GetDisplayVFB() {
|
|
|
|
return GetVFBAt(displayFramebufPtr_);
|
|
|
|
}
|
|
|
|
|
2014-09-09 15:12:42 +00:00
|
|
|
int GetRenderWidth() const { return currentRenderVfb_ ? currentRenderVfb_->renderWidth : 480; }
|
|
|
|
int GetRenderHeight() const { return currentRenderVfb_ ? currentRenderVfb_->renderHeight : 272; }
|
|
|
|
int GetTargetWidth() const { return currentRenderVfb_ ? currentRenderVfb_->width : 480; }
|
|
|
|
int GetTargetHeight() const { return currentRenderVfb_ ? currentRenderVfb_->height : 272; }
|
|
|
|
int GetTargetBufferWidth() const { return currentRenderVfb_ ? currentRenderVfb_->bufferWidth : 480; }
|
|
|
|
int GetTargetBufferHeight() const { return currentRenderVfb_ ? currentRenderVfb_->bufferHeight : 272; }
|
|
|
|
int GetTargetStride() const { return currentRenderVfb_ ? currentRenderVfb_->fb_stride : 512; }
|
|
|
|
GEBufferFormat GetTargetFormat() const { return currentRenderVfb_ ? currentRenderVfb_->format : displayFormat_; }
|
|
|
|
|
2014-09-13 21:44:18 +00:00
|
|
|
void SetDepthUpdated() {
|
|
|
|
if (currentRenderVfb_) {
|
2017-12-24 19:52:15 +00:00
|
|
|
currentRenderVfb_->last_frame_depth_render = gpuStats.numFlips;
|
|
|
|
currentRenderVfb_->last_frame_depth_updated = gpuStats.numFlips;
|
2014-09-13 21:44:18 +00:00
|
|
|
}
|
|
|
|
}
|
2015-07-26 20:38:40 +00:00
|
|
|
void SetColorUpdated(int skipDrawReason) {
|
2014-09-13 21:44:18 +00:00
|
|
|
if (currentRenderVfb_) {
|
2015-07-26 20:38:40 +00:00
|
|
|
SetColorUpdated(currentRenderVfb_, skipDrawReason);
|
2014-09-13 21:44:18 +00:00
|
|
|
}
|
|
|
|
}
|
2015-01-23 09:40:49 +00:00
|
|
|
void SetRenderSize(VirtualFramebuffer *vfb);
|
2016-05-20 03:55:34 +00:00
|
|
|
void SetSafeSize(u16 w, u16 h);
|
2014-09-13 21:44:18 +00:00
|
|
|
|
2017-04-24 18:58:16 +00:00
|
|
|
virtual void Resized();
|
2016-12-21 17:13:58 +00:00
|
|
|
|
2018-05-06 15:57:44 +00:00
|
|
|
Draw::Framebuffer *GetTempFBO(TempFBO reason, u16 w, u16 h, Draw::FBColorDepth colorDepth = Draw::FBO_8888);
|
2017-02-06 23:24:38 +00:00
|
|
|
|
2017-02-14 11:42:35 +00:00
|
|
|
// Debug features
|
2017-10-11 11:37:00 +00:00
|
|
|
virtual bool GetFramebuffer(u32 fb_address, int fb_stride, GEBufferFormat format, GPUDebugBuffer &buffer, int maxRes);
|
2017-10-11 13:21:53 +00:00
|
|
|
virtual bool GetDepthbuffer(u32 fb_address, int fb_stride, u32 z_address, int z_stride, GPUDebugBuffer &buffer);
|
|
|
|
virtual bool GetStencilbuffer(u32 fb_address, int fb_stride, GPUDebugBuffer &buffer);
|
2017-10-16 14:27:16 +00:00
|
|
|
virtual bool GetOutputFramebuffer(GPUDebugBuffer &buffer);
|
2017-02-14 11:42:35 +00:00
|
|
|
|
2014-09-09 15:12:42 +00:00
|
|
|
protected:
|
2017-10-18 09:20:58 +00:00
|
|
|
virtual void PackFramebufferSync_(VirtualFramebuffer *vfb, int x, int y, int w, int h);
|
2017-03-05 12:59:16 +00:00
|
|
|
virtual void SetViewport2D(int x, int y, int w, int h);
|
2017-02-15 11:09:51 +00:00
|
|
|
void CalculatePostShaderUniforms(int bufferWidth, int bufferHeight, int renderWidth, int renderHeight, PostShaderUniforms *uniforms);
|
2017-03-23 03:56:26 +00:00
|
|
|
virtual void MakePixelTexture(const u8 *srcPixels, GEBufferFormat srcPixelFormat, int srcStride, int width, int height, float &u1, float &v1) = 0;
|
2017-06-01 06:24:56 +00:00
|
|
|
virtual void DrawActiveTexture(float x, float y, float w, float h, float destW, float destH, float u0, float v0, float u1, float v1, int uvRotation, int flags) = 0;
|
2017-02-15 15:01:59 +00:00
|
|
|
virtual void Bind2DShader() = 0;
|
2017-02-15 22:24:25 +00:00
|
|
|
virtual void BindPostShader(const PostShaderUniforms &uniforms) = 0;
|
2017-02-15 11:09:51 +00:00
|
|
|
|
2017-02-06 23:19:31 +00:00
|
|
|
// Cardboard Settings Calculator
|
|
|
|
void GetCardboardSettings(CardboardSettings *cardboardSettings);
|
|
|
|
|
2017-04-14 06:35:07 +00:00
|
|
|
bool UpdateSize();
|
2017-02-06 11:13:42 +00:00
|
|
|
void SetNumExtraFBOs(int num);
|
2015-09-27 17:59:47 +00:00
|
|
|
|
2017-10-18 10:26:02 +00:00
|
|
|
void FlushBeforeCopy();
|
2017-02-06 23:42:39 +00:00
|
|
|
virtual void DecimateFBOs(); // keeping it virtual to let D3D do a little extra
|
2014-09-13 21:23:18 +00:00
|
|
|
|
2014-09-13 22:12:06 +00:00
|
|
|
// Used by ReadFramebufferToMemory and later framebuffer block copies
|
2015-11-01 12:32:03 +00:00
|
|
|
virtual void BlitFramebuffer(VirtualFramebuffer *dst, int dstX, int dstY, VirtualFramebuffer *src, int srcX, int srcY, int w, int h, int bpp) = 0;
|
2017-04-07 01:49:48 +00:00
|
|
|
void CopyFramebufferForColorTexture(VirtualFramebuffer *dst, VirtualFramebuffer *src, int flags);
|
2014-09-13 22:12:06 +00:00
|
|
|
|
2015-08-05 00:43:40 +00:00
|
|
|
void EstimateDrawingSize(u32 fb_address, GEBufferFormat fb_format, int viewport_width, int viewport_height, int region_width, int region_height, int scissor_width, int scissor_height, int fb_stride, int &drawing_width, int &drawing_height);
|
2014-09-10 05:56:54 +00:00
|
|
|
u32 FramebufferByteSize(const VirtualFramebuffer *vfb) const;
|
2014-09-09 15:12:42 +00:00
|
|
|
|
2017-02-06 11:10:08 +00:00
|
|
|
void NotifyRenderFramebufferCreated(VirtualFramebuffer *vfb);
|
|
|
|
void NotifyRenderFramebufferUpdated(VirtualFramebuffer *vfb, bool vfbFormatChanged);
|
2017-02-06 23:29:02 +00:00
|
|
|
void NotifyRenderFramebufferSwitched(VirtualFramebuffer *prevVfb, VirtualFramebuffer *vfb, bool isClearingDepth);
|
2017-02-06 11:10:08 +00:00
|
|
|
|
|
|
|
virtual void ReformatFramebufferFrom(VirtualFramebuffer *vfb, GEBufferFormat old) = 0;
|
2017-02-06 23:29:02 +00:00
|
|
|
virtual void BlitFramebufferDepth(VirtualFramebuffer *src, VirtualFramebuffer *dst) = 0;
|
2014-09-10 05:56:54 +00:00
|
|
|
|
2017-11-04 10:41:44 +00:00
|
|
|
void ResizeFramebufFBO(VirtualFramebuffer *vfb, int w, int h, bool force = false, bool skipCopy = false);
|
2015-09-23 10:25:38 +00:00
|
|
|
void ShowScreenResolution();
|
|
|
|
|
2014-09-10 05:56:54 +00:00
|
|
|
bool ShouldDownloadFramebuffer(const VirtualFramebuffer *vfb) const;
|
2016-06-05 02:36:30 +00:00
|
|
|
void DownloadFramebufferOnSwitch(VirtualFramebuffer *vfb);
|
2018-11-04 23:28:01 +00:00
|
|
|
void FindTransferFramebuffers(VirtualFramebuffer *&dstBuffer, VirtualFramebuffer *&srcBuffer, u32 dstBasePtr, int dstStride, int &dstX, int &dstY, u32 srcBasePtr, int srcStride, int &srcX, int &srcY, int &srcWidth, int &srcHeight, int &dstWidth, int &dstHeight, int bpp);
|
2016-01-05 04:40:07 +00:00
|
|
|
VirtualFramebuffer *FindDownloadTempBuffer(VirtualFramebuffer *vfb);
|
2016-01-05 04:51:43 +00:00
|
|
|
virtual bool CreateDownloadTempBuffer(VirtualFramebuffer *nvfb) = 0;
|
|
|
|
virtual void UpdateDownloadTempBuffer(VirtualFramebuffer *nvfb) = 0;
|
2018-11-11 21:50:15 +00:00
|
|
|
|
|
|
|
VirtualFramebuffer *CreateRAMFramebuffer(uint32_t fbAddress, int width, int height, int stride, GEBufferFormat format);
|
2016-01-05 04:40:07 +00:00
|
|
|
void OptimizeDownloadRange(VirtualFramebuffer *vfb, int &x, int &y, int &w, int &h);
|
2014-09-10 05:56:54 +00:00
|
|
|
|
2015-03-14 21:58:32 +00:00
|
|
|
void UpdateFramebufUsage(VirtualFramebuffer *vfb);
|
|
|
|
|
2015-07-26 20:38:40 +00:00
|
|
|
void SetColorUpdated(VirtualFramebuffer *dstBuffer, int skipDrawReason) {
|
2014-09-09 15:12:42 +00:00
|
|
|
dstBuffer->memoryUpdated = false;
|
2016-01-05 06:21:33 +00:00
|
|
|
dstBuffer->clutUpdatedBytes = 0;
|
2014-09-09 15:12:42 +00:00
|
|
|
dstBuffer->dirtyAfterDisplay = true;
|
|
|
|
dstBuffer->drawnWidth = dstBuffer->width;
|
|
|
|
dstBuffer->drawnHeight = dstBuffer->height;
|
|
|
|
dstBuffer->drawnFormat = dstBuffer->format;
|
2015-07-26 20:38:40 +00:00
|
|
|
if ((skipDrawReason & SKIPDRAW_SKIPFRAME) == 0)
|
2014-09-09 15:12:42 +00:00
|
|
|
dstBuffer->reallyDirtyAfterDisplay = true;
|
|
|
|
}
|
|
|
|
|
2017-06-02 15:03:29 +00:00
|
|
|
Draw::DrawContext *draw_ = nullptr;
|
|
|
|
TextureCacheCommon *textureCache_ = nullptr;
|
|
|
|
ShaderManagerCommon *shaderManager_ = nullptr;
|
2017-10-18 10:26:02 +00:00
|
|
|
DrawEngineCommon *drawEngine_ = nullptr;
|
2017-06-02 15:03:29 +00:00
|
|
|
bool needBackBufferYSwap_ = false;
|
2017-02-05 18:51:50 +00:00
|
|
|
|
2017-06-02 15:03:29 +00:00
|
|
|
u32 displayFramebufPtr_ = 0;
|
|
|
|
u32 displayStride_ = 0;
|
2014-09-09 15:12:42 +00:00
|
|
|
GEBufferFormat displayFormat_;
|
|
|
|
|
2017-06-02 15:03:29 +00:00
|
|
|
VirtualFramebuffer *displayFramebuf_ = nullptr;
|
|
|
|
VirtualFramebuffer *prevDisplayFramebuf_ = nullptr;
|
|
|
|
VirtualFramebuffer *prevPrevDisplayFramebuf_ = nullptr;
|
|
|
|
int frameLastFramebufUsed_ = 0;
|
2014-09-09 15:12:42 +00:00
|
|
|
|
2017-06-02 15:03:29 +00:00
|
|
|
VirtualFramebuffer *currentRenderVfb_ = nullptr;
|
2014-09-09 15:12:42 +00:00
|
|
|
|
|
|
|
// The range of PSP memory that may contain FBOs. So we can skip iterating.
|
2017-06-02 15:03:29 +00:00
|
|
|
u32 framebufRangeEnd_ = 0;
|
2014-09-09 15:12:42 +00:00
|
|
|
|
2017-06-02 15:03:29 +00:00
|
|
|
bool useBufferedRendering_ = false;
|
|
|
|
bool usePostShader_ = false;
|
|
|
|
bool postShaderAtOutputResolution_ = false;
|
|
|
|
bool postShaderIsUpscalingFilter_ = false;
|
2018-04-01 16:47:26 +00:00
|
|
|
int postShaderSSAAFilterLevel_ = 0;
|
2014-09-10 05:56:54 +00:00
|
|
|
|
2014-09-09 15:12:42 +00:00
|
|
|
std::vector<VirtualFramebuffer *> vfbs_;
|
2016-01-05 04:40:07 +00:00
|
|
|
std::vector<VirtualFramebuffer *> bvfbs_; // blitting framebuffers (for download)
|
2014-09-10 05:56:54 +00:00
|
|
|
|
2017-06-02 15:03:29 +00:00
|
|
|
bool gameUsesSequentialCopies_ = false;
|
2014-09-10 06:11:25 +00:00
|
|
|
|
2015-09-19 14:19:03 +00:00
|
|
|
// Sampled in BeginFrame for safety.
|
2018-03-22 21:25:04 +00:00
|
|
|
float renderWidth_ = 0.0f;
|
|
|
|
float renderHeight_ = 0.0f;
|
2015-09-19 14:19:03 +00:00
|
|
|
int pixelWidth_;
|
|
|
|
int pixelHeight_;
|
2017-04-14 06:35:07 +00:00
|
|
|
int bloomHack_ = 0;
|
|
|
|
bool trueColor_ = false;
|
2015-09-19 14:19:03 +00:00
|
|
|
|
2017-02-06 11:13:42 +00:00
|
|
|
// Used by post-processing shaders
|
|
|
|
std::vector<Draw::Framebuffer *> extraFBOs_;
|
|
|
|
|
2017-05-23 09:12:10 +00:00
|
|
|
bool needGLESRebinds_ = false;
|
2017-02-06 23:24:38 +00:00
|
|
|
|
2018-05-06 15:57:44 +00:00
|
|
|
struct TempFBOInfo {
|
2017-02-06 23:24:38 +00:00
|
|
|
Draw::Framebuffer *fbo;
|
|
|
|
int last_frame_used;
|
|
|
|
};
|
|
|
|
|
2018-06-03 15:08:45 +00:00
|
|
|
std::unordered_map<u64, TempFBOInfo> tempFBOs_;
|
2017-02-06 23:24:38 +00:00
|
|
|
|
2017-10-25 18:28:12 +00:00
|
|
|
std::vector<Draw::Framebuffer *> fbosToDelete_;
|
|
|
|
|
2014-09-10 06:11:25 +00:00
|
|
|
// Aggressively delete unused FBOs to save gpu memory.
|
|
|
|
enum {
|
|
|
|
FBO_OLD_AGE = 5,
|
2015-03-14 21:58:32 +00:00
|
|
|
FBO_OLD_USAGE_FLAG = 15,
|
2014-09-10 06:11:25 +00:00
|
|
|
};
|
2014-09-09 15:12:42 +00:00
|
|
|
};
|
2015-05-12 19:01:15 +00:00
|
|
|
|
2015-11-02 23:24:19 +00:00
|
|
|
void CenterDisplayOutputRect(float *x, float *y, float *w, float *h, float origW, float origH, float frameW, float frameH, int rotation);
|