6 #include "../core/destructor.h" 7 #include "../core/log.h" 12 static bool logEmpty(
const std::string &s) {
14 if (c && !std::isspace(c)) {
26 void Shader::addShader(GLenum type,
const std::string &url) {
28 loadResource(url, source);
30 throw std::runtime_error(
"shader file " + url +
" is empty");
33 V_GL(shader = glCreateShader(type));
36 V_GL(glDeleteShader(shader));
40 auto *src = source.c_str();
41 V_GL(glShaderSource(shader, 1, &src, NULL));
42 V_GL(glCompileShader(shader));
44 std::vector<char> error_log(1024 * 32, 0);
45 V_GL(glGetShaderInfoLog(shader, error_log.size(), NULL, error_log.data()));
46 std::string error_string(error_log.data(), error_log.size());
48 V_GL(glGetShaderiv(shader, GL_COMPILE_STATUS, &success));
49 if (success != GL_TRUE) {
50 LOG_ERROR(
"failed to compile shader " << url <<
"\n" << error_string);
51 throw std::runtime_error(error_string);
53 if (!logEmpty(error_string)) {
54 LOG_WARN(
"shader log " << url <<
"\n" << error_string);
58 V_GL(glAttachShader(_program, shader));
61 void Shader::create() {
65 V_GL(_program = glCreateProgram());
66 addShader(GL_VERTEX_SHADER, _vertex_shader_url);
67 addShader(GL_FRAGMENT_SHADER, _fragment_shader_url);
69 V_GL(glBindAttribLocation(_program, (GLuint)VertexAttributes::position,
71 V_GL(glBindAttribLocation(_program, (GLuint)VertexAttributes::normal,
73 V_GL(glBindAttribLocation(_program, (GLuint)VertexAttributes::texcoord,
75 V_GL(glBindAttribLocation(_program, (GLuint)VertexAttributes::tangent,
77 V_GL(glBindAttribLocation(_program, (GLuint)VertexAttributes::bitangent,
79 V_GL(glBindAttribLocation(_program, (GLuint)VertexAttributes::color,
81 V_GL(glBindAttribLocation(_program, (GLuint)VertexAttributes::extra,
84 V_GL(glBindAttribLocation(_program, (GLuint)VertexAttributes::pose_x,
86 V_GL(glBindAttribLocation(_program, (GLuint)VertexAttributes::pose_y,
88 V_GL(glBindAttribLocation(_program, (GLuint)VertexAttributes::pose_z,
91 V_GL(glBindFragDataLocation(_program, 0,
"out_color"));
92 V_GL(glBindFragDataLocation(_program, 1,
"out_blend"));
93 V_GL(glBindFragDataLocation(_program, 2,
"out_id"));
95 V_GL(glLinkProgram(_program));
98 GLuint index = GL_INVALID_INDEX;
99 V_GL(index = glGetUniformBlockIndex(_program,
"camera_block"));
100 if (index != GL_INVALID_INDEX) {
101 V_GL(glUniformBlockBinding(_program, index,
102 (uint32_t)UniformBindingPoint::camera));
106 GLuint index = GL_INVALID_INDEX;
107 V_GL(index = glGetUniformBlockIndex(_program,
"material_block"));
108 if (index != GL_INVALID_INDEX) {
109 V_GL(glUniformBlockBinding(_program, index,
110 (uint32_t)UniformBindingPoint::material));
114 GLuint index = GL_INVALID_INDEX;
115 V_GL(index = glGetUniformBlockIndex(_program,
"light_block"));
116 if (index != GL_INVALID_INDEX) {
117 V_GL(glUniformBlockBinding(_program, index,
118 (uint32_t)UniformBindingPoint::lights));
123 std::vector<char> error_log(1024 * 32);
124 V_GL(glGetProgramInfoLog(_program, error_log.size(), NULL,
126 std::string error_string(error_log.data(), error_log.size());
128 V_GL(glGetProgramiv(_program, GL_LINK_STATUS, &success));
129 if (success != GL_TRUE) {
130 LOG_ERROR(
"failed to link shader " 131 <<
" " << _vertex_shader_url <<
" " << _fragment_shader_url
134 throw std::runtime_error(error_string);
136 if (!logEmpty(error_string)) {
137 LOG_WARN(
"shader program log " 138 <<
" " << _vertex_shader_url <<
" " << _fragment_shader_url
145 V_GL(glUseProgram(_program));
148 V_GL(index = glGetUniformLocation(_program,
"color_sampler"));
150 V_GL(glUniform1i(index, (
int)Samplers::color));
155 V_GL(index = glGetUniformLocation(_program,
"normal_sampler"));
157 V_GL(glUniform1i(index, (
int)Samplers::normal));
160 V_GL(glUseProgram(0));
164 void Shader::destroy() {
166 int program = _program;
167 cleanup([program]() { V_GL(glDeleteProgram(program)); });
172 int Shader::program() {
174 GLuint previous_program = _program;
178 }
catch (
const std::exception &e) {
179 LOG_ERROR(
"failed to recompile shader " << _vertex_shader_url <<
" " 180 << _fragment_shader_url);
183 V_GL(glDeleteProgram(_program));
188 V_GL(glDeleteProgram(previous_program));
190 _program = previous_program;
199 void Shader::use() { V_GL(glUseProgram(program())); }