mirror of
https://github.com/MaSzyna-EU07/maszyna.git
synced 2026-03-22 15:05:03 +01:00
335 lines
8.9 KiB
C++
335 lines
8.9 KiB
C++
#include "stdafx.h"
|
|
|
|
#include <fstream>
|
|
#include <sstream>
|
|
#include "shader.h"
|
|
#include "glsl_common.h"
|
|
#include "Logs.h"
|
|
|
|
inline bool strcend(std::string const &value, std::string const &ending)
|
|
{
|
|
if (ending.size() > value.size())
|
|
return false;
|
|
return std::equal(ending.rbegin(), ending.rend(), value.rbegin());
|
|
}
|
|
|
|
std::string gl::shader::read_file(const std::string &filename)
|
|
{
|
|
std::stringstream stream;
|
|
std::ifstream f;
|
|
f.exceptions(std::ifstream::badbit);
|
|
|
|
f.open("shaders/" + filename);
|
|
stream << f.rdbuf();
|
|
f.close();
|
|
|
|
std::string str = stream.str();
|
|
|
|
return str;
|
|
}
|
|
|
|
void gl::shader::expand_includes(std::string &str)
|
|
{
|
|
size_t start_pos = 0;
|
|
|
|
std::string magic = "#include";
|
|
while ((start_pos = str.find(magic, start_pos)) != str.npos)
|
|
{
|
|
size_t fp = str.find('<', start_pos);
|
|
size_t fe = str.find('>', start_pos);
|
|
if (fp == str.npos || fe == str.npos)
|
|
return;
|
|
|
|
std::string filename = str.substr(fp + 1, fe - fp - 1);
|
|
std::string content;
|
|
if (filename != "common")
|
|
content = read_file(filename);
|
|
else
|
|
content = glsl_common;
|
|
|
|
str.replace(start_pos, fe - start_pos + 1, content);
|
|
}
|
|
}
|
|
|
|
std::unordered_map<std::string, gl::shader::components_e> gl::shader::components_mapping =
|
|
{
|
|
{ "R", components_e::R },
|
|
{ "RG", components_e::RG },
|
|
{ "RGB", components_e::RGB },
|
|
{ "RGBA", components_e::RGBA },
|
|
{ "sRGB", components_e::sRGB },
|
|
{ "sRGB_A", components_e::sRGB_A }
|
|
};
|
|
|
|
std::unordered_map<std::string, gl::shader::defaultparam_e> gl::shader::defaultparams_mapping =
|
|
{
|
|
{ "required", defaultparam_e::required },
|
|
{ "nan", defaultparam_e::nan },
|
|
{ "zero", defaultparam_e::zero },
|
|
{ "one", defaultparam_e::one },
|
|
{ "ambient", defaultparam_e::ambient },
|
|
{ "diffuse", defaultparam_e::diffuse },
|
|
{ "specular", defaultparam_e::specular }
|
|
};
|
|
|
|
void gl::shader::process_source(std::string &str)
|
|
{
|
|
expand_includes(str);
|
|
parse_texture_entries(str);
|
|
parse_param_entries(str);
|
|
}
|
|
|
|
void gl::shader::parse_texture_entries(std::string &str)
|
|
{
|
|
size_t start_pos = 0;
|
|
|
|
std::string magic = "#texture";
|
|
while ((start_pos = str.find(magic, start_pos)) != str.npos)
|
|
{
|
|
size_t fp = str.find('(', start_pos);
|
|
size_t fe = str.find(')', start_pos);
|
|
if (fp == str.npos || fe == str.npos)
|
|
return;
|
|
|
|
std::istringstream ss(str.substr(fp + 1, fe - fp - 1));
|
|
std::string token;
|
|
|
|
std::string name;
|
|
texture_entry conf;
|
|
|
|
size_t arg = 0;
|
|
while (std::getline(ss, token, ','))
|
|
{
|
|
std::istringstream token_ss(token);
|
|
if (arg == 0)
|
|
token_ss >> name;
|
|
else if (arg == 1)
|
|
token_ss >> conf.id;
|
|
else if (arg == 2)
|
|
{
|
|
std::string comp;
|
|
token_ss >> comp;
|
|
if (components_mapping.find(comp) == components_mapping.end())
|
|
log_error("unknown components: " + comp);
|
|
else
|
|
conf.components = components_mapping[comp];
|
|
}
|
|
arg++;
|
|
}
|
|
|
|
if (arg == 3)
|
|
{
|
|
if (name.empty())
|
|
log_error("empty name");
|
|
else if (conf.id >= gl::MAX_TEXTURES)
|
|
log_error("invalid texture binding: " + std::to_string(conf.id));
|
|
else
|
|
texture_conf.emplace(std::make_pair(name, conf));
|
|
}
|
|
else
|
|
log_error("invalid argument count to #texture");
|
|
|
|
str.erase(start_pos, fe - start_pos + 1);
|
|
}
|
|
}
|
|
|
|
void gl::shader::parse_param_entries(std::string &str)
|
|
{
|
|
size_t start_pos = 0;
|
|
|
|
std::string magic = "#param";
|
|
while ((start_pos = str.find(magic, start_pos)) != str.npos)
|
|
{
|
|
size_t fp = str.find('(', start_pos);
|
|
size_t fe = str.find(')', start_pos);
|
|
if (fp == str.npos || fe == str.npos)
|
|
return;
|
|
|
|
std::istringstream ss(str.substr(fp + 1, fe - fp - 1));
|
|
std::string token;
|
|
|
|
std::string name;
|
|
param_entry conf;
|
|
|
|
size_t arg = 0;
|
|
while (std::getline(ss, token, ','))
|
|
{
|
|
std::istringstream token_ss(token);
|
|
if (arg == 0)
|
|
token_ss >> name;
|
|
else if (arg == 1)
|
|
token_ss >> conf.location;
|
|
else if (arg == 2)
|
|
token_ss >> conf.offset;
|
|
else if (arg == 3)
|
|
token_ss >> conf.size;
|
|
else if (arg == 4)
|
|
{
|
|
std::string tok;
|
|
token_ss >> tok;
|
|
if (defaultparams_mapping.find(tok) == defaultparams_mapping.end())
|
|
log_error("unknown param default: " + tok);
|
|
conf.defaultparam = defaultparams_mapping[tok];
|
|
}
|
|
arg++;
|
|
}
|
|
|
|
if (arg == 5)
|
|
{
|
|
if (name.empty())
|
|
log_error("empty name");
|
|
else if (conf.location >= gl::MAX_PARAMS)
|
|
log_error("invalid param binding: " + std::to_string(conf.location));
|
|
else if (conf.offset > 3)
|
|
log_error("invalid offset: " + std::to_string(conf.offset));
|
|
else if (conf.offset + conf.size > 4)
|
|
log_error("invalid size: " + std::to_string(conf.size));
|
|
else
|
|
param_conf.emplace(std::make_pair(name, conf));
|
|
}
|
|
else
|
|
log_error("invalid argument count to #param");
|
|
|
|
str.erase(start_pos, fe - start_pos + 1);
|
|
}
|
|
}
|
|
|
|
void gl::shader::log_error(const std::string &str)
|
|
{
|
|
ErrorLog("bad shader: " + name + ": " + str, logtype::shader);
|
|
}
|
|
|
|
gl::shader::shader(const std::string &filename)
|
|
{
|
|
name = filename;
|
|
|
|
GLuint type;
|
|
if (strcend(filename, ".vert"))
|
|
type = GL_VERTEX_SHADER;
|
|
else if (strcend(filename, ".frag"))
|
|
type = GL_FRAGMENT_SHADER;
|
|
else if (strcend(filename, ".geom"))
|
|
type = GL_GEOMETRY_SHADER;
|
|
else
|
|
throw shader_exception("unknown shader " + filename);
|
|
|
|
std::string str;
|
|
if (!Global.gfx_usegles)
|
|
{
|
|
str += "#version 330 core\n";
|
|
}
|
|
else
|
|
{
|
|
if (GLAD_GL_ES_VERSION_3_1) {
|
|
str += "#version 310 es\n";
|
|
if (type == GL_GEOMETRY_SHADER)
|
|
str += "#extension GL_EXT_geometry_shader : require\n";
|
|
} else {
|
|
str += "#version 300 es\n";
|
|
}
|
|
str += "precision highp float;\n";
|
|
str += "precision highp sampler2DShadow;\n";
|
|
}
|
|
str += "vec4 FBOUT(vec4 x) { return " + (Global.gfx_shadergamma ? std::string("vec4(pow(x.rgb, vec3(1.0 / 2.2)), x.a)") : std::string("x")) + "; }\n";
|
|
|
|
str += read_file(filename);
|
|
process_source(str);
|
|
|
|
const GLchar *cstr = str.c_str();
|
|
|
|
if (!cstr[0])
|
|
throw shader_exception("cannot read shader: " + filename);
|
|
|
|
**this = glCreateShader(type);
|
|
glShaderSource(*this, 1, &cstr, 0);
|
|
glCompileShader(*this);
|
|
|
|
GLint status;
|
|
glGetShaderiv(*this, GL_COMPILE_STATUS, &status);
|
|
if (!status)
|
|
{
|
|
GLchar info[512];
|
|
glGetShaderInfoLog(*this, 512, 0, info);
|
|
std::cerr << std::string(info) << std::endl;
|
|
throw shader_exception("failed to compile " + filename + ": " + std::string(info));
|
|
}
|
|
}
|
|
|
|
gl::shader::~shader()
|
|
{
|
|
glDeleteShader(*this);
|
|
}
|
|
|
|
void gl::program::init()
|
|
{
|
|
bind();
|
|
|
|
for (auto it : texture_conf)
|
|
{
|
|
shader::texture_entry &e = it.second;
|
|
GLuint loc = glGetUniformLocation(*this, it.first.c_str());
|
|
glUniform1i(loc, e.id);
|
|
}
|
|
|
|
glUniform1i(glGetUniformLocation(*this, "shadowmap"), MAX_TEXTURES + 0);
|
|
glUniform1i(glGetUniformLocation(*this, "envmap"), MAX_TEXTURES + 1);
|
|
|
|
GLuint index;
|
|
|
|
if ((index = glGetUniformBlockIndex(*this, "scene_ubo")) != GL_INVALID_INDEX)
|
|
glUniformBlockBinding(*this, 0, index);
|
|
|
|
if ((index = glGetUniformBlockIndex(*this, "model_ubo")) != GL_INVALID_INDEX)
|
|
glUniformBlockBinding(*this, 1, index);
|
|
|
|
if ((index = glGetUniformBlockIndex(*this, "light_ubo")) != GL_INVALID_INDEX)
|
|
glUniformBlockBinding(*this, 2, index);
|
|
}
|
|
|
|
gl::program::program()
|
|
{
|
|
**this = glCreateProgram();
|
|
}
|
|
|
|
gl::program::program(std::vector<std::reference_wrapper<const gl::shader>> shaders) : program()
|
|
{
|
|
for (const gl::shader &s : shaders)
|
|
attach(s);
|
|
link();
|
|
}
|
|
|
|
void gl::program::attach(const gl::shader &s)
|
|
{
|
|
for (auto it : s.texture_conf)
|
|
texture_conf.emplace(std::make_pair(it.first, std::move(it.second)));
|
|
for (auto it : s.param_conf)
|
|
param_conf.emplace(std::make_pair(it.first, std::move(it.second)));
|
|
glAttachShader(*this, *s);
|
|
}
|
|
|
|
void gl::program::link()
|
|
{
|
|
glLinkProgram(*this);
|
|
|
|
GLint status;
|
|
glGetProgramiv(*this, GL_LINK_STATUS, &status);
|
|
if (!status)
|
|
{
|
|
GLchar info[512];
|
|
glGetProgramInfoLog(*this, 512, 0, info);
|
|
throw shader_exception("failed to link program: " + std::string(info));
|
|
}
|
|
|
|
init();
|
|
}
|
|
|
|
gl::program::~program()
|
|
{
|
|
glDeleteProgram(*this);
|
|
}
|
|
|
|
void gl::program::bind(GLuint i)
|
|
{
|
|
glUseProgram(i);
|
|
}
|