mirror of
https://bitbucket.org/CPMADevs/cnq3
synced 2024-12-02 08:52:28 +00:00
361 lines
8.3 KiB
C++
361 lines
8.3 KiB
C++
/*
|
|
===========================================================================
|
|
Copyright (C) 2022-2024 Gian 'myT' Schellenbaum
|
|
|
|
This file is part of Challenge Quake 3 (CNQ3).
|
|
|
|
Challenge Quake 3 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; either version 2 of the License,
|
|
or (at your option) any later version.
|
|
|
|
Challenge Quake 3 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 for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with Challenge Quake 3. If not, see <https://www.gnu.org/licenses/>.
|
|
===========================================================================
|
|
*/
|
|
// Gameplay Rendering Pipeline - private declarations
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
#include "srp_local.h"
|
|
|
|
|
|
#pragma pack(push, 4)
|
|
|
|
struct WorldVertexRC
|
|
{
|
|
float modelViewMatrix[16];
|
|
float projectionMatrix[16];
|
|
float clipPlane[4];
|
|
};
|
|
|
|
struct WorldPixelRC
|
|
{
|
|
// general
|
|
uint32_t stageIndices[8]; // sampler: 16 - texture: 16
|
|
float greyscale;
|
|
|
|
// r_shaderTrace - dynamically enabled
|
|
uint32_t shaderTrace; // shader index: 14 - enable: 1
|
|
uint16_t centerPixelX;
|
|
uint16_t centerPixelY;
|
|
|
|
// r_depthFade - statically enabled
|
|
uint16_t hFadeDistance;
|
|
uint16_t hFadeOffset;
|
|
uint32_t depthFadeColorTex; // texture index: 12 - color bias: 4 - color scale: 4
|
|
|
|
// r_dither - statically enabled
|
|
uint16_t hFrameSeed;
|
|
uint16_t hNoiseScale;
|
|
uint16_t hInvGamma;
|
|
uint16_t hInvBrightness;
|
|
};
|
|
|
|
#pragma pack(pop)
|
|
|
|
struct VertexBuffers : BufferBase
|
|
{
|
|
enum BaseId
|
|
{
|
|
BasePosition,
|
|
BaseNormal,
|
|
BaseCount
|
|
};
|
|
|
|
enum StageId
|
|
{
|
|
StageTexCoords,
|
|
StageColors,
|
|
StageCount
|
|
};
|
|
|
|
void Create(const char* name, MemoryUsage::Id memoryUsage, uint32_t vertexCount);
|
|
void BeginUpload();
|
|
void EndUpload();
|
|
void Upload(uint32_t firstStage, uint32_t stageCount);
|
|
|
|
static const uint32_t BufferCount = BaseCount + StageCount * MAX_SHADER_STAGES;
|
|
HBuffer buffers[BufferCount] = {};
|
|
uint32_t strides[BufferCount] = {};
|
|
uint8_t* mapped[BufferCount] = {};
|
|
};
|
|
|
|
struct GeometryBuffers
|
|
{
|
|
void Rewind()
|
|
{
|
|
vertexBuffers.Rewind();
|
|
indexBuffer.Rewind();
|
|
}
|
|
|
|
VertexBuffers vertexBuffers;
|
|
IndexBuffer indexBuffer;
|
|
};
|
|
|
|
struct StaticGeometryChunk
|
|
{
|
|
uint32_t vertexCount;
|
|
uint32_t indexCount;
|
|
uint32_t firstGPUVertex;
|
|
uint32_t firstGPUIndex;
|
|
uint32_t firstCPUIndex;
|
|
};
|
|
|
|
struct BatchType
|
|
{
|
|
enum Id
|
|
{
|
|
Standard,
|
|
DynamicLight,
|
|
DepthFade,
|
|
Count
|
|
};
|
|
};
|
|
|
|
struct World
|
|
{
|
|
void Init();
|
|
void BeginFrame();
|
|
void EndFrame();
|
|
void Begin();
|
|
void End();
|
|
void DrawPrePass(const drawSceneViewCommand_t& cmd);
|
|
void BeginBatch(const shader_t* shader, bool hasStaticGeo, BatchType::Id batchType);
|
|
void EndBatch();
|
|
void EndSkyBatch();
|
|
void RestartBatch();
|
|
void DrawGUI();
|
|
void ProcessWorld(world_t& world);
|
|
void DrawSceneView(const drawSceneViewCommand_t& cmd);
|
|
void BindVertexBuffers(bool staticGeo, uint32_t firstStage, uint32_t stageCount);
|
|
void BindIndexBuffer(bool staticGeo);
|
|
void DrawFog();
|
|
void DrawLitSurfaces(dlight_t* dl, bool opaque);
|
|
void DrawDynamicLights(bool opaque);
|
|
void DrawSkyBox();
|
|
void DrawClouds();
|
|
|
|
typedef uint32_t Index;
|
|
const IndexType::Id indexType = IndexType::UInt32;
|
|
|
|
HTexture depthTexture;
|
|
uint32_t depthTextureIndex;
|
|
|
|
float clipPlane[4];
|
|
|
|
struct BufferFamily
|
|
{
|
|
enum Id
|
|
{
|
|
Invalid,
|
|
PrePass,
|
|
Static,
|
|
Dynamic
|
|
};
|
|
};
|
|
|
|
// Z pre-pass
|
|
HRootSignature zppRootSignature;
|
|
HDescriptorTable zppDescriptorTable;
|
|
HPipeline zppPipeline;
|
|
GeometryBuffer zppVertexBuffer;
|
|
|
|
// shared
|
|
BufferFamily::Id boundVertexBuffers;
|
|
BufferFamily::Id boundIndexBuffer;
|
|
uint32_t boundStaticVertexBuffersFirst;
|
|
uint32_t boundStaticVertexBuffersCount;
|
|
HPipeline batchPSO;
|
|
BatchType::Id batchType;
|
|
bool batchHasStaticGeo;
|
|
int psoChangeCount;
|
|
bool batchDepthHack;
|
|
bool batchOldDepthHack;
|
|
ShadingRate::Id batchShadingRate;
|
|
ShadingRate::Id batchOldShadingRate;
|
|
|
|
// dynamic
|
|
GeometryBuffers dynBuffers[FrameCount];
|
|
|
|
// static
|
|
GeometryBuffers statBuffers;
|
|
StaticGeometryChunk statChunks[32768];
|
|
uint32_t statChunkCount;
|
|
uint32_t statIndices[1 << 20];
|
|
uint32_t statIndexCount;
|
|
|
|
// fog
|
|
HRootSignature fogRootSignature;
|
|
HDescriptorTable fogDescriptorTable;
|
|
HPipeline fogOutsidePipeline;
|
|
HPipeline fogInsidePipeline;
|
|
HBuffer boxVertexBuffer;
|
|
HBuffer boxIndexBuffer;
|
|
|
|
// dynamic lights
|
|
HRootSignature dlRootSignature;
|
|
HPipeline dlPipelines[CT_COUNT * 2 * 2]; // { cull type, polygon offset, depth test }
|
|
bool dlOpaque;
|
|
float dlIntensity; // 1 for most surfaces, but can be scaled down for liquids etc.
|
|
bool dlDepthTestEqual;
|
|
// quick explanation on why dlOpaque is useful in the first place:
|
|
// - opaque surfaces can have a diffuse texture whose alpha isn't 255 everywhere
|
|
// - when that happens and we multiply the color by the the alpha (DL uses additive blending),
|
|
// we get "light holes" in opaque surfaces, which is not what we want
|
|
};
|
|
|
|
#pragma pack(push, 1)
|
|
|
|
struct PSODesc
|
|
{
|
|
cullType_t cullType;
|
|
bool polygonOffset;
|
|
bool clampDepth;
|
|
bool depthFade;
|
|
};
|
|
|
|
#pragma pack(pop)
|
|
|
|
struct CachedPSO
|
|
{
|
|
#if defined(_DEBUG)
|
|
// lets us know which Q3 shader
|
|
// triggered the creation of this PSO
|
|
char name[MAX_QPATH];
|
|
#endif
|
|
PSODesc desc;
|
|
uint32_t stageStateBits[MAX_SHADER_STAGES];
|
|
uint32_t stageCount;
|
|
HPipeline pipeline;
|
|
};
|
|
|
|
struct PostProcess
|
|
{
|
|
void Init();
|
|
void Draw(const char* renderPassName, HTexture renderTarget);
|
|
void ToneMap();
|
|
void InverseToneMap(int colorFormat);
|
|
void SetToneMapInput(HTexture toneMapInput);
|
|
void SetInverseToneMapInput(HTexture inverseToneMapInput);
|
|
|
|
private:
|
|
HPipeline toneMapPipeline;
|
|
HRootSignature toneMapRootSignature;
|
|
HDescriptorTable toneMapDescriptorTable;
|
|
|
|
HPipeline inverseToneMapPipelines[RTCF_COUNT];
|
|
HRootSignature inverseToneMapRootSignature;
|
|
HDescriptorTable inverseToneMapDescriptorTable;
|
|
};
|
|
|
|
struct SMAA
|
|
{
|
|
void Init();
|
|
void Update();
|
|
void Draw(const viewParms_t& parms);
|
|
|
|
// matches r_smaa
|
|
struct Mode
|
|
{
|
|
enum Id
|
|
{
|
|
Disabled,
|
|
Low,
|
|
Medium,
|
|
High,
|
|
Ultra,
|
|
Count
|
|
};
|
|
};
|
|
|
|
private:
|
|
// fixed
|
|
HTexture areaTexture;
|
|
HTexture searchTexture;
|
|
HRootSignature rootSignature;
|
|
HDescriptorTable descriptorTable;
|
|
|
|
// depends on render target resolution
|
|
HTexture edgeTexture;
|
|
HTexture blendTexture;
|
|
HTexture stencilTexture;
|
|
HTexture inputTexture; // tone mapped
|
|
HTexture outputTexture; // tone mapped
|
|
|
|
// depends on selected preset/mode
|
|
// SMAA has 3 passes:
|
|
// 1. edge detection
|
|
// 2. blend weight computation
|
|
// 3. neighborhood blending
|
|
HPipeline firstPassPipeline;
|
|
HPipeline secondPassPipeline;
|
|
HPipeline thirdPassPipeline;
|
|
|
|
Mode::Id mode = Mode::Disabled;
|
|
int width = -1;
|
|
int height = -1;
|
|
bool fixedLoaded = false;
|
|
};
|
|
|
|
struct GRP : IRenderPipeline
|
|
{
|
|
void Init() override;
|
|
void ShutDown(bool fullShutDown) override;
|
|
void CreateTexture(image_t* image, int mipCount, int width, int height) override;
|
|
void UpoadTextureAndGenerateMipMaps(image_t* image, const byte* data) override;
|
|
void BeginTextureUpload(MappedTexture& mappedTexture, image_t* image) override;
|
|
void EndTextureUpload() override;
|
|
void ProcessWorld(world_t& world) override;
|
|
void ProcessModel(model_t& model) override;
|
|
void ProcessShader(shader_t& shader) override;
|
|
|
|
void ExecuteRenderCommands(const byte* data, bool readbackRequested) override;
|
|
void TessellationOverflow() override { world.RestartBatch(); }
|
|
void DrawSkyBox() override { world.DrawSkyBox(); }
|
|
void DrawClouds() override { world.DrawClouds(); }
|
|
void ReadPixels(int w, int h, int alignment, colorSpace_t colorSpace, void* out) override;
|
|
|
|
uint32_t GetSamplerDescriptorIndexFromBaseIndex(uint32_t baseIndex) override { return baseIndex; }
|
|
|
|
void BeginFrame();
|
|
void EndFrame();
|
|
|
|
uint32_t RegisterTexture(HTexture htexture);
|
|
|
|
uint32_t CreatePSO(CachedPSO& cache, const char* name);
|
|
|
|
void UpdateReadbackTexture();
|
|
|
|
UI ui;
|
|
World world;
|
|
MipMapGenerator mipMapGen;
|
|
ImGUI imgui;
|
|
PostProcess post;
|
|
SMAA smaa;
|
|
Nuklear nuklear;
|
|
float frameSeed;
|
|
bool updateReadbackTexture;
|
|
|
|
HTexture renderTarget;
|
|
TextureFormat::Id renderTargetFormat;
|
|
HTexture readbackRenderTarget;
|
|
RootSignatureDesc rootSignatureDesc;
|
|
HRootSignature rootSignature;
|
|
HDescriptorTable descriptorTable;
|
|
uint32_t textureIndex;
|
|
HSampler samplers[BASE_SAMPLER_COUNT]; // all base samplers
|
|
|
|
CachedPSO psos[1024];
|
|
uint32_t psoCount;
|
|
HRootSignature uberRootSignature;
|
|
};
|
|
|
|
extern GRP grp;
|