Open Source Tomb Raider Engine
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

Shader.cpp 14KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472
  1. /*!
  2. * \file src/system/Shader.cpp
  3. * \brief OpenGL Shader Implementation
  4. *
  5. * \author xythobuz
  6. */
  7. #include <sstream>
  8. #include "global.h"
  9. #include "Log.h"
  10. #include "system/Window.h"
  11. #include "system/Shader.h"
  12. #include <glbinding/gl/gl33.h>
  13. ShaderBuffer::~ShaderBuffer() {
  14. if (created)
  15. gl::glDeleteBuffers(1, &buffer);
  16. }
  17. void ShaderBuffer::bufferData(int elem, int size, void* data) {
  18. if (!created) {
  19. gl::glGenBuffers(1, &buffer);
  20. created = true;
  21. }
  22. boundSize = elem;
  23. gl::glBindBuffer(gl::GL_ARRAY_BUFFER, buffer);
  24. gl::glBufferData(gl::GL_ARRAY_BUFFER, elem * size, data, gl::GL_DYNAMIC_DRAW);
  25. }
  26. void ShaderBuffer::bindBuffer() {
  27. if (!created) {
  28. gl::glGenBuffers(1, &buffer);
  29. created = true;
  30. }
  31. gl::glBindBuffer(gl::GL_ELEMENT_ARRAY_BUFFER, buffer);
  32. }
  33. void ShaderBuffer::bindBuffer(int location, int size) {
  34. if (!created) {
  35. gl::glGenBuffers(1, &buffer);
  36. created = true;
  37. }
  38. gl::glEnableVertexAttribArray(location);
  39. gl::glBindBuffer(gl::GL_ARRAY_BUFFER, buffer);
  40. gl::glVertexAttribPointer(location, size, gl::GL_FLOAT, gl::GL_FALSE, 0, nullptr);
  41. }
  42. void ShaderBuffer::unbind(int location) {
  43. orAssert(created == true);
  44. gl::glDisableVertexAttribArray(location);
  45. }
  46. // ----------------------------------------------------------------------------
  47. ShaderTexture::ShaderTexture(int w, int h) : width(w), height(h) {
  48. gl::glGenFramebuffers(1, &framebuffer);
  49. bind();
  50. texture = TextureManager::loadBufferSlot(nullptr, width, height, ColorMode::RGBA,
  51. 32, TextureStorage::SYSTEM, -1, false);
  52. gl::glGenRenderbuffers(1, &depth);
  53. gl::glBindRenderbuffer(gl::GL_RENDERBUFFER, depth);
  54. gl::glRenderbufferStorage(gl::GL_RENDERBUFFER, gl::GL_DEPTH_COMPONENT, width, height);
  55. gl::glFramebufferRenderbuffer(gl::GL_FRAMEBUFFER, gl::GL_DEPTH_ATTACHMENT, gl::GL_RENDERBUFFER, depth);
  56. gl::glFramebufferTexture(gl::GL_FRAMEBUFFER, gl::GL_COLOR_ATTACHMENT0,
  57. TextureManager::getTextureID(texture, TextureStorage::SYSTEM), 0);
  58. gl::GLenum drawBuffer = gl::GL_COLOR_ATTACHMENT0;
  59. gl::glDrawBuffers(1, &drawBuffer);
  60. orAssert(gl::glCheckFramebufferStatus(gl::GL_FRAMEBUFFER) == gl::GL_FRAMEBUFFER_COMPLETE);
  61. }
  62. ShaderTexture::~ShaderTexture() {
  63. gl::glDeleteRenderbuffers(1, &depth);
  64. gl::glDeleteFramebuffers(1, &framebuffer);
  65. //! \fixme free texture slot
  66. }
  67. void ShaderTexture::clear() {
  68. bind();
  69. gl::glClear(gl::GL_COLOR_BUFFER_BIT | gl::GL_DEPTH_BUFFER_BIT);
  70. }
  71. void ShaderTexture::bind() {
  72. gl::glBindFramebuffer(gl::GL_FRAMEBUFFER, framebuffer);
  73. gl::glViewport(0, 0, width, height);
  74. }
  75. // ----------------------------------------------------------------------------
  76. Shader::~Shader() {
  77. if (programID >= 0)
  78. gl::glDeleteProgram(programID);
  79. }
  80. int Shader::addUniform(const char* name) {
  81. orAssert(programID >= 0);
  82. int r = gl::glGetUniformLocation(programID, name);
  83. if (r < 0) {
  84. Log::get(LOG_ERROR) << "Can't find GLSL Uniform \"" << name << "\"!" << Log::endl;
  85. return -1;
  86. }
  87. uniforms.push_back(r);
  88. return uniforms.size() - 1;
  89. }
  90. unsigned int Shader::getUniform(int n) {
  91. orAssert(n >= 0);
  92. orAssert(n < uniforms.size());
  93. return uniforms.at(n);
  94. }
  95. void Shader::loadUniform(int uni, glm::vec2 vec) {
  96. gl::glUniform2f(getUniform(uni), vec.x, vec.y);
  97. }
  98. void Shader::loadUniform(int uni, glm::vec4 vec) {
  99. gl::glUniform4f(getUniform(uni), vec.r, vec.g, vec.b, vec.a);
  100. }
  101. void Shader::loadUniform(int uni, glm::mat4 mat) {
  102. gl::glUniformMatrix4fv(getUniform(uni), 1, gl::GL_FALSE, &mat[0][0]);
  103. }
  104. void Shader::loadUniform(int uni, int texture, TextureStorage store) {
  105. gl::glUniform1i(getUniform(uni), TextureManager::bindTexture(texture, store));
  106. }
  107. void Shader::use() {
  108. orAssert(programID >= 0);
  109. static int lastID = -1;
  110. if (programID != lastID) {
  111. gl::glUseProgram(programID);
  112. lastID = programID;
  113. }
  114. }
  115. int Shader::compile(const char* vertex, const char* fragment) {
  116. orAssert(vertex != nullptr);
  117. orAssert(fragment != nullptr);
  118. gl::GLuint vertexID = gl::glCreateShader(gl::GL_VERTEX_SHADER);
  119. gl::GLuint fragmentID = gl::glCreateShader(gl::GL_FRAGMENT_SHADER);
  120. gl::GLint result = gl::GLint(gl::GL_FALSE);
  121. gl::GLint logLength = 0;
  122. // Compile vertex shader
  123. gl::glShaderSource(vertexID, 1, &vertex, nullptr);
  124. gl::glCompileShader(vertexID);
  125. // Check vertex shader
  126. gl::glGetShaderiv(vertexID, gl::GL_COMPILE_STATUS, &result);
  127. gl::glGetShaderiv(vertexID, gl::GL_INFO_LOG_LENGTH, &logLength);
  128. if ((logLength > 0) && (result != gl::GLint(gl::GL_TRUE))) {
  129. std::vector<char> message(logLength + 1);
  130. gl::glGetShaderInfoLog(vertexID, logLength, nullptr, &message[0]);
  131. Log::get(LOG_ERROR) << "Vertex Shader compilation error:" << Log::endl;
  132. Log::get(LOG_ERROR) << &message[0] << Log::endl;
  133. gl::glDeleteShader(vertexID);
  134. gl::glDeleteShader(fragmentID);
  135. return -1;
  136. }
  137. // Compile fragment shader
  138. gl::glShaderSource(fragmentID, 1, &fragment, nullptr);
  139. gl::glCompileShader(fragmentID);
  140. // Check fragment shader
  141. gl::glGetShaderiv(fragmentID, gl::GL_COMPILE_STATUS, &result);
  142. gl::glGetShaderiv(fragmentID, gl::GL_INFO_LOG_LENGTH, &logLength);
  143. if ((logLength > 0) && (result != gl::GLint(gl::GL_TRUE))) {
  144. std::vector<char> message(logLength + 1);
  145. gl::glGetShaderInfoLog(fragmentID, logLength, nullptr, &message[0]);
  146. Log::get(LOG_ERROR) << "Fragment Shader compilation error:" << Log::endl;
  147. Log::get(LOG_ERROR) << &message[0] << Log::endl;
  148. gl::glDeleteShader(vertexID);
  149. gl::glDeleteShader(fragmentID);
  150. return -2;
  151. }
  152. // Link both shaders
  153. programID = gl::glCreateProgram();
  154. gl::glAttachShader(programID, vertexID);
  155. gl::glAttachShader(programID, fragmentID);
  156. gl::glLinkProgram(programID);
  157. // Check resulting program
  158. gl::glGetProgramiv(programID, gl::GL_LINK_STATUS, &result);
  159. gl::glGetProgramiv(programID, gl::GL_INFO_LOG_LENGTH, &logLength);
  160. if ((logLength > 0) && (result != gl::GLint(gl::GL_TRUE))) {
  161. std::vector<char> message(logLength + 1);
  162. gl::glGetProgramInfoLog(programID, logLength, nullptr, &message[0]);
  163. Log::get(LOG_ERROR) << "Shader link error:" << Log::endl;
  164. Log::get(LOG_ERROR) << &message[0] << Log::endl;
  165. gl::glDeleteShader(vertexID);
  166. gl::glDeleteShader(fragmentID);
  167. gl::glDeleteProgram(programID);
  168. return -3;
  169. }
  170. gl::glDeleteShader(vertexID);
  171. gl::glDeleteShader(fragmentID);
  172. return programID;
  173. }
  174. // ----------------------------------------------------------------------------
  175. std::string Shader::getVersion(bool linked) {
  176. static std::string cache;
  177. static std::string cacheLinked;
  178. static bool cacheFilled = false;
  179. if (!cacheFilled) {
  180. std::ostringstream str;
  181. str << "OpenGL v" << gl::glGetString(gl::GL_VERSION);
  182. cache = str.str();
  183. str.str("");
  184. str << "OpenGL " << gl::glGetString(gl::GL_SHADING_LANGUAGE_VERSION) << " "
  185. << gl::glGetString(gl::GL_RENDERER) << " (" << gl::glGetString(gl::GL_VENDOR) << ")";
  186. cacheLinked = str.str();
  187. cacheFilled = true;
  188. }
  189. if (linked) {
  190. return cacheLinked;
  191. } else {
  192. return cache;
  193. }
  194. }
  195. Shader Shader::textureShader;
  196. Shader Shader::colorShader;
  197. unsigned int Shader::vertexArrayID = 0;
  198. bool Shader::lastBufferWasNotFramebuffer = true;
  199. int Shader::initialize() {
  200. gl::glGenVertexArrays(1, &vertexArrayID);
  201. gl::glBindVertexArray(vertexArrayID);
  202. // Set background color
  203. gl::glClearColor(0.0f, 0.0f, 0.4f, 1.0f);
  204. set2DState(false);
  205. gl::glDepthFunc(gl::GL_LESS);
  206. gl::glEnable(gl::GL_BLEND);
  207. gl::glBlendFunc(gl::GL_SRC_ALPHA, gl::GL_ONE_MINUS_SRC_ALPHA);
  208. gl::glPointSize(5.0f);
  209. if (textureShader.compile(textureShaderVertex, textureShaderFragment) < 0)
  210. return -1;
  211. if (textureShader.addUniform("MVP") < 0)
  212. return -2;
  213. if (textureShader.addUniform("textureSampler") < 0)
  214. return -3;
  215. if (colorShader.compile(colorShaderVertex, colorShaderFragment) < 0)
  216. return -4;
  217. if (colorShader.addUniform("MVP") < 0)
  218. return -5;
  219. return 0;
  220. }
  221. void Shader::shutdown() {
  222. gl::glDeleteVertexArrays(1, &vertexArrayID);
  223. }
  224. void Shader::set2DState(bool on, bool depth) {
  225. if (on) {
  226. gl::glDisable(gl::GL_CULL_FACE);
  227. if (depth)
  228. gl::glDisable(gl::GL_DEPTH_TEST);
  229. } else {
  230. gl::glEnable(gl::GL_CULL_FACE);
  231. if (depth)
  232. gl::glEnable(gl::GL_DEPTH_TEST);
  233. }
  234. }
  235. void Shader::bindProperBuffer(ShaderTexture* target) {
  236. if ((target == nullptr) && lastBufferWasNotFramebuffer) {
  237. lastBufferWasNotFramebuffer = false;
  238. gl::glBindFramebuffer(gl::GL_FRAMEBUFFER, 0);
  239. gl::glViewport(0, 0, Window::getSize().x, Window::getSize().y);
  240. } else if (target != nullptr) {
  241. lastBufferWasNotFramebuffer = true;
  242. target->bind();
  243. }
  244. }
  245. void Shader::drawGL(std::vector<glm::vec3>& vertices, std::vector<glm::vec2>& uvs,
  246. glm::mat4 MVP, unsigned int texture, TextureStorage store,
  247. gl::GLenum mode, ShaderTexture* target, Shader& shader) {
  248. bindProperBuffer(target);
  249. shader.use();
  250. shader.loadUniform(0, MVP);
  251. shader.loadUniform(1, texture, store);
  252. shader.vertexBuffer.bufferData(vertices);
  253. shader.otherBuffer.bufferData(uvs);
  254. shader.vertexBuffer.bindBuffer(0, 3);
  255. shader.otherBuffer.bindBuffer(1, 2);
  256. gl::glDrawArrays(mode, 0, shader.vertexBuffer.getSize());
  257. shader.vertexBuffer.unbind(0);
  258. shader.otherBuffer.unbind(1);
  259. }
  260. void Shader::drawGL(std::vector<glm::vec3>& vertices, std::vector<glm::vec2>& uvs,
  261. std::vector<unsigned short>& indices, glm::mat4 MVP,
  262. unsigned int texture, TextureStorage store,
  263. gl::GLenum mode, ShaderTexture* target, Shader& shader) {
  264. drawGLBuffer(vertices, uvs);
  265. drawGLOnly(indices, MVP, texture, store, mode, target, shader);
  266. }
  267. void Shader::drawGLBuffer(std::vector<glm::vec3>& vertices, std::vector<glm::vec2>& uvs, Shader& shader) {
  268. shader.vertexBuffer.bufferData(vertices);
  269. shader.otherBuffer.bufferData(uvs);
  270. }
  271. void Shader::drawGLOnly(std::vector<unsigned short>& indices, glm::mat4 MVP,
  272. unsigned int texture, TextureStorage store,
  273. gl::GLenum mode, ShaderTexture* target, Shader& shader) {
  274. bindProperBuffer(target);
  275. shader.use();
  276. shader.loadUniform(0, MVP);
  277. shader.loadUniform(1, texture, store);
  278. shader.indexBuffer.bufferData(indices);
  279. shader.vertexBuffer.bindBuffer(0, 3);
  280. shader.otherBuffer.bindBuffer(1, 2);
  281. shader.indexBuffer.bindBuffer();
  282. gl::glDrawElements(mode, shader.indexBuffer.getSize(), gl::GL_UNSIGNED_SHORT, nullptr);
  283. shader.vertexBuffer.unbind(0);
  284. shader.otherBuffer.unbind(1);
  285. }
  286. void Shader::drawGL(std::vector<glm::vec3>& vertices, std::vector<glm::vec3>& colors,
  287. glm::mat4 MVP, gl::GLenum mode, ShaderTexture* target, Shader& shader) {
  288. bindProperBuffer(target);
  289. shader.use();
  290. shader.loadUniform(0, MVP);
  291. shader.vertexBuffer.bufferData(vertices);
  292. shader.otherBuffer.bufferData(colors);
  293. shader.vertexBuffer.bindBuffer(0, 3);
  294. shader.otherBuffer.bindBuffer(1, 3);
  295. gl::glDrawArrays(mode, 0, shader.vertexBuffer.getSize());
  296. shader.vertexBuffer.unbind(0);
  297. shader.otherBuffer.unbind(1);
  298. }
  299. void Shader::drawGL(std::vector<glm::vec3>& vertices, std::vector<glm::vec3>& colors,
  300. std::vector<unsigned short>& indices, glm::mat4 MVP,
  301. gl::GLenum mode, ShaderTexture* target, Shader& shader) {
  302. bindProperBuffer(target);
  303. shader.use();
  304. shader.loadUniform(0, MVP);
  305. shader.vertexBuffer.bufferData(vertices);
  306. shader.otherBuffer.bufferData(colors);
  307. shader.indexBuffer.bufferData(indices);
  308. shader.vertexBuffer.bindBuffer(0, 3);
  309. shader.otherBuffer.bindBuffer(1, 3);
  310. shader.indexBuffer.bindBuffer();
  311. gl::glDrawElements(mode, shader.indexBuffer.getSize(), gl::GL_UNSIGNED_SHORT, nullptr);
  312. shader.vertexBuffer.unbind(0);
  313. shader.otherBuffer.unbind(1);
  314. }
  315. // --------------------------------------
  316. // *INDENT-OFF*
  317. const char* Shader::textureShaderVertex = R"!?!(
  318. #version 330 core
  319. layout(location = 0) in vec3 vertexPosition_modelspace;
  320. layout(location = 1) in vec2 vertexUV;
  321. out vec2 UV;
  322. uniform mat4 MVP;
  323. void main() {
  324. vec4 pos = MVP * vec4(vertexPosition_modelspace.x,
  325. vertexPosition_modelspace.y,
  326. vertexPosition_modelspace.z,
  327. 1);
  328. gl_Position = pos;
  329. UV = vertexUV;
  330. }
  331. )!?!";
  332. const char* Shader::textureShaderFragment = R"!?!(
  333. #version 330 core
  334. in vec2 UV;
  335. layout(location = 0) out vec4 color;
  336. uniform sampler2D textureSampler;
  337. void main() {
  338. color = texture(textureSampler, UV);
  339. }
  340. )!?!";
  341. // --------------------------------------
  342. const char* Shader::colorShaderVertex = R"!?!(
  343. #version 330 core
  344. layout(location = 0) in vec3 vertexPosition_modelspace;
  345. layout(location = 1) in vec3 vertexColor;
  346. out vec3 color;
  347. uniform mat4 MVP;
  348. void main() {
  349. vec4 pos = MVP * vec4(vertexPosition_modelspace.x,
  350. vertexPosition_modelspace.y,
  351. vertexPosition_modelspace.z,
  352. 1);
  353. gl_Position = pos;
  354. color = vertexColor;
  355. }
  356. )!?!";
  357. const char* Shader::colorShaderFragment = R"!?!(
  358. #version 330 core
  359. in vec3 color;
  360. layout(location = 0) out vec4 color_out;
  361. void main() {
  362. color_out = vec4(color, 1);
  363. }
  364. )!?!";
  365. // --------------------------------------
  366. // *INDENT-ON*