From 660cc3f2611cfe9c366c88d5195b42eed6b824d2 Mon Sep 17 00:00:00 2001 From: rachytski Date: Sun, 2 Dec 2012 15:27:39 +0300 Subject: [PATCH] optimised working with attributes and uniforms. --- base/matrix.hpp | 5 + graphics/defines.cpp | 38 +++ graphics/defines.hpp | 25 +- graphics/graphics.pro | 1 + graphics/opengl/defines_conv.cpp | 78 ++++-- graphics/opengl/defines_conv.hpp | 4 + graphics/opengl/geometry_renderer.cpp | 29 ++- graphics/opengl/program.cpp | 340 +++++++++++++++++++------- graphics/opengl/program.hpp | 55 +++-- graphics/opengl/program_manager.cpp | 36 +-- graphics/opengl/vertex.cpp | 6 +- graphics/vertex_decl.cpp | 8 +- graphics/vertex_decl.hpp | 4 +- 13 files changed, 471 insertions(+), 158 deletions(-) create mode 100644 graphics/defines.cpp diff --git a/base/matrix.hpp b/base/matrix.hpp index 9338493728..f8322187f6 100644 --- a/base/matrix.hpp +++ b/base/matrix.hpp @@ -21,6 +21,11 @@ namespace math m_data[i] = src.m_data[i]; } + Matrix(T * data) + { + copy(data, data + Rows * Cols, m_data); + } + template Matrix const & operator=(Matrix const & src) { diff --git a/graphics/defines.cpp b/graphics/defines.cpp new file mode 100644 index 0000000000..4ffd88ef6a --- /dev/null +++ b/graphics/defines.cpp @@ -0,0 +1,38 @@ +#include "defines.hpp" +#include "../base/macros.hpp" +#include "../base/logging.hpp" +#include "../std/string.hpp" + +namespace graphics +{ + namespace { + struct Data + { + ESemantic m_val; + char const * m_name; + }; + } + + Data s_semantics[] = { + {ESemPosition, "Position"}, + {ESemNormal, "Normal"}, + {ESemTexCoord0, "TexCoord0"}, + {ESemSampler0, "Sampler0"}, + {ESemModelView, "ModelView"}, + {ESemProjection, "Projection"} + }; + + void convert(char const * name, ESemantic & sem) + { + for (unsigned i = 0; i < ARRAY_SIZE(s_semantics); ++i) + { + if (strcmp(name, s_semantics[i].m_name) == 0) + { + sem = s_semantics[i].m_val; + return; + } + } + + LOG(LINFO, ("Unknown Semantics=", name, "specified!")); + } +} diff --git a/graphics/defines.hpp b/graphics/defines.hpp index 3e294696b7..37b9e91a2e 100644 --- a/graphics/defines.hpp +++ b/graphics/defines.hpp @@ -10,11 +10,32 @@ namespace graphics EFragmentShader }; + enum ESemantic + { + ESemPosition, + ESemNormal, + ESemTexCoord0, + ESemSampler0, + ESemModelView, + ESemProjection + }; + + void convert(char const * name, ESemantic & sem); + enum EDataType { + EInteger, + EIntegerVec2, + EIntegerVec3, + EIntegerVec4, EFloat, - EUnsigned, - EInteger + EFloatVec2, + EFloatVec3, + EFloatVec4, + EFloatMat2, + EFloatMat3, + EFloatMat4, + ESampler2D }; enum EPrimitives diff --git a/graphics/graphics.pro b/graphics/graphics.pro index 3295d5f1a3..93d5fef60f 100644 --- a/graphics/graphics.pro +++ b/graphics/graphics.pro @@ -71,6 +71,7 @@ SOURCES += \ render_context.cpp \ coordinates.cpp \ render_target.cpp \ + defines.cpp HEADERS += \ opengl/opengl.hpp \ diff --git a/graphics/opengl/defines_conv.cpp b/graphics/opengl/defines_conv.cpp index 22754364cb..992376502a 100644 --- a/graphics/opengl/defines_conv.cpp +++ b/graphics/opengl/defines_conv.cpp @@ -1,26 +1,74 @@ #include "defines_conv.hpp" +#include "../../base/macros.hpp" namespace graphics { namespace gl { + struct Info + { + int m_firstFull; + int m_firstBase; + char const * m_firstName; + int m_secondFull; + int m_secondBase; + int m_countBase; + char const * m_secondName; + }; + + Info s_dataTypes [] = + { + {GL_FLOAT, GL_FLOAT, "GL_FLOAT", EFloat, EFloat, 1, "EFloatVec1"}, + {GL_FLOAT_VEC2, GL_FLOAT, "GL_FLOAT_VEC2", EFloatVec2, EFloat, 2, "EFloatVec2"}, + {GL_FLOAT_VEC3, GL_FLOAT, "GL_FLOAT_VEC3", EFloatVec3, EFloat, 3, "EFloatVec3"}, + {GL_FLOAT_VEC4, GL_FLOAT, "GL_FLOAT_VEC4", EFloatVec4, EFloat, 4, "EFloatVec4"}, + {GL_INT, GL_INT, "GL_INT", EInteger, EInteger, 1, "EIntegerVec1"}, + {GL_INT_VEC2, GL_INT, "GL_INT_VEC2", EIntegerVec2, EInteger, 2, "EIntegerVec2"}, + {GL_INT_VEC3, GL_INT, "GL_INT_VEC3", EIntegerVec3, EInteger, 3, "EIntegerVec3"}, + {GL_INT_VEC4, GL_INT, "GL_INT_VEC4", EIntegerVec4, EInteger, 4, "EIntegerVec4"}, + {GL_FLOAT_MAT2, GL_FLOAT_MAT2, "GL_FLOAT_MAT2", EFloatMat2, EFloatMat2, 1, "EFloatMat2"}, + {GL_FLOAT_MAT3, GL_FLOAT_MAT3, "GL_FLOAT_MAT3", EFloatMat3, EFloatMat3, 1, "EFloatMat3"}, + {GL_FLOAT_MAT4, GL_FLOAT_MAT4, "GL_FLOAT_MAT4", EFloatMat4, EFloatMat4, 1, "EFloatMat4"}, + {GL_SAMPLER_2D, GL_SAMPLER_2D, "GL_SAMPLER_2D", ESampler2D, ESampler2D, 1, "ESampler2D"} + }; + + void convert(GLenum from, EDataType & to) + { + for (unsigned i = 0; i < ARRAY_SIZE(s_dataTypes); ++i) + if (s_dataTypes[i].m_firstFull == from) + { + to = (EDataType)s_dataTypes[i].m_secondFull; + return; + } + to = (EDataType)0; + LOG(LINFO, ("Unknown GLenum for EDataType specified")); + } + void convert(EDataType from, GLenum & to) { - switch (from) - { - case EFloat: - to = GL_FLOAT; - break; - case EUnsigned: - to = GL_UNSIGNED_INT; - break; - case EInteger: - to = GL_INT; - break; - default: - LOG(LERROR, ("Unknown EDataType specified: ", from)); - to = 0; - } + for (unsigned i = 0; i < ARRAY_SIZE(s_dataTypes); ++i) + if (s_dataTypes[i].m_secondFull == from) + { + to = s_dataTypes[i].m_firstFull; + return; + } + + to = 0; + LOG(LINFO, ("Unknown EDataType specified")); + } + + void convert(GLenum from, EDataType & to, size_t & count) + { + for (unsigned i = 0; i < ARRAY_SIZE(s_dataTypes); ++i) + if (s_dataTypes[i].m_firstFull == from) + { + to = (EDataType)s_dataTypes[i].m_secondBase; + count = s_dataTypes[i].m_countBase; + return; + } + to = (EDataType)0; + count = 0; + LOG(LINFO, ("Unknown GLenum for EDataType specified")); } void convert(EShaderType from, GLenum & to) diff --git a/graphics/opengl/defines_conv.hpp b/graphics/opengl/defines_conv.hpp index 6bc2054b15..497904b038 100644 --- a/graphics/opengl/defines_conv.hpp +++ b/graphics/opengl/defines_conv.hpp @@ -8,6 +8,10 @@ namespace graphics namespace gl { void convert(EDataType from, GLenum & to); + void convert(GLenum from, EDataType & to); + + void convert(GLenum from, EDataType & to, size_t & cnt); + void convert(EShaderType from, GLenum & to); void convert(EPrimitives from, GLenum & to); } diff --git a/graphics/opengl/geometry_renderer.cpp b/graphics/opengl/geometry_renderer.cpp index 287e36f210..8d2f2e8d6d 100644 --- a/graphics/opengl/geometry_renderer.cpp +++ b/graphics/opengl/geometry_renderer.cpp @@ -168,9 +168,10 @@ namespace graphics ProgramManager * pm = rc->programManager(); shared_ptr prg = pm->getProgram("basic", "noalphatest"); - prg->setParam("ModelViewM", rc->matrix(EModelView)); - prg->setParam("ProjM", rc->matrix(EProjection)); - prg->setParam("Texture", 0); + prg->setParam(ESemModelView, rc->matrix(EModelView)); + prg->setParam(ESemProjection, rc->matrix(EProjection)); + prg->setParam(ESemSampler0, 0); + prg->setStorage(blitStorage); prg->setVertexDecl(gl::Vertex::getVertexDecl()); prg->makeCurrent(); @@ -211,6 +212,10 @@ namespace graphics shared_ptr const & prg = rc->program(); + prg->setParam(ESemModelView, rc->matrix(EModelView)); + prg->setParam(ESemProjection, rc->matrix(EProjection)); + prg->setParam(ESemSampler0, 0); + prg->setStorage(m_storage); prg->setVertexDecl(Vertex::getVertexDecl()); prg->makeCurrent(); @@ -407,9 +412,9 @@ namespace graphics ProgramManager * pm = rc->programManager(); shared_ptr prg = pm->getProgram("basic", "alphatest"); - prg->setParam("ModelViewM", rc->matrix(EModelView)); - prg->setParam("ProjM", rc->matrix(EProjection)); - prg->setParam("Texture", 0); + prg->setParam(ESemModelView, rc->matrix(EModelView)); + prg->setParam(ESemProjection, rc->matrix(EProjection)); + prg->setParam(ESemSampler0, 0); rc->setProgram(prg); } @@ -432,9 +437,9 @@ namespace graphics ProgramManager * pm = rc->programManager(); shared_ptr prg = pm->getProgram("basic", "noalphatest"); - prg->setParam("ModelViewM", rc->matrix(EModelView)); - prg->setParam("ProjM", rc->matrix(EProjection)); - prg->setParam("Texture", 0); + prg->setParam(ESemModelView, rc->matrix(EModelView)); + prg->setParam(ESemProjection, rc->matrix(EProjection)); + prg->setParam(ESemSampler0, 0); rc->setProgram(prg); } @@ -477,9 +482,9 @@ namespace graphics ProgramManager * pm = rc->programManager(); shared_ptr prg = pm->getProgram("sharp", "alphatest"); - prg->setParam("ModelViewM", rc->matrix(EModelView)); - prg->setParam("ProjM", rc->matrix(EProjection)); - prg->setParam("Texture", 0); + prg->setParam(ESemModelView, rc->matrix(EModelView)); + prg->setParam(ESemProjection, rc->matrix(EProjection)); + prg->setParam(ESemSampler0, 0); rc->setProgram(prg); } diff --git a/graphics/opengl/program.cpp b/graphics/opengl/program.cpp index 6e33b968e8..98f67bd85e 100644 --- a/graphics/opengl/program.cpp +++ b/graphics/opengl/program.cpp @@ -43,6 +43,50 @@ namespace graphics throw LinkException("Could not link program: ", "Unknown link error"); } + + /// getting all uniforms + int cnt = 0; + + OGLCHECK(glGetProgramivFn(m_handle, GL_ACTIVE_UNIFORMS, &cnt)); + + GLchar name[1024]; + GLsizei len = 0; + GLint size; + GLenum type; + + for (unsigned i = 0; i < cnt; ++i) + { + Uniform f; + ESemantic sem; + + OGLCHECK(glGetActiveUniform(m_handle, i, ARRAY_SIZE(name), &len, &size, &type, name)); + + f.m_handle = glGetUniformLocation(m_handle, name); + OGLCHECKAFTER; + + convert(type, f.m_type); + convert(name, sem); + + m_uniforms[sem] = f; + } + + OGLCHECK(glGetProgramivFn(m_handle, GL_ACTIVE_ATTRIBUTES, &cnt)); + + for (unsigned i = 0; i < cnt; ++i) + { + Attribute a; + ESemantic sem; + + OGLCHECK(glGetActiveAttrib(m_handle, i, ARRAY_SIZE(name), &len, &size, &type, name)); + + a.m_handle = glGetAttribLocation(m_handle, name); + OGLCHECKAFTER; + + convert(type, a.m_type, a.m_count); + convert(name, sem); + + m_attributes[sem] = a; + } } Program::~Program() @@ -50,103 +94,126 @@ namespace graphics OGLCHECK(glDeleteProgramFn(m_handle)); } - GLuint Program::getParam(char const * name) + void Program::setParam(ESemantic sem, float v0) { - GLuint res = glGetUniformLocationFn(m_handle, name); - OGLCHECKAFTER; - return res; + map::iterator it = m_uniforms.find(sem); + ASSERT(it != m_uniforms.end(), ()); + Uniform & f = it->second; + + f.m_type = EFloat; + f.m_data.m_floatVal[0] = v0; } - GLuint Program::getAttribute(char const * name) + void Program::setParam(ESemantic sem, float v0, float v1) { - GLuint res = glGetAttribLocationFn(m_handle, name); - OGLCHECKAFTER; - return res; + map::iterator it = m_uniforms.find(sem); + ASSERT(it != m_uniforms.end(), ()); + Uniform & f = it->second; + + f.m_type = EFloatVec2; + f.m_data.m_floatVal[0] = v0; + f.m_data.m_floatVal[1] = v1; } - void setParamImpl(GLuint prgID, char const * name, function fn) + void Program::setParam(ESemantic sem, float v0, float v1, float v2) { - GLuint res = glGetUniformLocationFn(prgID, name); - OGLCHECKAFTER; - OGLCHECK(fn(res)); + map::iterator it = m_uniforms.find(sem); + ASSERT(it != m_uniforms.end(), ()); + Uniform & f = it->second; + + f.m_type = EFloatVec3; + f.m_data.m_floatVal[0] = v0; + f.m_data.m_floatVal[1] = v1; + f.m_data.m_floatVal[2] = v2; } - void Program::setParam(char const * name, float v0) + void Program::setParam(ESemantic sem, float v0, float v1, float v2, float v3) { - function fn = bind(&glUniform1f, _1, v0); - m_uniforms[name] = bind(&setParamImpl, m_handle, name, fn); + map::iterator it = m_uniforms.find(sem); + ASSERT(it != m_uniforms.end(), ()); + Uniform & f = it->second; + + f.m_type = EFloatVec4; + f.m_data.m_floatVal[0] = v0; + f.m_data.m_floatVal[1] = v1; + f.m_data.m_floatVal[2] = v2; + f.m_data.m_floatVal[3] = v3; } - void Program::setParam(char const * name, float v0, float v1) + void Program::setParam(ESemantic sem, int v0) { - function fn = bind(&glUniform2f, _1, v0, v1); - m_uniforms[name] = bind(&setParamImpl, m_handle, name, fn); + map::iterator it = m_uniforms.find(sem); + ASSERT(it != m_uniforms.end(), ()); + Uniform & f = it->second; + + f.m_type = EInteger; + f.m_data.m_intVal[0] = v0; } - void Program::setParam(char const * name, float v0, float v1, float v2) + void Program::setParam(ESemantic sem, int v0, int v1) { - function fn = bind(&glUniform3f, _1, v0, v1, v2); - m_uniforms[name] = bind(&setParamImpl, m_handle, name, fn); + map::iterator it = m_uniforms.find(sem); + ASSERT(it != m_uniforms.end(), ()); + Uniform & f = it->second; + + f.m_type = EIntegerVec2; + f.m_data.m_intVal[0] = v0; + f.m_data.m_intVal[1] = v1; } - void Program::setParam(char const * name, float v0, float v1, float v2, float v3) + void Program::setParam(ESemantic sem, int v0, int v1, int v2) { - function fn = bind(&glUniform4f, _1, v0, v1, v2, v3); - m_uniforms[name] = bind(&setParamImpl, m_handle, name, fn); + map::iterator it = m_uniforms.find(sem); + ASSERT(it != m_uniforms.end(), ()); + Uniform & f = it->second; + + f.m_type = EIntegerVec3; + f.m_data.m_intVal[0] = v0; + f.m_data.m_intVal[1] = v1; + f.m_data.m_intVal[2] = v2; } - void Program::setParam(char const * name, int v0) + void Program::setParam(ESemantic sem, int v0, int v1, int v2, int v3) { - function fn = bind(&glUniform1i, _1, v0); - m_uniforms[name] = bind(&setParamImpl, m_handle, name, fn); + map::iterator it = m_uniforms.find(sem); + ASSERT(it != m_uniforms.end(), ()); + Uniform & f = it->second; + + f.m_type = EIntegerVec4; + f.m_data.m_intVal[0] = v0; + f.m_data.m_intVal[1] = v1; + f.m_data.m_intVal[2] = v2; + f.m_data.m_intVal[3] = v3; } - void Program::setParam(char const * name, int v0, int v1) + void Program::setParam(ESemantic sem, math::Matrix const & m) { - function fn = bind(&glUniform2i, _1, v0, v1); - m_uniforms[name] = bind(&setParamImpl, m_handle, name, fn); + map::iterator it = m_uniforms.find(sem); + ASSERT(it != m_uniforms.end(), ()); + Uniform & f = it->second; + + f.m_type = EFloatMat2; + copy(&m(0, 0), &m(0, 0) + 2 * 2, f.m_data.m_matVal); } - void Program::setParam(char const * name, int v0, int v1, int v2) + void Program::setParam(ESemantic sem, math::Matrix const & m) { - function fn = bind(&glUniform3i, _1, v0, v1, v2); - m_uniforms[name] = bind(&setParamImpl, m_handle, name, fn); + map::iterator it = m_uniforms.find(sem); + ASSERT(it != m_uniforms.end(), ()); + Uniform & f = it->second; + + f.m_type = EFloatMat3; + copy(&m(0, 0), &m(0, 0) + 3 * 3, f.m_data.m_matVal); } - void Program::setParam(char const * name, int v0, int v1, int v2, int v3) + void Program::setParam(ESemantic sem, math::Matrix const & m) { - function fn = bind(&glUniform4i, _1, v0, v1, v2, v3); - m_uniforms[name] = bind(&setParamImpl, m_handle, name, fn); - } + map::iterator it = m_uniforms.find(sem); + ASSERT(it != m_uniforms.end(), ()); + Uniform & f = it->second; - void Program::setParam(char const * name, math::Matrix const & m) - { - function fn = bind(&glUniformMatrix2fv, _1, 1, 0, &m(0, 0)); - m_uniforms[name] = bind(&setParamImpl, m_handle, name, fn); - } - - void Program::setParam(char const * name, math::Matrix const & m) - { - function fn = bind(&glUniformMatrix3fv, _1, 1, 0, &m(0, 0)); - m_uniforms[name] = bind(&setParamImpl, m_handle, name, fn); - } - - void Program::setParam(char const * name, math::Matrix const & m) - { - function fn = bind(&glUniformMatrix4fv, _1, 1, 0, &m(0, 0)); - m_uniforms[name] = bind(&setParamImpl, m_handle, name, fn); - } - - void enableAndSetVertexAttrib(GLuint id, GLint size, GLenum type, GLboolean normalized, GLsizei stride, void * ptr) - { - OGLCHECK(glEnableVertexAttribArray(id)); - OGLCHECK(glVertexAttribPointer(id, size, type, normalized, stride, ptr)); - } - - void Program::setAttribute(GLuint id, GLint size, GLenum type, GLboolean normalized, GLsizei stride, void *ptr) - { - function fn = bind(&enableAndSetVertexAttrib, id, size, type, normalized, stride, ptr); - m_attributes[id] = fn; + f.m_type = EFloatMat4; + copy(&m(0, 0), &m(0, 0) + 4 * 4, f.m_data.m_matVal); } void Program::setVertexDecl(VertexDecl const * decl) @@ -154,15 +221,23 @@ namespace graphics for (size_t i = 0; i < decl->size(); ++i) { VertexAttrib const * va = decl->getAttr(i); - GLuint attrID = getAttribute(va->m_name.c_str()); - GLenum glType; - convert(va->m_elemType, glType); - setAttribute(attrID, - va->m_elemCount, - glType, - false, - va->m_stride, - (void*)((unsigned char*)m_storage.m_vertices->glPtr() + va->m_offset)); + + map::iterator it = m_attributes.find(va->m_semantic); + ASSERT(it != m_attributes.end(), ()); + Attribute & a = it->second; + + a.m_offset = va->m_offset; + a.m_stride = va->m_stride; + a.m_count = va->m_elemCount; + + /// a.m_count could be different from va->m_elemCount + /// as GLSL could provide missing attribute components + /// with default values according to internal rules. + /// (f.e. all components except 4th in vec4 are made 0 + /// by default, and 4th is 1 by default). + +// ASSERT(a.m_count == va->m_elemCount, ()); + ASSERT(a.m_type == va->m_elemType, ()); } } @@ -171,27 +246,120 @@ namespace graphics m_storage = storage; } + void Program::applyAttributes() + { + /// setting all attributes streams; + for (TAttributes::const_iterator it = m_attributes.begin(); + it != m_attributes.end(); + ++it) + { + Attribute const & a = it->second; + + GLenum t; + convert(a.m_type, t); + + OGLCHECK(glEnableVertexAttribArray(a.m_handle)); + OGLCHECK(glVertexAttribPointer(a.m_handle, + a.m_count, + t, + false, + a.m_stride, + (void*)((unsigned char *)m_storage.m_vertices->glPtr() + a.m_offset))); + } + } + + void Program::applyUniforms() + { + /// setting all uniforms + for (TUniforms::const_iterator it = m_uniforms.begin(); + it != m_uniforms.end(); + ++it) + { + Uniform const & u = it->second; + + switch (u.m_type) + { + case EFloat: + OGLCHECK(glUniform1f(u.m_handle, + u.m_data.m_floatVal[0])); + break; + case EFloatVec2: + OGLCHECK(glUniform2f(u.m_handle, + u.m_data.m_floatVal[0], + u.m_data.m_floatVal[1])); + break; + case EFloatVec3: + OGLCHECK(glUniform3f(u.m_handle, + u.m_data.m_floatVal[0], + u.m_data.m_floatVal[1], + u.m_data.m_floatVal[2])); + break; + case EFloatVec4: + OGLCHECK(glUniform4f(u.m_handle, + u.m_data.m_floatVal[0], + u.m_data.m_floatVal[1], + u.m_data.m_floatVal[2], + u.m_data.m_floatVal[3])); + break; + case EInteger: + OGLCHECK(glUniform1i(u.m_handle, + u.m_data.m_intVal[0])); + break; + case EIntegerVec2: + OGLCHECK(glUniform2i(u.m_handle, + u.m_data.m_intVal[0], + u.m_data.m_intVal[1])); + break; + case EIntegerVec3: + OGLCHECK(glUniform3i(u.m_handle, + u.m_data.m_intVal[0], + u.m_data.m_intVal[1], + u.m_data.m_intVal[2])); + break; + case EIntegerVec4: + OGLCHECK(glUniform4i(u.m_handle, + u.m_data.m_intVal[0], + u.m_data.m_intVal[1], + u.m_data.m_intVal[2], + u.m_data.m_intVal[3])); + break; + case EFloatMat2: + OGLCHECK(glUniformMatrix2fv(u.m_handle, + 1, + false, + u.m_data.m_matVal)); + break; + case EFloatMat3: + OGLCHECK(glUniformMatrix3fv(u.m_handle, + 1, + false, + u.m_data.m_matVal)); + break; + case EFloatMat4: + OGLCHECK(glUniformMatrix4fv(u.m_handle, + 1, + false, + u.m_data.m_matVal)); + break; + case ESampler2D: + OGLCHECK(glUniform1i(u.m_handle, + u.m_data.m_intVal[0])); + break; + } + } + } + void Program::makeCurrent() { OGLCHECK(glUseProgramFn(m_handle)); m_storage.m_vertices->makeCurrent(); - /// setting all attributes streams; - for (TAttributes::const_iterator it = m_attributes.begin(); - it != m_attributes.end(); - ++it) - { - it->second(); - } + applyAttributes(); m_storage.m_indices->makeCurrent(); - /// setting all uniforms - for (TUniforms::const_iterator it = m_uniforms.begin(); - it != m_uniforms.end(); - ++it) - it->second(); + applyUniforms(); } } } diff --git a/graphics/opengl/program.hpp b/graphics/opengl/program.hpp index b6da9a9519..204de6319e 100644 --- a/graphics/opengl/program.hpp +++ b/graphics/opengl/program.hpp @@ -3,6 +3,8 @@ #include "opengl.hpp" #include "storage.hpp" +#include "../defines.hpp" + #include "../../base/matrix.hpp" #include "../../base/exception.hpp" @@ -21,12 +23,36 @@ namespace graphics { private: + void applyAttributes(); + void applyUniforms(); + + struct Uniform + { + struct Data + { + float m_matVal[4 * 4]; + int m_intVal[4]; + float m_floatVal[4]; + } m_data; + EDataType m_type; + GLint m_handle; + }; + + struct Attribute + { + GLint m_handle; + size_t m_offset; + EDataType m_type; + size_t m_count; + size_t m_stride; + }; + GLuint m_handle; - typedef map > TUniforms; + typedef map TUniforms; TUniforms m_uniforms; - typedef map > TAttributes; + typedef map TAttributes; TAttributes m_attributes; Storage m_storage; @@ -43,22 +69,19 @@ namespace graphics unsigned getParam(char const * name); - void setParam(char const * name, float v0); - void setParam(char const * name, float v0, float v1); - void setParam(char const * name, float v0, float v1, float v2); - void setParam(char const * name, float v0, float v1, float v2, float v3); + void setParam(ESemantic sem, float v0); + void setParam(ESemantic sem, float v0, float v1); + void setParam(ESemantic sem, float v0, float v1, float v2); + void setParam(ESemantic sem, float v0, float v1, float v2, float v3); - void setParam(char const * name, int v0); - void setParam(char const * name, int v0, int v1); - void setParam(char const * name, int v0, int v1, int v2); - void setParam(char const * name, int v0, int v1, int v2, int v3); + void setParam(ESemantic sem, int v0); + void setParam(ESemantic sem, int v0, int v1); + void setParam(ESemantic sem, int v0, int v1, int v2); + void setParam(ESemantic sem, int v0, int v1, int v2, int v3); - void setParam(char const * name, math::Matrix const & m); - void setParam(char const * name, math::Matrix const & m); - void setParam(char const * name, math::Matrix const & m); - - GLuint getAttribute(char const * name); - void setAttribute(GLuint id, GLint size, GLenum type, GLboolean normalized, GLsizei stride, void * ptr); + void setParam(ESemantic sem, math::Matrix const & m); + void setParam(ESemantic sem, math::Matrix const & m); + void setParam(ESemantic sem, math::Matrix const & m); void setVertexDecl(VertexDecl const * decl); diff --git a/graphics/opengl/program_manager.cpp b/graphics/opengl/program_manager.cpp index d3f89d1689..7a735cc3b2 100644 --- a/graphics/opengl/program_manager.cpp +++ b/graphics/opengl/program_manager.cpp @@ -17,13 +17,13 @@ namespace graphics static const char vxSrc[] = "attribute vec4 Position;\n" "attribute vec2 Normal;\n" - "attribute vec2 TexCoordIn;\n" - "uniform mat4 ProjM;\n" - "uniform mat4 ModelViewM;\n" - "varying vec2 TexCoordOut;\n" + "attribute vec2 TexCoord0;\n" + "uniform mat4 Projection;\n" + "uniform mat4 ModelView;\n" + "varying vec2 TexCoordOut0;\n" "void main(void) {\n" - " gl_Position = (vec4(Normal, 0.0, 0.0) + Position * ModelViewM) * ProjM;\n" - " TexCoordOut = TexCoordIn;\n" + " gl_Position = (vec4(Normal, 0.0, 0.0) + Position * ModelView) * Projection;\n" + " TexCoordOut0 = TexCoord0;\n" "}\n"; m_vxShaders["basic"].reset(new Shader(vxSrc, EVertexShader)); @@ -33,13 +33,13 @@ namespace graphics static const char sharpVxSrc[] = "attribute vec4 Position;\n" "attribute vec2 Normal;\n" - "attribute vec2 TexCoordIn;\n" - "uniform mat4 ProjM;\n" - "uniform mat4 ModelViewM;\n" - "varying vec2 TexCoordOut;\n" + "attribute vec2 TexCoord0;\n" + "uniform mat4 Projection;\n" + "uniform mat4 ModelView;\n" + "varying vec2 TexCoordOut0;\n" "void main(void) {\n" - " gl_Position = floor(vec4(Normal, 0.0, 0.0) + Position * ModelViewM) * ProjM;\n" - " TexCoordOut = TexCoordIn;\n" + " gl_Position = floor(vec4(Normal, 0.0, 0.0) + Position * ModelView) * Projection;\n" + " TexCoordOut0 = TexCoord0;\n" "}\n"; m_vxShaders["sharp"].reset(new Shader(sharpVxSrc, EVertexShader)); @@ -47,10 +47,10 @@ namespace graphics /// Fragment Shader with alphaTest static const char alphaTestFrgSrc [] = - "uniform sampler2D Texture;\n" - "varying " PRECISION " vec2 TexCoordOut;\n" + "uniform sampler2D Sampler0;\n" + "varying " PRECISION " vec2 TexCoordOut0;\n" "void main(void) {\n" - " gl_FragColor = texture2D(Texture, TexCoordOut);\n" + " gl_FragColor = texture2D(Sampler0, TexCoordOut0);\n" " if (gl_FragColor.a == 0.0)\n" " discard;\n" "}\n"; @@ -60,10 +60,10 @@ namespace graphics /// Fragment shader without alphaTest static const char noAlphaTestFrgSrc[] = - "uniform sampler2D Texture;\n" - "varying " PRECISION " vec2 TexCoordOut;\n" + "uniform sampler2D Sampler0;\n" + "varying " PRECISION " vec2 TexCoordOut0;\n" "void main(void) {\n" - " gl_FragColor = texture2D(Texture, TexCoordOut);\n" + " gl_FragColor = texture2D(Sampler0, TexCoordOut0);\n" "}\n"; m_frgShaders["noalphatest"].reset(new Shader(noAlphaTestFrgSrc, EFragmentShader)); diff --git a/graphics/opengl/vertex.cpp b/graphics/opengl/vertex.cpp index 1ae39866d8..1d58bdc576 100644 --- a/graphics/opengl/vertex.cpp +++ b/graphics/opengl/vertex.cpp @@ -43,9 +43,9 @@ namespace graphics { static VertexAttrib attrs [] = { - VertexAttrib("Position", vertexOffset, EFloat, 3, sizeof(Vertex)), - VertexAttrib("Normal", normalOffset, EFloat, 2, sizeof(Vertex)), - VertexAttrib("TexCoordIn", texCoordOffset, EFloat, 2, sizeof(Vertex)) + VertexAttrib(ESemPosition, vertexOffset, EFloat, 3, sizeof(Vertex)), + VertexAttrib(ESemNormal, normalOffset, EFloat, 2, sizeof(Vertex)), + VertexAttrib(ESemTexCoord0, texCoordOffset, EFloat, 2, sizeof(Vertex)) }; static VertexDecl vd(attrs, ARRAY_SIZE(attrs)); diff --git a/graphics/vertex_decl.cpp b/graphics/vertex_decl.cpp index 3c3a90ee12..6e19264c71 100644 --- a/graphics/vertex_decl.cpp +++ b/graphics/vertex_decl.cpp @@ -2,16 +2,16 @@ namespace graphics { - VertexAttrib::VertexAttrib(char const * name, + VertexAttrib::VertexAttrib(ESemantic semantic, size_t offset, EDataType elemType, size_t elemCount, size_t stride) - : m_offset(offset), + : m_semantic(semantic), + m_offset(offset), m_elemType(elemType), m_elemCount(elemCount), - m_stride(stride), - m_name(name) + m_stride(stride) {} VertexDecl::VertexDecl(VertexAttrib const * attrs, size_t cnt) diff --git a/graphics/vertex_decl.hpp b/graphics/vertex_decl.hpp index 9ed75f527d..375a0b541a 100644 --- a/graphics/vertex_decl.hpp +++ b/graphics/vertex_decl.hpp @@ -10,13 +10,13 @@ namespace graphics /// Single attribute of vertex. struct VertexAttrib { + ESemantic m_semantic; size_t m_offset; EDataType m_elemType; size_t m_elemCount; size_t m_stride; - string m_name; - VertexAttrib(char const * name, + VertexAttrib(ESemantic semantic, size_t offset, EDataType elemType, size_t elemCount,