2015-05-13 10:58:09 +08:00
|
|
|
/**
|
|
|
|
Copyright 2013 BlackBerry Inc.
|
2017-02-14 14:36:57 +08:00
|
|
|
Copyright (c) 2015-2017 Chukong Technologies
|
2018-01-29 16:25:32 +08:00
|
|
|
Copyright (c) 2017-2018 Xiamen Yaji Software Co., Ltd.
|
2015-05-13 10:58:09 +08:00
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
|
|
|
|
Original file from GamePlay3D: http://gameplay3d.org
|
|
|
|
|
|
|
|
This file was modified to fit the cocos2d-x project
|
|
|
|
*/
|
|
|
|
|
2019-01-30 09:35:17 +08:00
|
|
|
#include "renderer/backend/Program.h"
|
|
|
|
#include "renderer/CCPass.h"
|
2015-05-13 10:58:09 +08:00
|
|
|
#include "base/CCConfiguration.h"
|
|
|
|
#include "3d/CCMeshVertexIndexData.h"
|
2019-01-30 09:35:17 +08:00
|
|
|
#include "3d/CC3DProgramInfo.h"
|
2019-02-15 14:30:10 +08:00
|
|
|
#include "3d/CCVertexAttribBinding.h"
|
2015-05-13 10:58:09 +08:00
|
|
|
|
|
|
|
NS_CC_BEGIN
|
|
|
|
|
|
|
|
|
|
|
|
static GLuint __maxVertexAttribs = 0;
|
|
|
|
static std::vector<VertexAttribBinding*> __vertexAttribBindingCache;
|
|
|
|
|
|
|
|
VertexAttribBinding::VertexAttribBinding() :
|
2019-01-30 09:35:17 +08:00
|
|
|
_handle(0), _meshIndexData(nullptr), _programState(nullptr), _attributes()
|
2015-05-13 10:58:09 +08:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
VertexAttribBinding::~VertexAttribBinding()
|
|
|
|
{
|
|
|
|
// Delete from the vertex attribute binding cache.
|
|
|
|
std::vector<VertexAttribBinding*>::iterator itr = std::find(__vertexAttribBindingCache.begin(), __vertexAttribBindingCache.end(), this);
|
|
|
|
if (itr != __vertexAttribBindingCache.end())
|
|
|
|
{
|
|
|
|
__vertexAttribBindingCache.erase(itr);
|
|
|
|
}
|
|
|
|
|
|
|
|
CC_SAFE_RELEASE(_meshIndexData);
|
2019-01-30 09:35:17 +08:00
|
|
|
CC_SAFE_RELEASE(_programState);
|
2015-05-13 10:58:09 +08:00
|
|
|
_attributes.clear();
|
|
|
|
|
|
|
|
if (_handle)
|
|
|
|
{
|
|
|
|
glDeleteVertexArrays(1, &_handle);
|
|
|
|
_handle = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-30 09:35:17 +08:00
|
|
|
VertexAttribBinding* VertexAttribBinding::create(MeshIndexData* meshIndexData, Pass* pass)
|
2015-05-13 10:58:09 +08:00
|
|
|
{
|
2019-01-30 09:35:17 +08:00
|
|
|
CCASSERT(meshIndexData && pass && pass->getProgramState(), "Invalid MeshIndexData and/or programState");
|
2015-05-13 10:58:09 +08:00
|
|
|
|
|
|
|
// Search for an existing vertex attribute binding that can be used.
|
|
|
|
VertexAttribBinding* b;
|
|
|
|
for (size_t i = 0, count = __vertexAttribBindingCache.size(); i < count; ++i)
|
|
|
|
{
|
|
|
|
b = __vertexAttribBindingCache[i];
|
|
|
|
CC_ASSERT(b);
|
2019-01-30 09:35:17 +08:00
|
|
|
if (b->_meshIndexData == meshIndexData && b->_programState == pass->getProgramState())
|
2015-05-13 10:58:09 +08:00
|
|
|
{
|
|
|
|
// Found a match!
|
|
|
|
return b;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
b = new (std::nothrow) VertexAttribBinding();
|
2019-01-30 09:35:17 +08:00
|
|
|
if (b && b->init(meshIndexData, pass))
|
2015-05-13 10:58:09 +08:00
|
|
|
{
|
|
|
|
b->autorelease();
|
|
|
|
__vertexAttribBindingCache.push_back(b);
|
|
|
|
}
|
|
|
|
|
|
|
|
return b;
|
|
|
|
}
|
|
|
|
|
2019-01-30 09:35:17 +08:00
|
|
|
bool VertexAttribBinding::init(MeshIndexData* meshIndexData, Pass* pass)
|
2015-05-13 10:58:09 +08:00
|
|
|
{
|
|
|
|
|
2019-01-30 09:35:17 +08:00
|
|
|
CCASSERT(meshIndexData && pass && pass->getProgramState(), "Invalid arguments");
|
2015-05-13 10:58:09 +08:00
|
|
|
|
2019-01-30 09:35:17 +08:00
|
|
|
auto programState = pass->getProgramState();
|
|
|
|
|
|
|
|
_vertexLayout = pass->getVertexLayout();
|
|
|
|
|
|
|
|
// One-time initialization.
|
|
|
|
//TODO arnold
|
|
|
|
// if (__maxVertexAttribs == 0)
|
|
|
|
// {
|
|
|
|
// GLint temp;
|
|
|
|
// glGetIntegerv(GL_MAX_VERTEX_ATTRIBS, &temp);
|
|
|
|
//
|
|
|
|
// __maxVertexAttribs = temp;
|
|
|
|
// if (__maxVertexAttribs <= 0)
|
|
|
|
// {
|
|
|
|
// CCLOGERROR("The maximum number of vertex attributes supported by OpenGL on the current device is 0 or less.");
|
|
|
|
// return false;
|
|
|
|
// }
|
|
|
|
// }
|
2015-05-13 10:58:09 +08:00
|
|
|
|
|
|
|
_meshIndexData = meshIndexData;
|
|
|
|
_meshIndexData->retain();
|
2019-01-30 09:35:17 +08:00
|
|
|
_programState = programState;
|
|
|
|
_programState->retain();
|
2015-05-13 10:58:09 +08:00
|
|
|
|
|
|
|
auto meshVertexData = meshIndexData->getMeshVertexData();
|
|
|
|
auto attributeCount = meshVertexData->getMeshVertexAttribCount();
|
|
|
|
|
|
|
|
|
|
|
|
// Parse and set attributes
|
|
|
|
parseAttributes();
|
2019-01-30 09:35:17 +08:00
|
|
|
int offset = 0;
|
2015-05-13 10:58:09 +08:00
|
|
|
for (auto k = 0; k < attributeCount; k++)
|
|
|
|
{
|
|
|
|
auto meshattribute = meshVertexData->getMeshVertexAttrib(k);
|
|
|
|
setVertexAttribPointer(
|
2019-01-30 09:35:17 +08:00
|
|
|
shaderinfos::getAttributeName(meshattribute.vertexAttrib),
|
2015-05-13 10:58:09 +08:00
|
|
|
meshattribute.type,
|
|
|
|
GL_FALSE,
|
2019-02-15 14:30:10 +08:00
|
|
|
offset,
|
|
|
|
1 << k //FIXME: wrong flag
|
|
|
|
);
|
2019-01-30 09:35:17 +08:00
|
|
|
offset += meshattribute.getAttribSizeBytes();
|
2015-05-13 10:58:09 +08:00
|
|
|
}
|
|
|
|
|
2019-01-30 09:35:17 +08:00
|
|
|
_vertexLayout->setLayout(offset, backend::VertexStepMode::VERTEX);
|
2015-05-13 10:58:09 +08:00
|
|
|
|
2019-01-30 09:35:17 +08:00
|
|
|
CCASSERT(offset == meshVertexData->getSizePerVertex(), "vertex layout mismatch!");
|
2015-05-13 10:58:09 +08:00
|
|
|
|
2019-01-30 09:35:17 +08:00
|
|
|
//// VAO hardware
|
|
|
|
//if (Configuration::getInstance()->supportsShareableVAO())
|
|
|
|
//{
|
|
|
|
// glGenVertexArrays(1, &_handle);
|
|
|
|
// glBindVertexArray(_handle);
|
|
|
|
// glBindBuffer(GL_ARRAY_BUFFER, meshVertexData->getVertexBuffer()->getVBO());
|
2015-05-13 10:58:09 +08:00
|
|
|
|
2019-01-30 09:35:17 +08:00
|
|
|
// enableVertexAttributes(_vertexAttribsFlags);
|
2015-05-13 10:58:09 +08:00
|
|
|
|
2019-01-30 09:35:17 +08:00
|
|
|
// glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, meshIndexData->getIndexBuffer()->getVBO());
|
2015-05-13 10:58:09 +08:00
|
|
|
|
2019-01-30 09:35:17 +08:00
|
|
|
// for(auto &attribute : _attributes)
|
|
|
|
// {
|
|
|
|
// attribute.second.apply();
|
|
|
|
// }
|
2015-05-13 10:58:09 +08:00
|
|
|
|
2019-01-30 09:35:17 +08:00
|
|
|
// glBindVertexArray(0);
|
|
|
|
// glBindBuffer(GL_ARRAY_BUFFER, 0);
|
|
|
|
// glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, 0);
|
|
|
|
//}
|
2015-05-13 10:58:09 +08:00
|
|
|
|
2019-01-30 09:35:17 +08:00
|
|
|
return true;
|
2015-05-13 10:58:09 +08:00
|
|
|
}
|
|
|
|
|
2019-01-30 09:35:17 +08:00
|
|
|
//void VertexAttribBinding::bind(backend::VertexLayout &layout)
|
|
|
|
//{
|
|
|
|
//
|
|
|
|
// if (_handle)
|
|
|
|
// {
|
|
|
|
// // hardware
|
|
|
|
// glBindVertexArray(_handle);
|
|
|
|
// }
|
|
|
|
// else
|
|
|
|
// {
|
|
|
|
// // software
|
|
|
|
// auto meshVertexData = _meshIndexData->getMeshVertexData();
|
|
|
|
// glBindBuffer(GL_ARRAY_BUFFER, meshVertexData->getVertexBuffer()->getVBO());
|
|
|
|
// glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, _meshIndexData->getIndexBuffer()->getVBO());
|
|
|
|
//
|
|
|
|
// // Software mode
|
|
|
|
// enableVertexAttributes(_vertexAttribsFlags);
|
|
|
|
// // set attributes
|
|
|
|
// for(auto &attribute : _attributes)
|
|
|
|
// {
|
|
|
|
// attribute.second.apply();
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
//}
|
|
|
|
|
|
|
|
//void VertexAttribBinding::unbind()
|
|
|
|
//{
|
|
|
|
// if (_handle)
|
|
|
|
// {
|
|
|
|
// // Hardware
|
|
|
|
// glBindVertexArray(0);
|
|
|
|
// }
|
|
|
|
// else
|
|
|
|
// {
|
|
|
|
// // Software
|
|
|
|
// glBindBuffer(GL_ARRAY_BUFFER, 0);
|
|
|
|
// glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, 0);
|
|
|
|
// }
|
|
|
|
//}
|
2015-05-13 10:58:09 +08:00
|
|
|
|
|
|
|
uint32_t VertexAttribBinding::getVertexAttribsFlags() const
|
|
|
|
{
|
|
|
|
return _vertexAttribsFlags;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VertexAttribBinding::parseAttributes()
|
|
|
|
{
|
2019-01-30 09:35:17 +08:00
|
|
|
CCASSERT(_programState, "invalid glprogram");
|
2015-05-13 10:58:09 +08:00
|
|
|
|
|
|
|
_attributes.clear();
|
|
|
|
_vertexAttribsFlags = 0;
|
|
|
|
|
2019-01-30 09:35:17 +08:00
|
|
|
auto program = _programState->getProgram();
|
|
|
|
|
|
|
|
auto attributes = program->getActiveAttributes();
|
2015-05-13 10:58:09 +08:00
|
|
|
|
2019-01-30 09:35:17 +08:00
|
|
|
for(auto &attrib: attributes)
|
2018-09-11 14:39:30 +08:00
|
|
|
{
|
2019-01-30 09:35:17 +08:00
|
|
|
_attributes[attrib.attributeName] = attrib;
|
2018-09-11 14:39:30 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-15 14:30:10 +08:00
|
|
|
bool VertexAttribBinding::hasAttribute(const shaderinfos::VertexKey &key) const
|
|
|
|
{
|
|
|
|
auto &name = shaderinfos::getAttributeName(key);
|
|
|
|
return _attributes.find(name) != _attributes.end();
|
|
|
|
}
|
|
|
|
|
2019-01-30 09:35:17 +08:00
|
|
|
//void VertexAttribBinding::enableVertexAttributes(uint32_t flags) const
|
|
|
|
//{
|
|
|
|
// auto tmpFlags = flags;
|
|
|
|
// for (int i = 0; tmpFlags > 0; i++)
|
|
|
|
// {
|
|
|
|
// int flag = 1 << i;
|
|
|
|
// if (flag & tmpFlags)
|
|
|
|
// glEnableVertexAttribArray(i);
|
|
|
|
//
|
|
|
|
// tmpFlags &= ~flag;
|
|
|
|
// }
|
|
|
|
//}
|
|
|
|
|
|
|
|
backend::AttributeBindInfo* VertexAttribBinding::getVertexAttribValue(const std::string& name)
|
2015-05-13 10:58:09 +08:00
|
|
|
{
|
|
|
|
const auto itr = _attributes.find(name);
|
|
|
|
if( itr != _attributes.end())
|
|
|
|
return &itr->second;
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2019-02-15 14:30:10 +08:00
|
|
|
void VertexAttribBinding::setVertexAttribPointer(const std::string &name, backend::VertexFormat type, GLboolean normalized, int offset, int flag)
|
2015-05-13 10:58:09 +08:00
|
|
|
{
|
|
|
|
auto v = getVertexAttribValue(name);
|
|
|
|
if(v) {
|
2019-02-15 14:30:10 +08:00
|
|
|
// CCLOG("cocos2d: set attribute '%s' location: %d, offset: %d", name.c_str(), v->location, offset);
|
2019-01-30 09:35:17 +08:00
|
|
|
_vertexLayout->setAtrribute(name, v->location, type, offset, normalized);
|
2019-02-15 14:30:10 +08:00
|
|
|
_vertexAttribsFlags |= flag;
|
2015-05-13 10:58:09 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
CCLOG("cocos2d: warning: Attribute not found: %s", name.c_str());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_CC_END
|