2019-12-22 23:13:09 +00:00
|
|
|
/*
|
|
|
|
** BuilderNative Renderer
|
|
|
|
** Copyright (c) 2019 Magnus Norddahl
|
|
|
|
**
|
|
|
|
** This software is provided 'as-is', without any express or implied
|
|
|
|
** warranty. In no event will the authors be held liable for any damages
|
|
|
|
** arising from the use of this software.
|
|
|
|
**
|
|
|
|
** Permission is granted to anyone to use this software for any purpose,
|
|
|
|
** including commercial applications, and to alter it and redistribute it
|
|
|
|
** freely, subject to the following restrictions:
|
|
|
|
**
|
|
|
|
** 1. The origin of this software must not be misrepresented; you must not
|
|
|
|
** claim that you wrote the original software. If you use this software
|
|
|
|
** in a product, an acknowledgment in the product documentation would be
|
|
|
|
** appreciated but is not required.
|
|
|
|
** 2. Altered source versions must be plainly marked as such, and must not be
|
|
|
|
** misrepresented as being the original software.
|
|
|
|
** 3. This notice may not be removed or altered from any source distribution.
|
|
|
|
*/
|
2019-08-12 06:33:40 +00:00
|
|
|
|
|
|
|
#include "Precomp.h"
|
2019-12-23 19:09:38 +00:00
|
|
|
#include "GLShader.h"
|
|
|
|
#include "GLRenderDevice.h"
|
2019-08-12 06:33:40 +00:00
|
|
|
#include <stdexcept>
|
|
|
|
|
2019-12-23 19:09:38 +00:00
|
|
|
void GLShader::Setup(const std::string& identifier, const std::string& vertexShader, const std::string& fragmentShader, bool alphatest)
|
2019-08-12 06:33:40 +00:00
|
|
|
{
|
2019-12-15 21:53:33 +00:00
|
|
|
mIdentifier = identifier;
|
2019-08-22 13:46:24 +00:00
|
|
|
mVertexText = vertexShader;
|
|
|
|
mFragmentText = fragmentShader;
|
|
|
|
mAlphatest = alphatest;
|
2019-08-12 06:33:40 +00:00
|
|
|
}
|
|
|
|
|
2019-12-23 19:09:38 +00:00
|
|
|
bool GLShader::CheckCompile(GLRenderDevice* device)
|
2019-08-22 13:46:24 +00:00
|
|
|
{
|
|
|
|
bool firstCall = !mProgramBuilt;
|
|
|
|
if (firstCall)
|
|
|
|
{
|
|
|
|
mProgramBuilt = true;
|
2019-12-18 03:22:47 +00:00
|
|
|
CreateProgram(device);
|
2019-12-15 21:53:33 +00:00
|
|
|
glUseProgram(mProgram);
|
|
|
|
glUniform1i(glGetUniformLocation(mProgram, "texture1"), 0);
|
2022-01-04 19:17:12 +00:00
|
|
|
glUniform1i(glGetUniformLocation(mProgram, "texture2"), 1);
|
|
|
|
glUniform1i(glGetUniformLocation(mProgram, "texture3"), 2);
|
2019-12-15 21:53:33 +00:00
|
|
|
glUseProgram(0);
|
2019-08-22 13:46:24 +00:00
|
|
|
}
|
|
|
|
|
2019-12-15 21:53:33 +00:00
|
|
|
return !mErrors.size();
|
|
|
|
}
|
2019-08-22 13:46:24 +00:00
|
|
|
|
2019-12-23 19:09:38 +00:00
|
|
|
std::string GLShader::GetCompileError()
|
2019-12-15 21:53:33 +00:00
|
|
|
{
|
|
|
|
std::string lines = "Error compiling ";
|
|
|
|
if (!mVertexShader)
|
|
|
|
lines += "vertex ";
|
|
|
|
else if (!mFragmentShader)
|
|
|
|
lines += "fragment ";
|
|
|
|
lines += "shader \"" + mIdentifier + "\":\r\n";
|
|
|
|
for (auto c : mErrors)
|
2019-08-22 13:46:24 +00:00
|
|
|
{
|
2019-12-15 21:53:33 +00:00
|
|
|
if (c == '\r')
|
|
|
|
continue;
|
|
|
|
if (c == '\n')
|
|
|
|
lines += "\r\n";
|
|
|
|
else lines += c;
|
2019-08-22 13:46:24 +00:00
|
|
|
}
|
2019-12-15 21:53:33 +00:00
|
|
|
return lines;
|
|
|
|
}
|
|
|
|
|
2019-12-23 19:09:38 +00:00
|
|
|
void GLShader::Bind()
|
2019-12-15 21:53:33 +00:00
|
|
|
{
|
|
|
|
if (!mProgram || !mProgramBuilt || mErrors.size())
|
|
|
|
return;
|
|
|
|
|
|
|
|
glUseProgram(mProgram);
|
2019-08-22 13:46:24 +00:00
|
|
|
}
|
|
|
|
|
2019-12-23 19:09:38 +00:00
|
|
|
void GLShader::CreateProgram(GLRenderDevice* device)
|
2019-08-12 06:33:40 +00:00
|
|
|
{
|
2019-08-16 02:10:03 +00:00
|
|
|
const char* prefixNAT = R"(
|
2020-01-01 20:41:04 +00:00
|
|
|
#version 330
|
2019-08-16 02:10:03 +00:00
|
|
|
#line 1
|
|
|
|
)";
|
|
|
|
const char* prefixAT = R"(
|
2020-01-01 20:41:04 +00:00
|
|
|
#version 330
|
2019-08-16 02:10:03 +00:00
|
|
|
#define ALPHA_TEST
|
|
|
|
#line 1
|
|
|
|
)";
|
|
|
|
|
2019-08-22 13:46:24 +00:00
|
|
|
const char* prefix = mAlphatest ? prefixAT : prefixNAT;
|
2019-08-16 02:10:03 +00:00
|
|
|
|
2019-08-22 13:46:24 +00:00
|
|
|
mVertexShader = CompileShader(prefix + mVertexText, GL_VERTEX_SHADER);
|
2019-08-12 06:33:40 +00:00
|
|
|
if (!mVertexShader)
|
2019-08-22 13:46:24 +00:00
|
|
|
return;
|
2019-08-12 06:33:40 +00:00
|
|
|
|
2019-08-22 13:46:24 +00:00
|
|
|
mFragmentShader = CompileShader(prefix + mFragmentText, GL_FRAGMENT_SHADER);
|
2019-08-12 06:33:40 +00:00
|
|
|
if (!mFragmentShader)
|
2019-08-22 13:46:24 +00:00
|
|
|
return;
|
2019-08-12 06:33:40 +00:00
|
|
|
|
|
|
|
mProgram = glCreateProgram();
|
|
|
|
glAttachShader(mProgram, mVertexShader);
|
|
|
|
glAttachShader(mProgram, mFragmentShader);
|
|
|
|
glBindAttribLocation(mProgram, (GLuint)DeclarationUsage::Position, "AttrPosition");
|
|
|
|
glBindAttribLocation(mProgram, (GLuint)DeclarationUsage::Color, "AttrColor");
|
|
|
|
glBindAttribLocation(mProgram, (GLuint)DeclarationUsage::TextureCoordinate, "AttrUV");
|
|
|
|
glBindAttribLocation(mProgram, (GLuint)DeclarationUsage::Normal, "AttrNormal");
|
|
|
|
glLinkProgram(mProgram);
|
|
|
|
|
|
|
|
GLint status = 0;
|
|
|
|
glGetProgramiv(mProgram, GL_LINK_STATUS, &status);
|
|
|
|
if (status != GL_TRUE)
|
|
|
|
{
|
|
|
|
GLsizei length = 0;
|
|
|
|
glGetProgramiv(mProgram, GL_INFO_LOG_LENGTH, &length);
|
|
|
|
std::vector<GLchar> errors(length + (size_t)1);
|
|
|
|
glGetProgramInfoLog(mProgram, (GLsizei)errors.size(), &length, errors.data());
|
|
|
|
mErrors = { errors.begin(), errors.begin() + length };
|
|
|
|
|
|
|
|
glDeleteProgram(mProgram);
|
|
|
|
glDeleteShader(mVertexShader);
|
|
|
|
glDeleteShader(mFragmentShader);
|
|
|
|
mProgram = 0;
|
|
|
|
mVertexShader = 0;
|
|
|
|
mFragmentShader = 0;
|
2019-08-22 13:46:24 +00:00
|
|
|
return;
|
2019-08-12 06:33:40 +00:00
|
|
|
}
|
|
|
|
|
2019-12-21 01:14:42 +00:00
|
|
|
UniformLastUpdates.resize(device->mUniformInfo.size());
|
|
|
|
UniformLocations.resize(device->mUniformInfo.size(), (GLuint)-1);
|
|
|
|
|
|
|
|
int count = (int)UniformLocations.size();
|
|
|
|
for (int i = 0; i < count; i++)
|
2019-08-14 10:36:33 +00:00
|
|
|
{
|
2019-12-21 01:14:42 +00:00
|
|
|
const auto& name = device->mUniformInfo[i].Name;
|
|
|
|
if (!name.empty())
|
|
|
|
UniformLocations[i] = glGetUniformLocation(mProgram, name.c_str());
|
2019-08-14 10:36:33 +00:00
|
|
|
}
|
2019-08-12 06:33:40 +00:00
|
|
|
}
|
|
|
|
|
2019-12-23 19:09:38 +00:00
|
|
|
GLuint GLShader::CompileShader(const std::string& code, GLenum type)
|
2019-08-12 06:33:40 +00:00
|
|
|
{
|
|
|
|
GLuint shader = glCreateShader(type);
|
|
|
|
const GLchar* sources[] = { (GLchar*)code.data() };
|
|
|
|
const GLint lengths[] = { (GLint)code.size() };
|
|
|
|
glShaderSource(shader, 1, sources, lengths);
|
|
|
|
glCompileShader(shader);
|
|
|
|
|
|
|
|
GLint status = 0;
|
|
|
|
glGetShaderiv(shader, GL_COMPILE_STATUS, &status);
|
|
|
|
if (status != GL_TRUE)
|
|
|
|
{
|
|
|
|
GLsizei length = 0;
|
|
|
|
glGetShaderiv(shader, GL_INFO_LOG_LENGTH, &length);
|
|
|
|
std::vector<GLchar> errors(length + (size_t)1);
|
|
|
|
glGetShaderInfoLog(shader, (GLsizei)errors.size(), &length, errors.data());
|
|
|
|
mErrors = { errors.begin(), errors.begin() + length };
|
|
|
|
glDeleteShader(shader);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return shader;
|
|
|
|
}
|
2019-08-22 13:46:24 +00:00
|
|
|
|
2019-12-23 19:09:38 +00:00
|
|
|
void GLShader::ReleaseResources()
|
2019-08-22 13:46:24 +00:00
|
|
|
{
|
|
|
|
if (mProgram)
|
|
|
|
glDeleteProgram(mProgram);
|
|
|
|
if (mVertexShader)
|
|
|
|
glDeleteShader(mVertexShader);
|
|
|
|
if (mFragmentShader)
|
|
|
|
glDeleteShader(mFragmentShader);
|
|
|
|
mProgram = 0;
|
|
|
|
mVertexShader = 0;
|
|
|
|
mFragmentShader = 0;
|
|
|
|
}
|