Initial multitexturing support

This commit is contained in:
Ricardo Quesada 2014-05-12 23:51:37 -07:00
parent a3aed414fa
commit 71848aca93
6 changed files with 76 additions and 32 deletions

View File

@ -328,7 +328,7 @@ void GLProgram::parseUniforms()
glGetActiveUniform(_program, i, length, NULL, &uniform.size, &uniform.type, uniformName); glGetActiveUniform(_program, i, length, NULL, &uniform.size, &uniform.type, uniformName);
uniformName[length] = '\0'; uniformName[length] = '\0';
// Only add uniforms that are not build-in. // Only add uniforms that are not built-in.
// The ones that start with 'CC_' are built-ins // The ones that start with 'CC_' are built-ins
if(strncmp("CC_", uniformName, 3) != 0) { if(strncmp("CC_", uniformName, 3) != 0) {

View File

@ -31,6 +31,7 @@ THE SOFTWARE.
#include "renderer/CCGLProgramStateCache.h" #include "renderer/CCGLProgramStateCache.h"
#include "renderer/CCGLProgramCache.h" #include "renderer/CCGLProgramCache.h"
#include "renderer/ccGLStateCache.h" #include "renderer/ccGLStateCache.h"
#include "2d/CCTexture2D.h"
NS_CC_BEGIN NS_CC_BEGIN
@ -68,15 +69,20 @@ void UniformValue::apply()
else else
{ {
switch (_uniform->type) { switch (_uniform->type) {
case GL_FLOAT: case GL_SAMPLER_2D:
_glprogram->setUniformLocationWith1f(_uniform->location, _value.floatValue); _glprogram->setUniformLocationWith1i(_uniform->location, _value.tex.textureUnit);
GL::activeTexture(_value.tex.textureUnit);
GL::bindTexture2DN(_value.tex.textureUnit, _value.tex.textureId);
break; break;
case GL_INT: case GL_INT:
case GL_SAMPLER_2D:
_glprogram->setUniformLocationWith1i(_uniform->location, _value.intValue); _glprogram->setUniformLocationWith1i(_uniform->location, _value.intValue);
break; break;
case GL_FLOAT:
_glprogram->setUniformLocationWith1f(_uniform->location, _value.floatValue);
break;
case GL_FLOAT_VEC2: case GL_FLOAT_VEC2:
_glprogram->setUniformLocationWith2f(_uniform->location, _value.v2Value[0], _value.v2Value[1]); _glprogram->setUniformLocationWith2f(_uniform->location, _value.v2Value[0], _value.v2Value[1]);
break; break;
@ -122,9 +128,16 @@ void UniformValue::setFloat(float value)
_useCallback = false; _useCallback = false;
} }
void UniformValue::setTexture(GLuint textureId, GLuint textureUnit)
{
CCASSERT(_uniform->type == GL_SAMPLER_2D, "Wrong type. expecting GL_SAMPLER_2D");
_value.tex.textureId = textureId;
_value.tex.textureUnit = textureUnit;
_useCallback = false;
}
void UniformValue::setInt(int value) void UniformValue::setInt(int value)
{ {
CCASSERT ((_uniform->type == GL_INT || _uniform->type == GL_SAMPLER_2D), ""); CCASSERT(_uniform->type == GL_INT, "Wrong type: expecting GL_INT");
_value.intValue = value; _value.intValue = value;
_useCallback = false; _useCallback = false;
} }
@ -255,6 +268,7 @@ GLProgramState* GLProgramState::getOrCreate(GLProgram *glprogram)
GLProgramState::GLProgramState() GLProgramState::GLProgramState()
: _vertexAttribsFlags(0) : _vertexAttribsFlags(0)
, _glprogram(nullptr) , _glprogram(nullptr)
, _textureUnitIndex(1)
{ {
} }
@ -288,6 +302,8 @@ void GLProgramState::resetGLProgram()
CC_SAFE_RELEASE(_glprogram); CC_SAFE_RELEASE(_glprogram);
_uniforms.clear(); _uniforms.clear();
_attributes.clear(); _attributes.clear();
// first texture is GL_TEXTURE1
_textureUnitIndex = 1;
} }
void GLProgramState::apply(const Matrix& modelView) void GLProgramState::apply(const Matrix& modelView)
@ -296,12 +312,10 @@ void GLProgramState::apply(const Matrix& modelView)
// set shader // set shader
_glprogram->use(); _glprogram->use();
_glprogram->setUniformsForBuiltins(modelView); _glprogram->setUniformsForBuiltins(modelView);
// Don't set attributes if they weren't set
// quick hack: // Use Case: Auto-batching
// Don't deal with attributes if they were set
if(_vertexAttribsFlags) { if(_vertexAttribsFlags) {
// enable/disable vertex attribs // enable/disable vertex attribs
GL::enableVertexAttribs(_vertexAttribsFlags); GL::enableVertexAttribs(_vertexAttribsFlags);
@ -439,4 +453,22 @@ void GLProgramState::setUniformMat4(const std::string &uniformName, const Matrix
CCLOG("cocos2d: warning: Uniform not found: %s", uniformName.c_str()); CCLOG("cocos2d: warning: Uniform not found: %s", uniformName.c_str());
} }
// Textures
void GLProgramState::setUniformTexture(const std::string &uniformName, Texture2D *texture)
{
CCASSERT(texture, "Invalid texture");
setUniformTexture(uniformName, texture->getName());
}
void GLProgramState::setUniformTexture(const std::string &uniformName, GLuint textureId)
{
auto v = getUniformValue(uniformName);
if (v)
v->setTexture(textureId, _textureUnitIndex++);
else
CCLOG("cocos2d: warning: Uniform not found: %s", uniformName.c_str());
}
NS_CC_END NS_CC_END

View File

@ -32,10 +32,13 @@ THE SOFTWARE.
#include "math/Vector4.h" #include "math/Vector4.h"
#include <unordered_map> #include <unordered_map>
#include <vector>
NS_CC_BEGIN NS_CC_BEGIN
class GLProgram; class GLProgram;
class Texture2D;
struct Uniform; struct Uniform;
struct VertexAttrib; struct VertexAttrib;
@ -60,6 +63,7 @@ public:
void setVec4(const Vector4& value); void setVec4(const Vector4& value);
void setMat4(const Matrix& value); void setMat4(const Matrix& value);
void setCallback(const std::function<void(Uniform*)> &callback); void setCallback(const std::function<void(Uniform*)> &callback);
void setTexture(GLuint textureId, GLuint activeTexture);
void apply(); void apply();
@ -75,6 +79,10 @@ protected:
float v3Value[3]; float v3Value[3];
float v4Value[4]; float v4Value[4];
float matrixValue[16]; float matrixValue[16];
struct {
GLuint textureId;
GLuint textureUnit;
} tex;
std::function<void(Uniform*)> *callback; std::function<void(Uniform*)> *callback;
U() { memset( this, 0, sizeof(*this) ); } U() { memset( this, 0, sizeof(*this) ); }
@ -154,20 +162,23 @@ public:
void setGLProgram(GLProgram* glprogram); void setGLProgram(GLProgram* glprogram);
GLProgram* getGLProgram() const { return _glprogram; } GLProgram* getGLProgram() const { return _glprogram; }
// vertex attribs
uint32_t getVertexAttribsFlags() const { return _vertexAttribsFlags; } uint32_t getVertexAttribsFlags() const { return _vertexAttribsFlags; }
ssize_t getVertexAttribCount() const { return _attributes.size(); } ssize_t getVertexAttribCount() const { return _attributes.size(); }
void setVertexAttribCallback(const std::string &name, const std::function<void(VertexAttrib*)> &callback); void setVertexAttribCallback(const std::string &name, const std::function<void(VertexAttrib*)> &callback);
void setVertexAttribPointer(const std::string &name, GLint size, GLenum type, GLboolean normalized, GLsizei stride, GLvoid *pointer); void setVertexAttribPointer(const std::string &name, GLint size, GLenum type, GLboolean normalized, GLsizei stride, GLvoid *pointer);
// user defined uniforms
ssize_t getUniformCount() const { return _uniforms.size(); } ssize_t getUniformCount() const { return _uniforms.size(); }
void setUniformFloat(const std::string &uniformName, float value);
void setUniformInt(const std::string &uniformName, int value); void setUniformInt(const std::string &uniformName, int value);
void setUniformFloat(const std::string &uniformName, float value);
void setUniformVec2(const std::string &uniformName, const Vector2& value); void setUniformVec2(const std::string &uniformName, const Vector2& value);
void setUniformVec3(const std::string &uniformName, const Vector3& value); void setUniformVec3(const std::string &uniformName, const Vector3& value);
void setUniformVec4(const std::string &uniformName, const Vector4& value); void setUniformVec4(const std::string &uniformName, const Vector4& value);
void setUniformMat4(const std::string &uniformName, const Matrix& value); void setUniformMat4(const std::string &uniformName, const Matrix& value);
void setUniformCallback(const std::string &uniformName, const std::function<void(Uniform*)> &callback); void setUniformCallback(const std::string &uniformName, const std::function<void(Uniform*)> &callback);
void setUniformTexture(const std::string &uniformName, Texture2D *texture);
void setUniformTexture(const std::string &uniformName, GLuint textureId);
protected: protected:
GLProgramState(); GLProgramState();
@ -180,6 +191,7 @@ protected:
std::unordered_map<std::string, UniformValue> _uniforms; std::unordered_map<std::string, UniformValue> _uniforms;
std::unordered_map<std::string, VertexAttribValue> _attributes; std::unordered_map<std::string, VertexAttribValue> _attributes;
int _textureUnitIndex;
uint32_t _vertexAttribsFlags; uint32_t _vertexAttribsFlags;
GLProgram *_glprogram; GLProgram *_glprogram;
}; };

View File

@ -74,7 +74,7 @@ void QuadCommand::generateMaterialID()
if(_glProgramState->getUniformCount() > 0) if(_glProgramState->getUniformCount() > 0)
{ {
_materialID = QuadCommand::MATERIAL_ID_IGNORE; _materialID = QuadCommand::MATERIAL_ID_DO_NOT_BATCH;
} }
else else
{ {

View File

@ -35,7 +35,7 @@ NS_CC_BEGIN
class QuadCommand : public RenderCommand class QuadCommand : public RenderCommand
{ {
public: public:
static const int MATERIAL_ID_IGNORE = 0; static const int MATERIAL_ID_DO_NOT_BATCH = 0;
QuadCommand(); QuadCommand();
~QuadCommand(); ~QuadCommand();

View File

@ -453,7 +453,7 @@ void Renderer::drawBatchedQuads()
for(const auto& cmd : _batchedQuadCommands) for(const auto& cmd : _batchedQuadCommands)
{ {
auto newMaterialID = cmd->getMaterialID(); auto newMaterialID = cmd->getMaterialID();
if(_lastMaterialID != newMaterialID || newMaterialID == QuadCommand::MATERIAL_ID_IGNORE) if(_lastMaterialID != newMaterialID || newMaterialID == QuadCommand::MATERIAL_ID_DO_NOT_BATCH)
{ {
//Draw quads //Draw quads
if(quadsToDraw > 0) if(quadsToDraw > 0)