TAMSVIZ
Visualization and annotation tool for ROS
shader.cpp
1 // TAMSVIZ
2 // (c) 2020 Philipp Ruppel
3 
4 #include "shader.h"
5 
6 #include "../core/destructor.h"
7 #include "../core/log.h"
8 #include "opengl.h"
9 
10 #include <vector>
11 
12 static bool logEmpty(const std::string &s) {
13  for (auto c : s) {
14  if (c && !std::isspace(c)) {
15  return false;
16  }
17  }
18  return true;
19 }
20 
21 ShaderManager &Shader::manager() {
22  static ShaderManager instance;
23  return instance;
24 }
25 
26 void Shader::addShader(GLenum type, const std::string &url) {
27  std::string source;
28  loadResource(url, source);
29  if (source.empty()) {
30  throw std::runtime_error("shader file " + url + " is empty");
31  }
32  GLuint shader = 0;
33  V_GL(shader = glCreateShader(type));
34  Destructor dtor([&]() {
35  if (shader) {
36  V_GL(glDeleteShader(shader));
37  shader = 0;
38  }
39  });
40  auto *src = source.c_str();
41  V_GL(glShaderSource(shader, 1, &src, NULL));
42  V_GL(glCompileShader(shader));
43  {
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());
47  int success = 0;
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);
52  } else {
53  if (!logEmpty(error_string)) {
54  LOG_WARN("shader log " << url << "\n" << error_string);
55  }
56  }
57  }
58  V_GL(glAttachShader(_program, shader));
59 }
60 
61 void Shader::create() {
62  if (!_program) {
63  destroy();
64 
65  V_GL(_program = glCreateProgram());
66  addShader(GL_VERTEX_SHADER, _vertex_shader_url);
67  addShader(GL_FRAGMENT_SHADER, _fragment_shader_url);
68 
69  V_GL(glBindAttribLocation(_program, (GLuint)VertexAttributes::position,
70  "position"));
71  V_GL(glBindAttribLocation(_program, (GLuint)VertexAttributes::normal,
72  "normal"));
73  V_GL(glBindAttribLocation(_program, (GLuint)VertexAttributes::texcoord,
74  "texcoord"));
75  V_GL(glBindAttribLocation(_program, (GLuint)VertexAttributes::tangent,
76  "tangent"));
77  V_GL(glBindAttribLocation(_program, (GLuint)VertexAttributes::bitangent,
78  "bitangent"));
79  V_GL(glBindAttribLocation(_program, (GLuint)VertexAttributes::color,
80  "color"));
81  V_GL(glBindAttribLocation(_program, (GLuint)VertexAttributes::extra,
82  "extra"));
83 
84  V_GL(glBindAttribLocation(_program, (GLuint)VertexAttributes::pose_x,
85  "pose_x"));
86  V_GL(glBindAttribLocation(_program, (GLuint)VertexAttributes::pose_y,
87  "pose_y"));
88  V_GL(glBindAttribLocation(_program, (GLuint)VertexAttributes::pose_z,
89  "pose_z"));
90 
91  V_GL(glBindFragDataLocation(_program, 0, "out_color"));
92  V_GL(glBindFragDataLocation(_program, 1, "out_blend"));
93  V_GL(glBindFragDataLocation(_program, 2, "out_id"));
94 
95  V_GL(glLinkProgram(_program));
96 
97  {
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));
103  }
104  }
105  {
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));
111  }
112  }
113  {
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));
119  }
120  }
121 
122  {
123  std::vector<char> error_log(1024 * 32);
124  V_GL(glGetProgramInfoLog(_program, error_log.size(), NULL,
125  error_log.data()));
126  std::string error_string(error_log.data(), error_log.size());
127  int success = 0;
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
132  << "\n"
133  << error_string);
134  throw std::runtime_error(error_string);
135  } else {
136  if (!logEmpty(error_string)) {
137  LOG_WARN("shader program log "
138  << " " << _vertex_shader_url << " " << _fragment_shader_url
139  << "\n"
140  << error_string);
141  }
142  }
143  }
144 
145  V_GL(glUseProgram(_program));
146  {
147  GLint index = -1;
148  V_GL(index = glGetUniformLocation(_program, "color_sampler"));
149  if (index != -1) {
150  V_GL(glUniform1i(index, (int)Samplers::color));
151  }
152  }
153  {
154  GLint index = -1;
155  V_GL(index = glGetUniformLocation(_program, "normal_sampler"));
156  if (index != -1) {
157  V_GL(glUniform1i(index, (int)Samplers::normal));
158  }
159  }
160  V_GL(glUseProgram(0));
161  }
162 }
163 
164 void Shader::destroy() {
165  if (_program) {
166  int program = _program;
167  cleanup([program]() { V_GL(glDeleteProgram(program)); });
168  _program = 0;
169  }
170 }
171 
172 int Shader::program() {
173  if (invalidated()) {
174  GLuint previous_program = _program;
175  _program = 0;
176  try {
177  create();
178  } catch (const std::exception &e) {
179  LOG_ERROR("failed to recompile shader " << _vertex_shader_url << " "
180  << _fragment_shader_url);
181  LOG_ERROR(e.what());
182  if (_program) {
183  V_GL(glDeleteProgram(_program));
184  _program = 0;
185  }
186  }
187  if (_program) {
188  V_GL(glDeleteProgram(previous_program));
189  } else {
190  _program = previous_program;
191  }
192  }
193  if (!_program) {
194  create();
195  }
196  return _program;
197 }
198 
199 void Shader::use() { V_GL(glUseProgram(program())); }