TDME2 1.9.121
GL3Renderer.cpp
Go to the documentation of this file.
2
3#if defined (__APPLE__)
4 #include <OpenGL/gl3.h>
5 #include <OpenGL/OpenGL.h>
6 #include <OpenCL/opencl.h>
7 #include <OpenCL/cl_gl.h>
8#elif defined(__FreeBSD__) || defined(__NetBSD__) || defined(__OpenBSD__) || defined(__linux__) || defined(_WIN32) || defined(__HAIKU__)
9 #define GLEW_NO_GLU
10 #include <GL/glew.h>
11#endif
12
13#include <string.h>
14
15#include <array>
16#include <map>
17#include <string>
18#include <vector>
19
20#include <tdme/tdme.h>
23#include <tdme/engine/Engine.h>
25#include <tdme/math/Math.h>
26#include <tdme/math/Matrix4x4.h>
37#include <tdme/utilities/Time.h>
38
39using std::array;
40using std::map;
41using std::string;
42using std::to_string;
43using std::vector;
44
46
63
64GL3Renderer::GL3Renderer()
65{
67 // setup GL3 consts
68 ID_NONE = 0;
69 CLEAR_DEPTH_BUFFER_BIT = GL_DEPTH_BUFFER_BIT;
70 CLEAR_COLOR_BUFFER_BIT = GL_COLOR_BUFFER_BIT;
71 CULLFACE_FRONT = GL_FRONT;
72 CULLFACE_BACK = GL_BACK;
73 FRONTFACE_CW = GL_CW;
74 FRONTFACE_CCW = GL_CCW;
75 SHADER_FRAGMENT_SHADER = GL_FRAGMENT_SHADER;
76 SHADER_VERTEX_SHADER = GL_VERTEX_SHADER;
77 #if defined (__APPLE__)
82 #else
83 SHADER_COMPUTE_SHADER = GL_COMPUTE_SHADER;
84 #endif
85 DEPTHFUNCTION_ALWAYS = GL_ALWAYS;
86 DEPTHFUNCTION_EQUAL = GL_EQUAL;
87 DEPTHFUNCTION_LESSEQUAL = GL_LEQUAL;
89 CUBEMAPTEXTUREINDEX_NEGATIVE_X = GL_TEXTURE_CUBE_MAP_NEGATIVE_X;
90 CUBEMAPTEXTUREINDEX_POSITIVE_X = GL_TEXTURE_CUBE_MAP_POSITIVE_X;
91 CUBEMAPTEXTUREINDEX_POSITIVE_Y = GL_TEXTURE_CUBE_MAP_POSITIVE_Y;
92 CUBEMAPTEXTUREINDEX_NEGATIVE_Y = GL_TEXTURE_CUBE_MAP_NEGATIVE_Y;
93 CUBEMAPTEXTUREINDEX_POSITIVE_Z = GL_TEXTURE_CUBE_MAP_POSITIVE_Z;
94 CUBEMAPTEXTUREINDEX_NEGATIVE_Z = GL_TEXTURE_CUBE_MAP_NEGATIVE_Z;
98}
99
101 auto vendor = (char*)glGetString(GL_VENDOR);
102 return string(vendor);
103}
104
106 auto renderer = (char*)glGetString(GL_RENDERER);
107 return string(renderer) + " [GL3+/CORE]";
108}
109
111{
112 return "gl3";
113}
114
116 return false;
117}
118
119#if defined (__APPLE__)
120 void GL3Renderer::clErrorCallback(const char* errorInfo, const void* privateInfo, size_t cb, void* userData) {
121 Console::println(string("GL3Renderer::clErrorCallback(): ") + errorInfo);
122 }
123#endif
124
126{
127 glGetError();
128 // get default framebuffer
129 FRAMEBUFFER_DEFAULT = 0; //getContext()->getDefaultDrawFramebuffer();
130 // setup open gl
131 glClearColor(0.0f, 0.0f, 0.0f, 1.0f);
132 glClearDepth(1.0f);
133 glEnable(GL_DEPTH_TEST);
134 glEnable(GL_CULL_FACE);
135 glDepthFunc(GL_LEQUAL);
136 glBlendEquation(GL_FUNC_ADD);
137 glDisable(GL_BLEND);
138 #if !defined (__APPLE__)
139 glEnable(GL_POINT_SPRITE);
140 #endif
141 glEnable(GL_PROGRAM_POINT_SIZE);
143 // port-macosx requires this
144 glGenVertexArrays(1, &engineVAO);
145 glBindVertexArray(engineVAO);
146 //
147 #if defined (__APPLE__)
148 // TODO: error management
149 // shader source
150 auto skinningKernelProgramSource = FileSystem::getInstance()->getContentAsString("shader/gl3/skinning", "skinning.cl");
151 auto skinningKernelProgramSourceSize = skinningKernelProgramSource.size();
152 char* skinningKernelProgramSourceHeap = new char[skinningKernelProgramSourceSize + 1];
153 strcpy(skinningKernelProgramSourceHeap, skinningKernelProgramSource.c_str());
154
155 // context, device
156 cl_int clError = 0;
157 cl_device_id clDeviceId = 0;
158 auto clCurrentContext = CGLGetCurrentContext();
159 auto clShareGroup = CGLGetShareGroup(clCurrentContext);
160 gcl_gl_set_sharegroup(clShareGroup);
161 auto clDispatchQueue = gcl_create_dispatch_queue(CL_DEVICE_TYPE_GPU, nullptr);
162 clDeviceId = gcl_get_device_id_with_dispatch_queue(clDispatchQueue);
163
164 // device vendor + names
165 size_t clSize = 0;
166 cl_char clDeviceVendorName[1024] = {0};
167 cl_char clDeviceDeviceName[1024] = {0};
168 clError = clGetDeviceInfo(clDeviceId, CL_DEVICE_VENDOR, sizeof(clDeviceVendorName) - 1, clDeviceVendorName, &clSize);
169 clError|= clGetDeviceInfo(clDeviceId, CL_DEVICE_NAME, sizeof(clDeviceDeviceName) - 1, clDeviceDeviceName, &clSize);
170 Console::println(string("GL3Renderer::initialize(): Using OpenCL CL device: ") + (char*)clDeviceVendorName + ": " + (char*)clDeviceDeviceName);
171
172 // CL context, skinning kernel
173 cl_context_properties properties[] = {
174 CL_CONTEXT_PROPERTY_USE_CGL_SHAREGROUP_APPLE,
175 (cl_context_properties)clShareGroup,
176 0
177 };
178 clContext = clCreateContext(properties, 1, &clDeviceId, clErrorCallback, nullptr, &clError);
179 clCommandQueue = clCreateCommandQueue(clContext, clDeviceId, 0, &clError);
180 clSkinningKernelProgram = clCreateProgramWithSource(clContext, 1, (const char**)&skinningKernelProgramSourceHeap, &skinningKernelProgramSourceSize, &clError);
181 clError = clBuildProgram(clSkinningKernelProgram, 1, &clDeviceId, nullptr, nullptr, nullptr);
182 auto clBuildInfo = clGetProgramBuildInfo(clSkinningKernelProgram, clDeviceId, CL_PROGRAM_BUILD_STATUS, 0, nullptr, &clSize);
183 clSkinningKernel = clCreateKernel(clSkinningKernelProgram, "computeSkinning", &clError);
184 #endif
185 // check if deferred shading is available
186 int glMaxColorAttachments = 0;
187 glGetIntegerv(GL_MAX_COLOR_ATTACHMENTS, &glMaxColorAttachments);
188 int glMaxDrawBuffers = 0;
189 glGetIntegerv(GL_MAX_DRAW_BUFFERS, &glMaxDrawBuffers);
190 deferredShadingAvailable = glMaxColorAttachments >= 8 && glMaxDrawBuffers >= 8;
191 // renderer contexts
192 rendererContexts.resize(1);
193 for (auto& rendererContext: rendererContexts) {
194 for (auto i = 0; i < rendererContext.lights.size(); i++) {
195 rendererContext.lights[i].spotCosCutoff = static_cast<float>(Math::cos(Math::PI / 180.0f * 180.0f));
196 }
197 rendererContext.textureMatrix.identity();
198 }
199}
200
202{
203}
204
206{
207}
208
210{
211 return true;
212}
213
215{
216 return true;
217}
218
220{
221 return false;
222}
223
225 return true;
226}
227
229{
230 return true;
231}
232
234{
235 return true;
236}
237
239 return true;
240}
241
243{
244 return true;
245}
246
248 #if defined(__HAIKU__)
249 // does not work for now with Haiku OS
250 return false;
251 #else
252 int glMajorVersion;
253 int glMinorVersion;
254 glGetIntegerv(GL_MAJOR_VERSION, &glMajorVersion);
255 glGetIntegerv(GL_MINOR_VERSION, &glMinorVersion);
256 return (glMajorVersion == 4 && glMinorVersion >= 3) || glMajorVersion > 4;
257 #endif
258}
259
261 #if defined (__APPLE__)
262 return true;
263 #else
264 return false;
265 #endif
266}
267
269 return false;
270}
271
274}
275
277{
278 return activeTextureUnit;
279}
280
281int32_t GL3Renderer::loadShader(int32_t type, const string& pathName, const string& fileName, const string& definitions, const string& functions)
282{
283 // create shader
284 int32_t handle = glCreateShader(type);
285 // exit if no handle returned
286 if (handle == 0) return 0;
287
288 // shader source
289 auto shaderSource = StringTools::replace(
290 StringTools::replace(
291 FileSystem::getInstance()->getContentAsString(pathName, fileName),
292 "{$DEFINITIONS}",
293 definitions + "\n\n"
294 ),
295 "{$FUNCTIONS}",
296 functions + "\n\n"
297 );
298 string sourceString = (shaderSource);
299 char* sourceHeap = new char[sourceString.length() + 1];
300 strcpy(sourceHeap, sourceString.c_str());
301 // load source
302 glShaderSource(handle, 1, &sourceHeap, nullptr);
303 // compile
304 glCompileShader(handle);
305 //
306 delete [] sourceHeap;
307 // check state
308 int32_t compileStatus;
309 glGetShaderiv(handle, GL_COMPILE_STATUS, &compileStatus);
310 if (compileStatus == 0) {
311 // get error
312 int32_t infoLogLengthBuffer;
313 glGetShaderiv(handle, GL_INFO_LOG_LENGTH, &infoLogLengthBuffer);
314 char* infoLogBuffer = new char[infoLogLengthBuffer];
315 glGetShaderInfoLog(handle, infoLogLengthBuffer, &infoLogLengthBuffer, infoLogBuffer);
316 auto infoLogString = (string(infoLogBuffer, infoLogLengthBuffer));
317 // be verbose
318 Console::println(
319 string(
320 string("GL3Renderer::loadShader") +
321 string("[") +
322 to_string(handle) +
323 string("]") +
324 pathName +
325 string("/") +
326 fileName +
327 string(": failed: ") +
328 infoLogString
329 )
330 );
331 Console::println(shaderSource);
332 //
333 delete [] infoLogBuffer;
334 // remove shader
335 glDeleteShader(handle);
336 return 0;
337 }
338
339 return handle;
340}
341
342void GL3Renderer::useProgram(int contextIdx, int32_t programId)
343{
344 glUseProgram(programId);
345}
346
348{
349 auto glProgram = glCreateProgram();
350 return glProgram;
351}
352
353void GL3Renderer::attachShaderToProgram(int32_t programId, int32_t shaderId)
354{
355 glAttachShader(programId, shaderId);
356}
357
358bool GL3Renderer::linkProgram(int32_t programId)
359{
360 glLinkProgram(programId);
361 // check state
362 int32_t linkStatus;
363 glGetProgramiv(programId, GL_LINK_STATUS, &linkStatus);
364 if (linkStatus == 0) {
365 // get error
366 int32_t infoLogLength = 0;
367 glGetProgramiv(programId, GL_INFO_LOG_LENGTH, &infoLogLength);
368 char* infoLog = new char[infoLogLength];
369 glGetProgramInfoLog(programId, infoLogLength, &infoLogLength, infoLog);
370 auto infoLogString = (string(infoLog, infoLogLength));
371 // be verbose
372 Console::println(
373 string(
374 "[" +
375 to_string(programId) +
376 "]: failed: " +
377 infoLogString
378 )
379 );
380 //
381 delete [] infoLog;
382 //
383 return false;
384 }
385 return true;
386}
387
388int32_t GL3Renderer::getProgramUniformLocation(int32_t programId, const string& name)
389{
390 auto uniformLocation = glGetUniformLocation(programId, (name).c_str());
391 return uniformLocation;
392}
393
394void GL3Renderer::setProgramUniformInteger(int contextIdx, int32_t uniformId, int32_t value)
395{
396 #if defined (__APPLE__)
397 if (uniformId == UNIFORM_CL_SKINNING_VERTEX_COUNT) {
398 clSkinningParameters.vertexCount = value;
399 } else
400 if (uniformId == UNIFORM_CL_SKINNING_MATRIX_COUNT) {
401 clSkinningParameters.matrixCount = value;
402 } else
403 if (uniformId == UNIFORM_CL_SKINNING_INSTANCE_COUNT) {
404 clSkinningParameters.instanceCount = value;
405 } else {
406 glUniform1i(uniformId, value);
407 }
408 #else
409 glUniform1i(uniformId, value);
410 #endif
411}
412
413void GL3Renderer::setProgramUniformFloat(int contextIdx, int32_t uniformId, float value)
414{
415 glUniform1f(uniformId, value);
416}
417
418void GL3Renderer::setProgramUniformFloatMatrix3x3(int contextIdx, int32_t uniformId, const array<float, 9>& data)
419{
420 glUniformMatrix3fv(uniformId, 1, false, data.data());
421}
422
423void GL3Renderer::setProgramUniformFloatMatrix4x4(int contextIdx, int32_t uniformId, const array<float, 16>& data)
424{
425 glUniformMatrix4fv(uniformId, 1, false, data.data());
426}
427
428void GL3Renderer::setProgramUniformFloatMatrices4x4(int contextIdx, int32_t uniformId, int32_t count, FloatBuffer* data)
429{
430 glUniformMatrix4fv(uniformId, count, false, (float*)data->getBuffer());
431}
432
433void GL3Renderer::setProgramUniformFloatVec4(int contextIdx, int32_t uniformId, const array<float, 4>& data)
434{
435 glUniform4fv(uniformId, 1, data.data());
436}
437
438void GL3Renderer::setProgramUniformFloatVec3(int contextIdx, int32_t uniformId, const array<float, 3>& data)
439{
440 glUniform3fv(uniformId, 1, data.data());
441}
442
443void GL3Renderer::setProgramUniformFloatVec2(int contextIdx, int32_t uniformId, const array<float, 2>& data)
444{
445 glUniform2fv(uniformId, 1, data.data());
446}
447
448void GL3Renderer::setProgramAttributeLocation(int32_t programId, int32_t location, const string& name)
449{
450 glBindAttribLocation(programId, location, (name).c_str());
451}
452
453void GL3Renderer::setViewPort(int32_t width, int32_t height)
454{
455 this->viewPortWidth = width;
456 this->viewPortHeight = height;
457}
458
460{
461 glViewport(0, 0, viewPortWidth, viewPortHeight);
462}
463
464void GL3Renderer::setClearColor(float red, float green, float blue, float alpha)
465{
466 glClearColor(red, green, blue, alpha);
467}
468
469void GL3Renderer::enableCulling(int contextIdx)
470{
471 glEnable(GL_CULL_FACE);
472}
473
474void GL3Renderer::disableCulling(int contextIdx)
475{
476 glDisable(GL_CULL_FACE);
477}
478
479void GL3Renderer::setFrontFace(int contextIdx, int32_t frontFace)
480{
481 glFrontFace(frontFace);
482}
483
484void GL3Renderer::setCullFace(int32_t cullFace)
485{
486 glCullFace(cullFace);
487}
488
490{
491 glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
492 glEnable(GL_BLEND);
493}
494
496 glBlendFunc(GL_ONE, GL_ONE);
497 glEnable(GL_BLEND);
498}
499
501{
502 glDisable(GL_BLEND);
503}
504
506{
507 glDepthMask(true);
508}
509
511{
512 glDepthMask(false);
513}
514
516{
517 glDisable(GL_DEPTH_TEST);
518}
519
521{
522 glEnable(GL_DEPTH_TEST);
523}
524
525void GL3Renderer::setDepthFunction(int32_t depthFunction)
526{
527 // TODO: remove me
528 glDepthFunc(depthFunction);
529}
530
531void GL3Renderer::setColorMask(bool red, bool green, bool blue, bool alpha)
532{
533 glColorMask(red, green, blue, alpha);
534}
535
536void GL3Renderer::clear(int32_t mask)
537{
538 glClear(mask);
540}
541
543{
544 // generate open gl texture
545 uint32_t textureId;
546 glGenTextures(1, &textureId);
547 return textureId;
548}
549
550int32_t GL3Renderer::createDepthBufferTexture(int32_t width, int32_t height, int32_t cubeMapTextureId, int32_t cubeMapTextureIndex)
551{
552 uint32_t depthTextureId;
553 // create depth texture
554 glGenTextures(1, &depthTextureId);
555 glBindTexture(GL_TEXTURE_2D, depthTextureId);
556 // create depth texture
557 glTexImage2D(GL_TEXTURE_2D, 0, GL_DEPTH_COMPONENT, width, height, 0, GL_DEPTH_COMPONENT, GL_FLOAT, nullptr);
558 // depth texture parameter
559 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
560 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
561 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE);
562 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE);
563 // unbind, return
564 glBindTexture(GL_TEXTURE_2D, ID_NONE);
565 return depthTextureId;
566}
567
568int32_t GL3Renderer::createColorBufferTexture(int32_t width, int32_t height, int32_t cubeMapTextureId, int32_t cubeMapTextureIndex)
569{
570 uint32_t colorBufferTextureId;
571 // create color texture
572 glGenTextures(1, &colorBufferTextureId);
573 glBindTexture(GL_TEXTURE_2D, colorBufferTextureId);
574 // create color texture
575 glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, width, height, 0, GL_RGBA, GL_UNSIGNED_BYTE, nullptr);
576 // color texture parameter
577 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
578 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
579 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT);
580 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT);
581 // unbind, return
582 glBindTexture(GL_TEXTURE_2D, ID_NONE);
583 return colorBufferTextureId;
584}
585
586int32_t GL3Renderer::createGBufferGeometryTexture(int32_t width, int32_t height) {
587 uint32_t gBufferGeometryTextureId;
588 glGenTextures(1, &gBufferGeometryTextureId);
589 glBindTexture(GL_TEXTURE_2D, gBufferGeometryTextureId);
590 glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA16F, width, height, 0, GL_RGBA, GL_FLOAT, nullptr);
591 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
592 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
593 glBindTexture(GL_TEXTURE_2D, ID_NONE);
594 return gBufferGeometryTextureId;
595}
596
597int32_t GL3Renderer::createGBufferColorTexture(int32_t width, int32_t height) {
598 uint32_t gBufferColorTextureId;
599 glGenTextures(1, &gBufferColorTextureId);
600 glBindTexture(GL_TEXTURE_2D, gBufferColorTextureId);
601 glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, width, height, 0, GL_RGBA, GL_UNSIGNED_BYTE, nullptr);
602 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
603 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
604 glBindTexture(GL_TEXTURE_2D, ID_NONE);
605 return gBufferColorTextureId;
606}
607
608void GL3Renderer::uploadTexture(int contextIdx, Texture* texture)
609{
610 glTexImage2D(
611 GL_TEXTURE_2D,
612 0,
613 texture->getDepth() == 32?GL_RGBA:GL_RGB,
614 texture->getTextureWidth(),
615 texture->getTextureHeight(),
616 0,
617 texture->getDepth() == 32?GL_RGBA:GL_RGB,
618 GL_UNSIGNED_BYTE,
619 texture->getTextureData()->getBuffer()
620 );
621 if (texture->getAtlasSize() > 1) {
622 if (texture->isUseMipMap() == true) {
623 float maxLodBias;
624 glGetFloatv(GL_MAX_TEXTURE_LOD_BIAS, &maxLodBias);
625 glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_LOD_BIAS, -Math::clamp(static_cast<float>(texture->getAtlasSize()) * 0.125f, 0.0f, maxLodBias));
626 auto borderSize = 32;
627 auto maxLevel = 0;
628 while (borderSize > 4) {
629 maxLevel++;
630 borderSize/= 2;
631 }
632 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_BASE_LEVEL, 0);
633 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAX_LEVEL, maxLevel - 1);
634 glGenerateMipmap(GL_TEXTURE_2D);
635 }
636 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE);
637 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE);
638 } else {
639 if (texture->isUseMipMap() == true) glGenerateMipmap(GL_TEXTURE_2D);
640 if (texture->isRepeat() == true) {
641 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT);
642 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT);
643 } else {
644 float color[4] = {0.0f, 0.0f, 0.0f, 0.0f};
645 if (texture->getClampMode() == Texture::CLAMPMODE_TRANSPARENTPIXEL) glTexParameterfv(GL_TEXTURE_2D, GL_TEXTURE_BORDER_COLOR, color);
646 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, texture->getClampMode() == Texture::CLAMPMODE_EDGE?GL_CLAMP_TO_EDGE:GL_CLAMP_TO_BORDER);
647 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, texture->getClampMode() == Texture::CLAMPMODE_EDGE?GL_CLAMP_TO_EDGE:GL_CLAMP_TO_BORDER);
648 }
649 }
650 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, texture->isUseMipMap() == true?GL_LINEAR_MIPMAP_LINEAR:GL_LINEAR);
651 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
653}
654
655void GL3Renderer::uploadCubeMapTexture(int contextIdx, Texture* textureLeft, Texture* textureRight, Texture* textureTop, Texture* textureBottom, Texture* textureFront, Texture* textureBack) {
656 glTexImage2D(
657 GL_TEXTURE_CUBE_MAP_NEGATIVE_X,
658 0,
659 textureLeft->getDepth() == 32?GL_RGBA:GL_RGB,
660 textureLeft->getTextureWidth(),
661 textureLeft->getTextureHeight(),
662 0,
663 textureLeft->getDepth() == 32?GL_RGBA:GL_RGB,
664 GL_UNSIGNED_BYTE,
665 textureLeft->getTextureData()->getBuffer()
666 );
667 glTexImage2D(
668 GL_TEXTURE_CUBE_MAP_POSITIVE_X,
669 0,
670 textureRight->getDepth() == 32?GL_RGBA:GL_RGB,
671 textureRight->getTextureWidth(),
672 textureRight->getTextureHeight(),
673 0,
674 textureRight->getDepth() == 32?GL_RGBA:GL_RGB,
675 GL_UNSIGNED_BYTE,
676 textureRight->getTextureData()->getBuffer()
677 );
678 glTexImage2D(
679 GL_TEXTURE_CUBE_MAP_POSITIVE_Y,
680 0,
681 textureTop->getDepth() == 32?GL_RGBA:GL_RGB,
682 textureTop->getTextureWidth(),
683 textureTop->getTextureHeight(),
684 0,
685 textureTop->getDepth() == 32?GL_RGBA:GL_RGB,
686 GL_UNSIGNED_BYTE,
687 textureTop->getTextureData()->getBuffer()
688 );
689 glTexImage2D(
690 GL_TEXTURE_CUBE_MAP_NEGATIVE_Y,
691 0,
692 textureBottom->getDepth() == 32?GL_RGBA:GL_RGB,
693 textureBottom->getTextureWidth(),
694 textureBottom->getTextureHeight(),
695 0,
696 textureBottom->getDepth() == 32?GL_RGBA:GL_RGB,
697 GL_UNSIGNED_BYTE,
698 textureBottom->getTextureData()->getBuffer()
699 );
700 glTexImage2D(
701 GL_TEXTURE_CUBE_MAP_POSITIVE_Z,
702 0,
703 textureFront->getDepth() == 32?GL_RGBA:GL_RGB,
704 textureFront->getTextureWidth(),
705 textureFront->getTextureHeight(),
706 0,
707 textureFront->getDepth() == 32?GL_RGBA:GL_RGB,
708 GL_UNSIGNED_BYTE,
709 textureFront->getTextureData()->getBuffer()
710 );
711 glTexImage2D(
712 GL_TEXTURE_CUBE_MAP_NEGATIVE_Z,
713 0,
714 textureBack->getDepth() == 32?GL_RGBA:GL_RGB,
715 textureBack->getTextureWidth(),
716 textureBack->getTextureHeight(),
717 0,
718 textureBack->getDepth() == 32?GL_RGBA:GL_RGB,
719 GL_UNSIGNED_BYTE,
720 textureBack->getTextureData()->getBuffer()
721 );
722 glTexParameteri(GL_TEXTURE_CUBE_MAP, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
723 glTexParameteri(GL_TEXTURE_CUBE_MAP, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
724 glTexParameteri(GL_TEXTURE_CUBE_MAP, GL_TEXTURE_WRAP_R, GL_CLAMP_TO_EDGE);
725 glTexParameteri(GL_TEXTURE_CUBE_MAP, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE);
726 glTexParameteri(GL_TEXTURE_CUBE_MAP, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE);
728}
729
730int32_t GL3Renderer::createCubeMapTexture(int contextIdx, int32_t width, int32_t height) {
731 // generate open gl texture
732 uint32_t textureId;
733 glGenTextures(1, &textureId);
734 glBindTexture(GL_TEXTURE_CUBE_MAP, textureId);
735 glTexParameteri(GL_TEXTURE_CUBE_MAP, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
736 glTexParameteri(GL_TEXTURE_CUBE_MAP, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
737 glTexParameteri(GL_TEXTURE_CUBE_MAP, GL_TEXTURE_WRAP_R, GL_CLAMP_TO_EDGE);
738 glTexParameteri(GL_TEXTURE_CUBE_MAP, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE);
739 glTexParameteri(GL_TEXTURE_CUBE_MAP, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE);
740 glTexImage2D(
741 GL_TEXTURE_CUBE_MAP_NEGATIVE_X,
742 0,
743 GL_RGBA,
744 width,
745 height,
746 0,
747 GL_RGBA,
748 GL_UNSIGNED_BYTE,
749 nullptr
750 );
751 glTexImage2D(
752 GL_TEXTURE_CUBE_MAP_POSITIVE_X,
753 0,
754 GL_RGBA,
755 width,
756 height,
757 0,
758 GL_RGBA,
759 GL_UNSIGNED_BYTE,
760 nullptr
761 );
762 glTexImage2D(
763 GL_TEXTURE_CUBE_MAP_POSITIVE_Y,
764 0,
765 GL_RGBA,
766 width,
767 height,
768 0,
769 GL_RGBA,
770 GL_UNSIGNED_BYTE,
771 nullptr
772 );
773 glTexImage2D(
774 GL_TEXTURE_CUBE_MAP_NEGATIVE_Y,
775 0,
776 GL_RGBA,
777 width,
778 height,
779 0,
780 GL_RGBA,
781 GL_UNSIGNED_BYTE,
782 nullptr
783 );
784 glTexImage2D(
785 GL_TEXTURE_CUBE_MAP_POSITIVE_Z,
786 0,
787 GL_RGBA,
788 width,
789 height,
790 0,
791 GL_RGBA,
792 GL_UNSIGNED_BYTE,
793 nullptr
794 );
795 glTexImage2D(
796 GL_TEXTURE_CUBE_MAP_NEGATIVE_Z,
797 0,
798 GL_RGBA,
799 width,
800 height,
801 0,
802 GL_RGBA,
803 GL_UNSIGNED_BYTE,
804 nullptr
805 );
806 glBindTexture(GL_TEXTURE_CUBE_MAP, 0);
807 return textureId;
808}
809
810void GL3Renderer::resizeDepthBufferTexture(int32_t textureId, int32_t width, int32_t height)
811{
812 glBindTexture(GL_TEXTURE_2D, textureId);
813 glTexImage2D(GL_TEXTURE_2D, 0, GL_DEPTH_COMPONENT, width, height, 0, GL_DEPTH_COMPONENT, GL_FLOAT, nullptr);
814 glBindTexture(GL_TEXTURE_2D, 0);
815}
816
817void GL3Renderer::resizeColorBufferTexture(int32_t textureId, int32_t width, int32_t height)
818{
819 glBindTexture(GL_TEXTURE_2D, textureId);
820 glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, width, height, 0, GL_RGBA, GL_UNSIGNED_BYTE, nullptr);
821 glBindTexture(GL_TEXTURE_2D, 0);
822}
823
824void GL3Renderer::resizeGBufferGeometryTexture(int32_t textureId, int32_t width, int32_t height) {
825 glBindTexture(GL_TEXTURE_2D, textureId);
826 glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA16F, width, height, 0, GL_RGBA, GL_FLOAT, nullptr);
827 glBindTexture(GL_TEXTURE_2D, 0);
828}
829
830void GL3Renderer::resizeGBufferColorTexture(int32_t textureId, int32_t width, int32_t height) {
831 glBindTexture(GL_TEXTURE_2D, textureId);
832 glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, width, height, 0, GL_RGBA, GL_UNSIGNED_BYTE, nullptr);
833 glBindTexture(GL_TEXTURE_2D, 0);
834}
835
836void GL3Renderer::bindTexture(int contextIdx, int32_t textureId)
837{
838 glBindTexture(GL_TEXTURE_2D, textureId);
839 onBindTexture(contextIdx, textureId);
840}
841
842void GL3Renderer::bindCubeMapTexture(int contextIdx, int32_t textureId) {
843 glBindTexture(GL_TEXTURE_CUBE_MAP, textureId);
844 onBindTexture(contextIdx, textureId);
845}
846
847void GL3Renderer::disposeTexture(int32_t textureId)
848{
849 glDeleteTextures(1, (const uint32_t*)&textureId);
851}
852
853int32_t GL3Renderer::createFramebufferObject(int32_t depthBufferTextureId, int32_t colorBufferTextureId, int32_t cubeMapTextureId, int32_t cubeMapTextureIndex)
854{
855 uint32_t frameBufferId;
856 // create a frame buffer object
857 glGenFramebuffers(1, &frameBufferId);
858 glBindFramebuffer(GL_FRAMEBUFFER, frameBufferId);
859 // attach the depth buffer texture to FBO
860 if (depthBufferTextureId != ID_NONE) {
861 glFramebufferTexture(GL_FRAMEBUFFER, GL_DEPTH_ATTACHMENT, depthBufferTextureId, 0);
862 }
863 // attach the depth buffer texture to FBO
864 if (colorBufferTextureId != ID_NONE) {
865 glFramebufferTexture(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, colorBufferTextureId, 0);
866 glDrawBuffer(GL_COLOR_ATTACHMENT0);
867 glReadBuffer(GL_COLOR_ATTACHMENT0);
868 } else
869 if (cubeMapTextureId != ID_NONE) {
870 glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, cubeMapTextureIndex, cubeMapTextureId, 0);
871 } else {
872 glDrawBuffer(GL_NONE);
873 glReadBuffer(GL_NONE);
874 }
875 // check FBO status
876 int32_t fboStatus = glCheckFramebufferStatus(GL_FRAMEBUFFER);
877 if (fboStatus != GL_FRAMEBUFFER_COMPLETE) {
878 Console::println(string("GL_FRAMEBUFFER_COMPLETE_EXT failed, CANNOT use FBO: "+ to_string(fboStatus)));
879 }
880 // switch back to window-system-provided framebuffer
881 glBindFramebuffer(GL_FRAMEBUFFER, 0);
882 return frameBufferId;
883}
884
886 int32_t depthBufferTextureId,
887 int32_t geometryBufferTextureId1,
888 int32_t geometryBufferTextureId2,
889 int32_t geometryBufferTextureId3,
890 int32_t colorBufferTextureId1,
891 int32_t colorBufferTextureId2,
892 int32_t colorBufferTextureId3,
893 int32_t colorBufferTextureId4,
894 int32_t colorBufferTextureId5
895) {
896 uint32_t frameBufferId;
897 // create a frame buffer object
898 glGenFramebuffers(1, &frameBufferId);
899 glBindFramebuffer(GL_FRAMEBUFFER, frameBufferId);
900 // attach the depth buffer texture to FBO
901 if (depthBufferTextureId != ID_NONE) {
902 glFramebufferTexture(GL_FRAMEBUFFER, GL_DEPTH_ATTACHMENT, depthBufferTextureId, 0);
903 }
904 vector<uint32_t> drawBuffers;
905 // attach geometry textures
906 if (geometryBufferTextureId1 != ID_NONE) {
907 glFramebufferTexture(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, geometryBufferTextureId1, 0);
908 drawBuffers.push_back(GL_COLOR_ATTACHMENT0);
909 }
910 if (geometryBufferTextureId2 != ID_NONE) {
911 glFramebufferTexture(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT1, geometryBufferTextureId2, 0);
912 drawBuffers.push_back(GL_COLOR_ATTACHMENT1);
913 }
914 if (geometryBufferTextureId3 != ID_NONE) {
915 glFramebufferTexture(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT2, geometryBufferTextureId3, 0);
916 drawBuffers.push_back(GL_COLOR_ATTACHMENT2);
917 }
918 // attach color textures
919 if (colorBufferTextureId1 != ID_NONE) {
920 glFramebufferTexture(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT3, colorBufferTextureId1, 0);
921 drawBuffers.push_back(GL_COLOR_ATTACHMENT3);
922 }
923 if (colorBufferTextureId2 != ID_NONE) {
924 glFramebufferTexture(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT4, colorBufferTextureId2, 0);
925 drawBuffers.push_back(GL_COLOR_ATTACHMENT4);
926 }
927 if (colorBufferTextureId3 != ID_NONE) {
928 glFramebufferTexture(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT5, colorBufferTextureId3, 0);
929 drawBuffers.push_back(GL_COLOR_ATTACHMENT5);
930 }
931 if (colorBufferTextureId4 != ID_NONE) {
932 glFramebufferTexture(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT6, colorBufferTextureId4, 0);
933 drawBuffers.push_back(GL_COLOR_ATTACHMENT6);
934 }
935 if (colorBufferTextureId5 != ID_NONE) {
936 glFramebufferTexture(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT7, colorBufferTextureId5, 0);
937 drawBuffers.push_back(GL_COLOR_ATTACHMENT7);
938 }
939 //
940 glDrawBuffers(drawBuffers.size(), drawBuffers.data());
941 // check FBO status
942 int32_t fboStatus = glCheckFramebufferStatus(GL_FRAMEBUFFER);
943 if (fboStatus != GL_FRAMEBUFFER_COMPLETE) {
944 Console::println(string("GL_FRAMEBUFFER_COMPLETE_EXT failed, CANNOT use FBO: "+ to_string(fboStatus)));
945 }
946 // switch back to window-system-provided framebuffer
947 glBindFramebuffer(GL_FRAMEBUFFER, 0);
948 return frameBufferId;
949}
950
951void GL3Renderer::bindFrameBuffer(int32_t frameBufferId)
952{
953 glBindFramebuffer(GL_FRAMEBUFFER, frameBufferId);
954}
955
956void GL3Renderer::disposeFrameBufferObject(int32_t frameBufferId)
957{
958 glDeleteFramebuffers(1, (uint32_t*)&frameBufferId);
959}
960
961vector<int32_t> GL3Renderer::createBufferObjects(int32_t buffers, bool useGPUMemory, bool shared)
962{
963 vector<int32_t> bufferObjectIds;
964 bufferObjectIds.resize(buffers);
965 glGenBuffers(buffers, (uint32_t*)bufferObjectIds.data());
966 for (auto bufferObjectId: bufferObjectIds) vbosUsage[bufferObjectId] = useGPUMemory == true?GL_STATIC_DRAW:GL_STREAM_DRAW;
967 return bufferObjectIds;
968}
969
970void GL3Renderer::uploadBufferObject(int contextIdx, int32_t bufferObjectId, int32_t size, FloatBuffer* data)
971{
972 glBindBuffer(GL_ARRAY_BUFFER, bufferObjectId);
973 glBufferData(GL_ARRAY_BUFFER, size, data->getBuffer(), vbosUsage[bufferObjectId]);
974 glBindBuffer(GL_ARRAY_BUFFER, ID_NONE);
976}
977
978void GL3Renderer::uploadBufferObject(int contextIdx, int32_t bufferObjectId, int32_t size, ShortBuffer* data)
979{
980 glBindBuffer(GL_ARRAY_BUFFER, bufferObjectId);
981 glBufferData(GL_ARRAY_BUFFER, size, data->getBuffer(), vbosUsage[bufferObjectId]);
982 glBindBuffer(GL_ARRAY_BUFFER, ID_NONE);
984}
985
986void GL3Renderer::uploadBufferObject(int contextIdx, int32_t bufferObjectId, int32_t size, IntBuffer* data)
987{
988 glBindBuffer(GL_ARRAY_BUFFER, bufferObjectId);
989 glBufferData(GL_ARRAY_BUFFER, size, data->getBuffer(), vbosUsage[bufferObjectId]);
990 glBindBuffer(GL_ARRAY_BUFFER, ID_NONE);
992}
993
994void GL3Renderer::uploadIndicesBufferObject(int contextIdx, int32_t bufferObjectId, int32_t size, ShortBuffer* data)
995{
996 Console::println(string("GL3Renderer::uploadIndicesBufferObject()::not implemented yet"));
997}
998
999void GL3Renderer::uploadIndicesBufferObject(int contextIdx, int32_t bufferObjectId, int32_t size, IntBuffer* data)
1000{
1001 glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, bufferObjectId);
1002 glBufferData(GL_ELEMENT_ARRAY_BUFFER, size, data->getBuffer(), vbosUsage[bufferObjectId]);
1003 glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, ID_NONE);
1005}
1006
1007void GL3Renderer::bindIndicesBufferObject(int contextIdx, int32_t bufferObjectId)
1008{
1009 glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, bufferObjectId);
1010}
1011
1012void GL3Renderer::bindTextureCoordinatesBufferObject(int contextIdx, int32_t bufferObjectId)
1013{
1014 glBindBuffer(GL_ARRAY_BUFFER, bufferObjectId);
1015 glEnableVertexAttribArray(2);
1016 glVertexAttribPointer(2, 2, GL_FLOAT, false, 0, 0LL);
1017}
1018
1019void GL3Renderer::bindVerticesBufferObject(int contextIdx, int32_t bufferObjectId)
1020{
1021 glBindBuffer(GL_ARRAY_BUFFER, bufferObjectId);
1022 glEnableVertexAttribArray(0);
1023 glVertexAttribPointer(0, 3, GL_FLOAT, false, 0, 0LL);
1024}
1025
1026void GL3Renderer::bindNormalsBufferObject(int contextIdx, int32_t bufferObjectId)
1027{
1028 glBindBuffer(GL_ARRAY_BUFFER, bufferObjectId);
1029 glEnableVertexAttribArray(1);
1030 glVertexAttribPointer(1, 3, GL_FLOAT, false, 0, 0LL);
1031}
1032
1033void GL3Renderer::bindColorsBufferObject(int contextIdx, int32_t bufferObjectId)
1034{
1035 glBindBuffer(GL_ARRAY_BUFFER, bufferObjectId);
1036 glEnableVertexAttribArray(3);
1037 glVertexAttribPointer(3, 4, GL_FLOAT, false, 0, 0LL);
1038}
1039
1040void GL3Renderer::bindTangentsBufferObject(int contextIdx, int32_t bufferObjectId)
1041{
1042 glBindBuffer(GL_ARRAY_BUFFER, bufferObjectId);
1043 glEnableVertexAttribArray(4);
1044 glVertexAttribPointer(4, 3, GL_FLOAT, false, 0, 0LL);
1045}
1046
1047void GL3Renderer::bindBitangentsBufferObject(int contextIdx, int32_t bufferObjectId)
1048{
1049 glBindBuffer(GL_ARRAY_BUFFER, bufferObjectId);
1050 glEnableVertexAttribArray(5);
1051 glVertexAttribPointer(5, 3, GL_FLOAT, false, 0, 0LL);
1052}
1053
1054void GL3Renderer::bindModelMatricesBufferObject(int contextIdx, int32_t bufferObjectId) {
1055 glBindBuffer(GL_ARRAY_BUFFER, bufferObjectId);
1056 glEnableVertexAttribArray(6);
1057 glEnableVertexAttribArray(7);
1058 glEnableVertexAttribArray(8);
1059 glEnableVertexAttribArray(9);
1060 glVertexAttribPointer(6, 4, GL_FLOAT, false, 4 * 4 * sizeof(float), (void*)(0 * 4 * sizeof(float)));
1061 glVertexAttribPointer(7, 4, GL_FLOAT, false, 4 * 4 * sizeof(float), (void*)(1 * 4 * sizeof(float)));
1062 glVertexAttribPointer(8, 4, GL_FLOAT, false, 4 * 4 * sizeof(float), (void*)(2 * 4 * sizeof(float)));
1063 glVertexAttribPointer(9, 4, GL_FLOAT, false, 4 * 4 * sizeof(float), (void*)(3 * 4 * sizeof(float)));
1064 glVertexAttribDivisor(6, 1);
1065 glVertexAttribDivisor(7, 1);
1066 glVertexAttribDivisor(8, 1);
1067 glVertexAttribDivisor(9, 1);
1068}
1069
1070void GL3Renderer::bindEffectColorMulsBufferObject(int contextIdx, int32_t bufferObjectId, int32_t divisor) {
1071 glBindBuffer(GL_ARRAY_BUFFER, bufferObjectId);
1072 glEnableVertexAttribArray(10);
1073 glVertexAttribPointer(10, 4, GL_FLOAT, false, 0, 0LL);
1074 glVertexAttribDivisor(10, divisor);
1075}
1076
1077void GL3Renderer::bindEffectColorAddsBufferObject(int contextIdx, int32_t bufferObjectId, int32_t divisor) {
1078 glBindBuffer(GL_ARRAY_BUFFER, bufferObjectId);
1079 glEnableVertexAttribArray(11);
1080 glVertexAttribPointer(11, 4, GL_FLOAT, false, 0, 0LL);
1081 glVertexAttribDivisor(11, divisor);
1082}
1083
1084void GL3Renderer::bindOriginsBufferObject(int contextIdx, int32_t bufferObjectId) {
1085 glBindBuffer(GL_ARRAY_BUFFER, bufferObjectId);
1086 glEnableVertexAttribArray(12);
1087 glVertexAttribPointer(12, 3, GL_FLOAT, false, 0, 0LL);
1088}
1089
1090void GL3Renderer::bindTextureSpriteIndicesBufferObject(int contextIdx, int32_t bufferObjectId) {
1091 glBindBuffer(GL_ARRAY_BUFFER, bufferObjectId);
1092 glEnableVertexAttribArray(1);
1093 glVertexAttribIPointer(1, 2, GL_UNSIGNED_SHORT, 0, 0LL);
1094}
1095
1096void GL3Renderer::bindPointSizesBufferObject(int contextIdx, int32_t bufferObjectId) {
1097 glBindBuffer(GL_ARRAY_BUFFER, bufferObjectId);
1098 glEnableVertexAttribArray(5);
1099 glVertexAttribPointer(5, 1, GL_FLOAT, false, 0, 0LL);
1100}
1101
1102void GL3Renderer::bindSpriteSheetDimensionBufferObject(int contextIdx, int32_t bufferObjectId) {
1103 glBindBuffer(GL_ARRAY_BUFFER, bufferObjectId);
1104 glEnableVertexAttribArray(6);
1105 glVertexAttribIPointer(6, 2, GL_UNSIGNED_SHORT, 0, 0LL);
1106 glVertexAttribDivisor(6, 0);
1107}
1108
1109void GL3Renderer::drawInstancedIndexedTrianglesFromBufferObjects(int contextIdx, int32_t triangles, int32_t trianglesOffset, int32_t instances)
1110{
1111 #define BUFFER_OFFSET(i) ((void*)(i))
1112 glDrawElementsInstanced(GL_TRIANGLES, triangles * 3, GL_UNSIGNED_INT, BUFFER_OFFSET(static_cast< int64_t >(trianglesOffset) * 3LL * 4LL), instances);
1114 statistics.instances+= instances;
1115 statistics.triangles+= triangles * instances;
1116}
1117
1118void GL3Renderer::drawIndexedTrianglesFromBufferObjects(int contextIdx, int32_t triangles, int32_t trianglesOffset)
1119{
1120 #define BUFFER_OFFSET(i) ((void*)(i))
1121 glDrawElements(GL_TRIANGLES, triangles * 3, GL_UNSIGNED_INT, BUFFER_OFFSET(static_cast< int64_t >(trianglesOffset) * 3LL * 4LL));
1124 statistics.triangles+= triangles;
1125}
1126
1127void GL3Renderer::drawInstancedTrianglesFromBufferObjects(int contextIdx, int32_t triangles, int32_t trianglesOffset, int32_t instances) {
1128 glDrawArraysInstanced(GL_TRIANGLES, trianglesOffset * 3, triangles * 3, instances);
1130 statistics.instances+= instances;
1131 statistics.triangles+= triangles * instances;
1132}
1133
1134void GL3Renderer::drawTrianglesFromBufferObjects(int contextIdx, int32_t triangles, int32_t trianglesOffset)
1135{
1136 glDrawArrays(GL_TRIANGLES, trianglesOffset * 3, triangles * 3);
1139 statistics.triangles+= triangles;
1140}
1141
1142void GL3Renderer::drawPointsFromBufferObjects(int contextIdx, int32_t points, int32_t pointsOffset)
1143{
1144 glDrawArrays(GL_POINTS, pointsOffset, points);
1146 statistics.points+= points;
1147}
1148
1149void GL3Renderer::setLineWidth(float lineWidth)
1150{
1151 glLineWidth(lineWidth);
1152}
1153
1154void GL3Renderer::drawLinesFromBufferObjects(int contextIdx, int32_t points, int32_t pointsOffset)
1155{
1156 glDrawArrays(GL_LINES, pointsOffset, points);
1158 statistics.linePoints+= points;
1159}
1160
1162{
1163 glDisableVertexAttribArray(0);
1164 glDisableVertexAttribArray(1);
1165 glDisableVertexAttribArray(2);
1166 glDisableVertexAttribArray(3);
1167 glDisableVertexAttribArray(4);
1168 glDisableVertexAttribArray(5);
1169 glDisableVertexAttribArray(6);
1170 glDisableVertexAttribArray(7);
1171 glDisableVertexAttribArray(8);
1172 glDisableVertexAttribArray(9);
1173 glDisableVertexAttribArray(10);
1174 glDisableVertexAttribArray(11);
1175 glDisableVertexAttribArray(12);
1176 glBindBuffer(GL_ARRAY_BUFFER, ID_NONE);
1177 glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, ID_NONE);
1178}
1179
1180void GL3Renderer::disposeBufferObjects(vector<int32_t>& bufferObjectIds)
1181{
1182 for (auto& bufferObjectId: bufferObjectIds) vbosUsage.erase(bufferObjectId);
1183 glDeleteBuffers(bufferObjectIds.size(), (const uint32_t*)bufferObjectIds.data());
1184 statistics.disposedBuffers+= bufferObjectIds.size();
1185}
1186
1187int32_t GL3Renderer::getTextureUnit(int contextIdx)
1188{
1189 return activeTextureUnit;
1190}
1191
1192void GL3Renderer::setTextureUnit(int contextIdx, int32_t textureUnit)
1193{
1194 this->activeTextureUnit = textureUnit;
1195 glActiveTexture(GL_TEXTURE0 + textureUnit);
1196}
1197
1198float GL3Renderer::readPixelDepth(int32_t x, int32_t y)
1199{
1200 float depth;
1201 glReadPixels(x, y, 1, 1, GL_DEPTH_COMPONENT, GL_FLOAT, &depth);
1202 return depth;
1203}
1204
1205ByteBuffer* GL3Renderer::readPixels(int32_t x, int32_t y, int32_t width, int32_t height)
1206{
1207 auto pixelBuffer = ByteBuffer::allocate(width * height * 4);
1208 glPixelStorei(GL_PACK_ALIGNMENT, 1);
1209 glReadPixels(x, y, width, height, GL_RGBA, GL_UNSIGNED_BYTE, pixelBuffer->getBuffer());
1210 return pixelBuffer;
1211}
1212
1214{
1216 glBindTexture(GL_TEXTURE_2D, ID_NONE);
1217 glBlendEquationSeparate(GL_FUNC_ADD, GL_FUNC_ADD);
1218 glBlendFuncSeparate(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA, GL_ONE, GL_ONE_MINUS_SRC_ALPHA);
1219 glEnable(GL_BLEND);
1220 glDisable(GL_DEPTH_TEST);
1221 glDisable(GL_CULL_FACE);
1222 glGetError();
1223}
1224
1226{
1227 glGetError();
1228 glBindTexture(GL_TEXTURE_2D, ID_NONE);
1229 glDisable(GL_BLEND);
1230 glEnable(GL_DEPTH_TEST);
1231 glEnable(GL_CULL_FACE);
1232}
1233
1235{
1236 auto error = glGetError();
1237 if (error != GL_NO_ERROR) {
1238 Console::println(string("OpenGL Error: (" + to_string(error) + ") @: " + __FILE__ + ":" + to_string(line)));
1239 }
1240}
1241
1242void GL3Renderer::dispatchCompute(int contextIdx, int32_t numGroupsX, int32_t numGroupsY, int32_t numGroupsZ) {
1243 #if defined (__APPLE__)
1244 clSkinningParameters.numGroupsX = numGroupsX;
1245 clSkinningParameters.numGroupsY = numGroupsY;
1246 glFinish();
1247 auto& _clSkinningParameters = clSkinningParameters;
1248 cl_int clError;
1249 array<cl_mem, 8> boundCLMemObjects;
1250 for (auto i = 0; i < _clSkinningParameters.boundGLBuffers.size(); i++) {
1251 boundCLMemObjects[i] = clCreateFromGLBuffer(clContext, _clSkinningParameters.boundGLBuffersWrite[i] == true?CL_MEM_WRITE_ONLY:CL_MEM_READ_ONLY, _clSkinningParameters.boundGLBuffers[i], &clError);
1252 clError = clSetKernelArg(clSkinningKernel, i, sizeof(cl_mem), &boundCLMemObjects[i]);
1253 }
1254 clSetKernelArg(clSkinningKernel, 8, sizeof(cl_int), &_clSkinningParameters.vertexCount);
1255 clSetKernelArg(clSkinningKernel, 9, sizeof(cl_int), &_clSkinningParameters.matrixCount);
1256 clSetKernelArg(clSkinningKernel, 10, sizeof(cl_int), &_clSkinningParameters.instanceCount);
1257 size_t local_size[] = {(size_t)16, (size_t)16};
1258 size_t global_size[] = {(size_t)_clSkinningParameters.numGroupsX * local_size[0], (size_t)_clSkinningParameters.numGroupsY * local_size[1]};
1259 clEnqueueAcquireGLObjects(clCommandQueue, boundCLMemObjects.size(), boundCLMemObjects.data(), 0, nullptr, nullptr);
1260 clEnqueueNDRangeKernel(clCommandQueue, clSkinningKernel, 2, nullptr, global_size, local_size, 0, nullptr, nullptr);
1261 clEnqueueReleaseGLObjects(clCommandQueue, boundCLMemObjects.size(), boundCLMemObjects.data(), 0, nullptr, nullptr);
1262 clFinish(clCommandQueue);
1263 clSkinningParameters = CLSkinningParameters();
1264 #else
1265 glDispatchCompute(numGroupsX, numGroupsY, numGroupsZ);
1266 #endif
1268}
1269
1271 #if defined (__APPLE__)
1272 // no op
1273 #else
1274 // TODO: put barrier bits into paramters
1275 glMemoryBarrier(GL_ALL_BARRIER_BITS);
1276 #endif
1277}
1278
1279void GL3Renderer::uploadSkinningBufferObject(int contextIdx, int32_t bufferObjectId, int32_t size, FloatBuffer* data) {
1280 #if defined (__APPLE__)
1281 glBindBuffer(GL_ARRAY_BUFFER, bufferObjectId);
1282 glBufferData(GL_ARRAY_BUFFER, size, data->getBuffer(), vbosUsage[bufferObjectId]);
1283 glBindBuffer(GL_ARRAY_BUFFER, ID_NONE);
1285 #else
1286 glBindBuffer(GL_SHADER_STORAGE_BUFFER, bufferObjectId);
1287 glBufferData(GL_SHADER_STORAGE_BUFFER, size, data->getBuffer(), vbosUsage[bufferObjectId]);
1288 glBindBuffer(GL_SHADER_STORAGE_BUFFER, ID_NONE);
1290 #endif
1291}
1292
1293void GL3Renderer::uploadSkinningBufferObject(int contextIdx, int32_t bufferObjectId, int32_t size, IntBuffer* data) {
1294 #if defined (__APPLE__)
1295 glBindBuffer(GL_ARRAY_BUFFER, bufferObjectId);
1296 glBufferData(GL_ARRAY_BUFFER, size, data->getBuffer(), vbosUsage[bufferObjectId]);
1297 glBindBuffer(GL_ARRAY_BUFFER, ID_NONE);
1299 #else
1300 glBindBuffer(GL_SHADER_STORAGE_BUFFER, bufferObjectId);
1301 glBufferData(GL_SHADER_STORAGE_BUFFER, size, data->getBuffer(), vbosUsage[bufferObjectId]);
1302 glBindBuffer(GL_SHADER_STORAGE_BUFFER, ID_NONE);
1304 #endif
1305}
1306
1307#if defined (__APPLE__)
1308 inline void GL3Renderer::clBindGLBuffer(int32_t idx, int32_t bufferObjectId, bool write) {
1309 clSkinningParameters.boundGLBuffers[idx] = bufferObjectId;
1310 clSkinningParameters.boundGLBuffersWrite[idx] = write;
1311 }
1312#endif
1313
1314void GL3Renderer::bindSkinningVerticesBufferObject(int contextIdx, int32_t bufferObjectId) {
1315 #if defined (__APPLE__)
1316 clBindGLBuffer(0, bufferObjectId, false);
1317 #else
1318 glBindBufferBase(GL_SHADER_STORAGE_BUFFER, 0, bufferObjectId);
1319 #endif
1320}
1321
1322void GL3Renderer::bindSkinningNormalsBufferObject(int contextIdx, int32_t bufferObjectId) {
1323 #if defined (__APPLE__)
1324 clBindGLBuffer(1, bufferObjectId, false);
1325 #else
1326 glBindBufferBase(GL_SHADER_STORAGE_BUFFER, 1, bufferObjectId);
1327 #endif
1328}
1329
1330void GL3Renderer::bindSkinningVertexJointsBufferObject(int contextIdx, int32_t bufferObjectId) {
1331 #if defined (__APPLE__)
1332 clBindGLBuffer(2, bufferObjectId, false);
1333 #else
1334 glBindBufferBase(GL_SHADER_STORAGE_BUFFER, 2, bufferObjectId);
1335 #endif
1336}
1337
1338void GL3Renderer::bindSkinningVertexJointIdxsBufferObject(int contextIdx, int32_t bufferObjectId) {
1339 #if defined (__APPLE__)
1340 clBindGLBuffer(3, bufferObjectId, false);
1341 #else
1342 glBindBufferBase(GL_SHADER_STORAGE_BUFFER, 3, bufferObjectId);
1343 #endif
1344}
1345
1346void GL3Renderer::bindSkinningVertexJointWeightsBufferObject(int contextIdx, int32_t bufferObjectId) {
1347 #if defined (__APPLE__)
1348 clBindGLBuffer(4, bufferObjectId, false);
1349 #else
1350 glBindBufferBase(GL_SHADER_STORAGE_BUFFER, 4, bufferObjectId);
1351 #endif
1352}
1353
1354void GL3Renderer::bindSkinningVerticesResultBufferObject(int contextIdx, int32_t bufferObjectId) {
1355 #if defined (__APPLE__)
1356 clBindGLBuffer(5, bufferObjectId, true);
1357 #else
1358 glBindBufferBase(GL_SHADER_STORAGE_BUFFER, 5, bufferObjectId);
1359 #endif
1360}
1361
1362void GL3Renderer::bindSkinningNormalsResultBufferObject(int contextIdx, int32_t bufferObjectId) {
1363 #if defined (__APPLE__)
1364 clBindGLBuffer(6, bufferObjectId, true);
1365 #else
1366 glBindBufferBase(GL_SHADER_STORAGE_BUFFER, 6, bufferObjectId);
1367 #endif
1368}
1369
1370void GL3Renderer::bindSkinningMatricesBufferObject(int contextIdx, int32_t bufferObjectId) {
1371 #if defined (__APPLE__)
1372 clBindGLBuffer(7, bufferObjectId, false);
1373 #else
1374 glBindBufferBase(GL_SHADER_STORAGE_BUFFER, 7, bufferObjectId);
1375 #endif
1376}
1377
1378void GL3Renderer::setVSync(bool vSync) {
1379 // no op
1380}
1381
1383 auto stats = statistics;
1384 statistics.time = Time::getCurrentMillis();
1392 statistics.points = 0;
1398 statistics.submits = 0;
1401 return stats;
1402}
#define BUFFER_OFFSET(i)
Engine main class.
Definition: Engine.h:122
Frame buffer class.
Definition: FrameBuffer.h:21
void setClearColor(float red, float green, float blue, float alpha) override
Set up clear color.
void enableDepthBufferWriting() override
Enable depth buffer writing.
void bindSpriteSheetDimensionBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind sprite sheet dimension buffer object.
void bindTexture(int contextIdx, int32_t textureId) override
Binds a texture with given id or unbinds when using ID_NONE.
void clear(int32_t mask) override
Clear render buffer with given mask.
void bindTangentsBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind tangents buffer object.
void disposeBufferObjects(vector< int32_t > &bufferObjectIds) override
Disposes a frame buffer object.
int32_t getTextureUnit(int contextIdx) override
Get texture unit.
void setProgramUniformFloatMatrix3x3(int contextIdx, int32_t uniformId, const array< float, 9 > &data) override
Set up a float matrix 3x3 uniform value.
void doneGuiMode() override
Set up renderer for 3d rendering.
void setColorMask(bool red, bool green, bool blue, bool alpha) override
Set up GL rendering colormask.
void bindTextureCoordinatesBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind texture coordinates buffer object.
void bindSkinningVertexJointWeightsBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind skinning vertex joint weights buffer object.
void dispatchCompute(int contextIdx, int32_t numGroupsX, int32_t numGroupsY, int32_t numGroupsZ) override
Dispatch compute.
void attachShaderToProgram(int32_t programId, int32_t shaderId) override
Attaches a shader to a program.
void setFrontFace(int contextIdx, int32_t frontFace) override
Set up clock wise or counter clock wise faces as front face.
void setTextureUnit(int contextIdx, int32_t textureUnit) override
Sets up texture unit.
void drawLinesFromBufferObjects(int contextIdx, int32_t points, int32_t pointsOffset) override
Draw lines from buffer objects.
void disableBlending() override
Disables blending.
void setProgramAttributeLocation(int32_t programId, int32_t location, const string &name) override
Bind attribute to a input location.
void setViewPort(int32_t width, int32_t height) override
Set up viewport parameter.
void bindSkinningVerticesResultBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind skinning vertices result buffer object.
void setProgramUniformFloat(int contextIdx, int32_t uniformId, float value) override
Set up a float uniform value.
const Renderer_Statistics getStatistics() override
void initialize() override
Initialize renderer.
void setVSync(bool vSync) override
Enable/Disable v-sync.
int32_t createGBufferColorTexture(int32_t width, int32_t height) override
Creates a geometry buffer color RGBA texture.
void setProgramUniformFloatVec3(int contextIdx, int32_t uniformId, const array< float, 3 > &data) override
Set up a float vec3 uniform value.
void bindEffectColorAddsBufferObject(int contextIdx, int32_t bufferObjectId, int32_t divisor) override
Bind effect color adds buffer object.
bool isBufferObjectsAvailable() override
Checks if buffer objects is available.
void disposeFrameBufferObject(int32_t frameBufferId) override
Disposes a frame buffer object.
void uploadBufferObject(int contextIdx, int32_t bufferObjectId, int32_t size, FloatBuffer *data) override
Uploads buffer data to buffer object.
void memoryBarrier() override
Memory barrier.
void setProgramUniformInteger(int contextIdx, int32_t uniformId, int32_t value) override
Set up a integer uniform value.
int32_t getProgramUniformLocation(int32_t programId, const string &name) override
Returns location of given uniform variable.
void disableDepthBufferWriting() override
Disable depth buffer writing.
void bindSkinningVertexJointIdxsBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind skinning vertex joint indices buffer object.
void bindNormalsBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind normals buffer object.
void drawPointsFromBufferObjects(int contextIdx, int32_t points, int32_t pointsOffset) override
Draw points from buffer objects.
void drawInstancedTrianglesFromBufferObjects(int contextIdx, int32_t triangles, int32_t trianglesOffset, int32_t instances) override
Draw instanced triangles from buffer objects.
void disposeTexture(int32_t textureId) override
Dispose a texture.
int32_t createColorBufferTexture(int32_t width, int32_t height, int32_t cubeMapTextureId, int32_t cubeMapTextureIndex) override
Creates a color buffer texture.
void disableCulling(int contextIdx) override
Disable culling.
void bindEffectColorMulsBufferObject(int contextIdx, int32_t bufferObjectId, int32_t divisor) override
Bind effect color muls buffer object.
int32_t createCubeMapTexture(int contextIdx, int32_t width, int32_t height) override
Create cube map texture from frame buffers.
int32_t createFramebufferObject(int32_t depthBufferTextureId, int32_t colorBufferTextureId, int32_t cubeMapTextureId=0, int32_t cubeMapTextureIndex=0) override
Creates a frame buffer object with depth texture attached.
int32_t createDepthBufferTexture(int32_t width, int32_t height, int32_t cubeMapTextureId, int32_t cubeMapTextureIndex) override
Creates a depth buffer texture.
void bindCubeMapTexture(int contextIdx, int32_t textureId) override
Binds a cube map texture with given id or unbinds when using ID_NONE.
void setProgramUniformFloatVec4(int contextIdx, int32_t uniformId, const array< float, 4 > &data) override
Set up a float vec4 uniform value.
void setDepthFunction(int32_t depthFunction) override
Set up depth function.
void uploadCubeMapTexture(int contextIdx, Texture *textureLeft, Texture *textureRight, Texture *textureTop, Texture *textureBottom, Texture *textureFront, Texture *textureBack) override
Uploads cube map texture data to current bound texture.
void bindFrameBuffer(int32_t frameBufferId) override
Enables a framebuffer to be rendered.
int32_t loadShader(int32_t type, const string &pathName, const string &fileName, const string &definitions=string(), const string &functions=string()) override
Loads a shader.
bool isDepthTextureAvailable() override
Checks if depth texture is available.
void resizeDepthBufferTexture(int32_t textureId, int32_t width, int32_t height) override
Resizes a depth texture.
void resizeGBufferGeometryTexture(int32_t textureId, int32_t width, int32_t height) override
Resizes a geometry buffer geometry texture.
bool isInstancedRenderingAvailable() override
Checks if instanced rendering is available.
void drawTrianglesFromBufferObjects(int contextIdx, int32_t triangles, int32_t trianglesOffset) override
Draw triangles from buffer objects.
void initGuiMode() override
Set up renderer for GUI rendering.
void bindSkinningVertexJointsBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind skinning vertex joints buffer object.
void setCullFace(int32_t cullFace) override
Sets up which face will be culled.
void enableBlending() override
Enables blending.
void uploadIndicesBufferObject(int contextIdx, int32_t bufferObjectId, int32_t size, ShortBuffer *data) override
Uploads buffer data to buffer object.
bool linkProgram(int32_t programId) override
Links attached shaders to a program.
void setProgramUniformFloatMatrix4x4(int contextIdx, int32_t uniformId, const array< float, 16 > &data) override
Set up a float matrix 4x4 uniform value.
void enableCulling(int contextIdx) override
Enable culling.
void bindOriginsBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind origins buffer object.
int32_t createProgram(int type) override
Creates a shader program.
void bindIndicesBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind indices buffer object.
void initializeFrame() override
Pre Frame Initialization.
int32_t createGBufferGeometryTexture(int32_t width, int32_t height) override
Creates a geometry buffer geometry texture.
void uploadSkinningBufferObject(int contextIdx, int32_t bufferObjectId, int32_t size, FloatBuffer *data) override
Upload skinning buffer object.
void enableAdditionBlending() override
Enable blending with c = a + b.
void bindVerticesBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind vertices buffer object.
void drawInstancedIndexedTrianglesFromBufferObjects(int contextIdx, int32_t triangles, int32_t trianglesOffset, int32_t instances) override
Draw instanced indexed triangles from buffer objects.
void useProgram(int contextIdx, int32_t programId) override
Use shader program.
void bindColorsBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind colors buffer object.
void bindPointSizesBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind point sizes buffer object.
void bindSkinningVerticesBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind skinning vertices buffer object.
void checkGLError(int line)
Checks if GL error did occour.
void bindBitangentsBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind bitangents buffer object.
void bindSkinningNormalsResultBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind skinning normals result buffer object.
void setProgramUniformFloatMatrices4x4(int contextIdx, int32_t uniformId, int32_t count, FloatBuffer *data) override
Set up a float matrices 4x4 uniform values.
void resizeGBufferColorTexture(int32_t textureId, int32_t width, int32_t height) override
Resizes a geometry buffer color RGBA texture.
void bindSkinningMatricesBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind skinning matrices result buffer object.
int32_t createGeometryBufferObject(int32_t depthBufferTextureId, int32_t geometryBufferTextureId1, int32_t geometryBufferTextureId2, int32_t geometryBufferTextureId3, int32_t colorBufferTextureId1, int32_t colorBufferTextureId2, int32_t colorBufferTextureId3, int32_t colorBufferTextureId4, int32_t colorBufferTextureId5) override
Creates a geometry frame buffer object.
vector< int32_t > createBufferObjects(int32_t buffers, bool useGPUMemory, bool shared) override
Generate buffer objects for vertex data and such.
void updateViewPort() override
Update viewport.
void bindModelMatricesBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind model matrices buffer object.
void drawIndexedTrianglesFromBufferObjects(int contextIdx, int32_t triangles, int32_t trianglesOffset) override
Draw indexed triangles from buffer objects.
void disableDepthBufferTest() override
Disable depth buffer test.
void bindSkinningNormalsBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind skinning normal buffer object.
void unbindBufferObjects(int contextIdx) override
Unbind buffer objects.
int32_t createTexture() override
Creates a texture.
float readPixelDepth(int32_t x, int32_t y) override
Reads a pixel depth.
void uploadTexture(int contextIdx, Texture *texture) override
Uploads texture data to current bound texture.
void setLineWidth(float lineWidth) override
Set line width.
ByteBuffer * readPixels(int32_t x, int32_t y, int32_t width, int32_t height) override
Read pixels.
void enableDepthBufferTest() override
Enable depth buffer test.
void resizeColorBufferTexture(int32_t textureId, int32_t width, int32_t height) override
Resize color buffer texture.
void bindTextureSpriteIndicesBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind texture and sprite indices buffer object.
void setProgramUniformFloatVec2(int contextIdx, int32_t uniformId, const array< float, 2 > &data) override
Set up a float vec2 uniform value.
vector< Renderer_Context > rendererContexts
Definition: Renderer.h:188
virtual void onBindTexture(int contextIdx, int32_t textureId)=0
On bind texture event.
Standard math functions.
Definition: Math.h:21
4x4 3D Matrix class
Definition: Matrix4x4.h:24
File system singleton class.
Definition: FileSystem.h:14
Base class of buffers.
Definition: Buffer.h:20
uint8_t * getBuffer()
Definition: Buffer.h:131
Byte buffer class.
Definition: ByteBuffer.h:24
Console class.
Definition: Console.h:26
Float buffer class.
Definition: FloatBuffer.h:18
Integer buffer class.
Definition: IntBuffer.h:14
Short buffer class.
Definition: ShortBuffer.h:14
String tools class.
Definition: StringTools.h:20
Time utility class.
Definition: Time.h:21