2017-07-06 03:36:01 +00:00
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
// Copyright(C) 2017 Magnus Norddahl
|
|
|
|
// All rights reserved.
|
|
|
|
//
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU Lesser General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
//
|
|
|
|
// 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 Lesser General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU Lesser General Public License
|
|
|
|
// along with this program. If not, see http://www.gnu.org/licenses/
|
|
|
|
//
|
|
|
|
//--------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
|
|
|
|
#include "gl/system/gl_system.h"
|
|
|
|
#include "m_swap.h"
|
|
|
|
#include "v_video.h"
|
|
|
|
#include "vectors.h"
|
|
|
|
#include "w_wad.h"
|
|
|
|
#include "gl/system/gl_interface.h"
|
|
|
|
#include "gl/system/gl_framebuffer.h"
|
|
|
|
#include "gl/system/gl_debug.h"
|
|
|
|
#include "gl/system/gl_cvars.h"
|
|
|
|
#include "gl/renderer/gl_renderer.h"
|
|
|
|
#include "gl/renderer/gl_postprocessstate.h"
|
|
|
|
#include "gl/renderer/gl_renderbuffers.h"
|
|
|
|
#include "gl/shaders/gl_postprocessshader.h"
|
2018-04-02 10:28:20 +00:00
|
|
|
#include "gl/shaders/gl_postprocessshaderinstance.h"
|
2017-07-30 11:59:25 +00:00
|
|
|
#include "textures/textures.h"
|
|
|
|
#include "textures/bitmap.h"
|
2017-07-06 03:36:01 +00:00
|
|
|
|
|
|
|
CVAR(Bool, gl_custompost, true, 0)
|
|
|
|
|
|
|
|
TArray<PostProcessShader> PostProcessShaders;
|
|
|
|
|
|
|
|
FCustomPostProcessShaders::FCustomPostProcessShaders()
|
|
|
|
{
|
|
|
|
for (unsigned int i = 0; i < PostProcessShaders.Size(); i++)
|
|
|
|
{
|
|
|
|
mShaders.push_back(std::unique_ptr<PostProcessShaderInstance>(new PostProcessShaderInstance(&PostProcessShaders[i])));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
FCustomPostProcessShaders::~FCustomPostProcessShaders()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void FCustomPostProcessShaders::Run(FString target)
|
|
|
|
{
|
|
|
|
if (!gl_custompost)
|
|
|
|
return;
|
|
|
|
|
|
|
|
for (auto &shader : mShaders)
|
|
|
|
{
|
|
|
|
if (shader->Desc->Target == target)
|
|
|
|
{
|
|
|
|
shader->Run();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2017-07-30 11:59:25 +00:00
|
|
|
PostProcessShaderInstance::~PostProcessShaderInstance()
|
|
|
|
{
|
|
|
|
for (const auto &it : mTextureHandles)
|
|
|
|
glDeleteTextures(1, (GLuint*)&it.second);
|
|
|
|
}
|
|
|
|
|
2017-07-06 03:36:01 +00:00
|
|
|
void PostProcessShaderInstance::Run()
|
|
|
|
{
|
2017-07-08 12:44:07 +00:00
|
|
|
if (!IsShaderSupported())
|
|
|
|
return;
|
|
|
|
|
|
|
|
CompileShader();
|
|
|
|
|
|
|
|
if (!Desc->Enabled)
|
|
|
|
return;
|
2017-07-06 03:36:01 +00:00
|
|
|
|
|
|
|
FGLDebug::PushGroup(Desc->ShaderLumpName.GetChars());
|
|
|
|
|
|
|
|
FGLPostProcessState savedState;
|
2017-07-30 11:59:25 +00:00
|
|
|
savedState.SaveTextureBindings(1 + Desc->Textures.CountUsed());
|
2017-07-06 03:36:01 +00:00
|
|
|
|
|
|
|
GLRenderer->mBuffers->BindNextFB();
|
|
|
|
GLRenderer->mBuffers->BindCurrentTexture(0);
|
|
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
|
|
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
|
|
|
|
|
2017-07-08 12:44:07 +00:00
|
|
|
mProgram.Bind();
|
2017-07-06 03:36:01 +00:00
|
|
|
|
2017-07-08 12:44:07 +00:00
|
|
|
UpdateUniforms();
|
2017-07-30 11:59:25 +00:00
|
|
|
BindTextures();
|
2017-07-06 03:36:01 +00:00
|
|
|
|
2017-07-08 12:44:07 +00:00
|
|
|
mInputTexture.Set(0);
|
2017-07-06 03:36:01 +00:00
|
|
|
|
|
|
|
GLRenderer->RenderScreenQuad();
|
|
|
|
|
2017-07-30 11:59:25 +00:00
|
|
|
glActiveTexture(GL_TEXTURE0);
|
2017-07-06 03:36:01 +00:00
|
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
|
|
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
|
|
|
|
GLRenderer->mBuffers->NextTexture();
|
|
|
|
|
|
|
|
FGLDebug::PopGroup();
|
|
|
|
}
|
2017-07-08 12:44:07 +00:00
|
|
|
|
|
|
|
bool PostProcessShaderInstance::IsShaderSupported()
|
|
|
|
{
|
|
|
|
int activeShaderVersion = (int)round(gl.glslversion * 10) * 10;
|
|
|
|
return activeShaderVersion >= Desc->ShaderVersion;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PostProcessShaderInstance::CompileShader()
|
|
|
|
{
|
|
|
|
if (mProgram)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// Get the custom shader
|
|
|
|
const char *lumpName = Desc->ShaderLumpName.GetChars();
|
|
|
|
int lump = Wads.CheckNumForFullName(lumpName);
|
|
|
|
if (lump == -1) I_FatalError("Unable to load '%s'", lumpName);
|
|
|
|
FString code = Wads.ReadLump(lump).GetString().GetChars();
|
|
|
|
|
|
|
|
// Build an uniform block to use be used as input
|
|
|
|
// (this is technically not an uniform block, but it could be changed into that for Vulkan GLSL support)
|
|
|
|
FString uniformBlock;
|
|
|
|
TMap<FString, PostProcessUniformValue>::Iterator it(Desc->Uniforms);
|
|
|
|
TMap<FString, PostProcessUniformValue>::Pair *pair;
|
|
|
|
while (it.NextPair(pair))
|
|
|
|
{
|
|
|
|
FString type;
|
|
|
|
FString name = pair->Key;
|
|
|
|
|
|
|
|
switch (pair->Value.Type)
|
|
|
|
{
|
|
|
|
case PostProcessUniformType::Float: type = "float"; break;
|
|
|
|
case PostProcessUniformType::Int: type = "int"; break;
|
|
|
|
case PostProcessUniformType::Vec2: type = "vec2"; break;
|
|
|
|
case PostProcessUniformType::Vec3: type = "vec3"; break;
|
|
|
|
default: break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!type.IsEmpty())
|
|
|
|
uniformBlock.AppendFormat("uniform %s %s;\n", type.GetChars(), name.GetChars());
|
|
|
|
}
|
|
|
|
|
|
|
|
// Build the input textures
|
|
|
|
FString uniformTextures;
|
|
|
|
uniformTextures += "uniform sampler2D InputTexture;\n";
|
|
|
|
|
2017-07-30 11:59:25 +00:00
|
|
|
TMap<FString, FString>::Iterator itTextures(Desc->Textures);
|
|
|
|
TMap<FString, FString>::Pair *pairTextures;
|
|
|
|
while (itTextures.NextPair(pairTextures))
|
|
|
|
{
|
|
|
|
uniformTextures.AppendFormat("uniform sampler2D %s;\n", pairTextures->Key.GetChars());
|
|
|
|
}
|
|
|
|
|
2017-07-08 12:44:07 +00:00
|
|
|
// Setup pipeline
|
|
|
|
FString pipelineInOut;
|
|
|
|
pipelineInOut += "in vec2 TexCoord;\n";
|
|
|
|
pipelineInOut += "out vec4 FragColor;\n";
|
|
|
|
|
|
|
|
FString prolog;
|
|
|
|
prolog += uniformBlock;
|
|
|
|
prolog += uniformTextures;
|
|
|
|
prolog += pipelineInOut;
|
|
|
|
|
|
|
|
mProgram.Compile(FShaderProgram::Vertex, "shaders/glsl/screenquad.vp", "", Desc->ShaderVersion);
|
|
|
|
mProgram.Compile(FShaderProgram::Fragment, lumpName, code, prolog.GetChars(), Desc->ShaderVersion);
|
|
|
|
mProgram.SetFragDataLocation(0, "FragColor");
|
|
|
|
mProgram.Link(Desc->ShaderLumpName.GetChars());
|
|
|
|
mProgram.SetAttribLocation(0, "PositionInProjection");
|
|
|
|
mInputTexture.Init(mProgram, "InputTexture");
|
|
|
|
}
|
|
|
|
|
|
|
|
void PostProcessShaderInstance::UpdateUniforms()
|
|
|
|
{
|
|
|
|
TMap<FString, PostProcessUniformValue>::Iterator it(Desc->Uniforms);
|
|
|
|
TMap<FString, PostProcessUniformValue>::Pair *pair;
|
|
|
|
while (it.NextPair(pair))
|
|
|
|
{
|
|
|
|
int location = glGetUniformLocation(mProgram, pair->Key.GetChars());
|
|
|
|
if (location != -1)
|
|
|
|
{
|
|
|
|
switch (pair->Value.Type)
|
|
|
|
{
|
|
|
|
case PostProcessUniformType::Float:
|
|
|
|
glUniform1f(location, (float)pair->Value.Values[0]);
|
|
|
|
break;
|
|
|
|
case PostProcessUniformType::Int:
|
|
|
|
glUniform1i(location, (int)pair->Value.Values[0]);
|
|
|
|
break;
|
|
|
|
case PostProcessUniformType::Vec2:
|
|
|
|
glUniform2f(location, (float)pair->Value.Values[0], (float)pair->Value.Values[1]);
|
|
|
|
break;
|
|
|
|
case PostProcessUniformType::Vec3:
|
|
|
|
glUniform3f(location, (float)pair->Value.Values[0], (float)pair->Value.Values[1], (float)pair->Value.Values[2]);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-07-30 11:59:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void PostProcessShaderInstance::BindTextures()
|
|
|
|
{
|
|
|
|
int textureUnit = 1;
|
|
|
|
TMap<FString, FString>::Iterator it(Desc->Textures);
|
|
|
|
TMap<FString, FString>::Pair *pair;
|
|
|
|
while (it.NextPair(pair))
|
|
|
|
{
|
|
|
|
int location = glGetUniformLocation(mProgram, pair->Key.GetChars());
|
|
|
|
if (location == -1)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
FString name = pair->Value;
|
2018-03-25 18:26:16 +00:00
|
|
|
FTexture *tex = TexMan(TexMan.CheckForTexture(name, ETextureType::Any));
|
|
|
|
if (tex && tex->UseType != ETextureType::Null)
|
2017-07-30 11:59:25 +00:00
|
|
|
{
|
|
|
|
glUniform1i(location, textureUnit);
|
|
|
|
|
|
|
|
glActiveTexture(GL_TEXTURE0 + 1);
|
|
|
|
auto it = mTextureHandles.find(tex);
|
|
|
|
if (it == mTextureHandles.end())
|
|
|
|
{
|
|
|
|
FBitmap bitmap;
|
|
|
|
bitmap.Create(tex->GetWidth(), tex->GetHeight());
|
2017-10-12 16:44:00 +00:00
|
|
|
tex->CopyTrueColorPixels(&bitmap, 0, 0);
|
2017-07-30 11:59:25 +00:00
|
|
|
|
|
|
|
GLuint handle = 0;
|
|
|
|
glGenTextures(1, &handle);
|
|
|
|
glBindTexture(GL_TEXTURE_2D, handle);
|
2017-10-12 15:54:56 +00:00
|
|
|
glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA8, tex->GetWidth(), tex->GetHeight(), 0, GL_BGRA, GL_UNSIGNED_BYTE, bitmap.GetPixels());
|
2017-07-30 11:59:25 +00:00
|
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
|
|
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
|
|
|
|
mTextureHandles[tex] = handle;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
glBindTexture(GL_TEXTURE_2D, it->second);
|
|
|
|
}
|
|
|
|
|
|
|
|
textureUnit++;
|
|
|
|
}
|
|
|
|
}
|
2017-07-08 12:44:07 +00:00
|
|
|
}
|