2013-03-08 01:50:15 +00:00
|
|
|
//
|
|
|
|
// Copyright (C) Pixar. All rights reserved.
|
|
|
|
//
|
|
|
|
// This license governs use of the accompanying software. If you
|
|
|
|
// use the software, you accept this license. If you do not accept
|
|
|
|
// the license, do not use the software.
|
|
|
|
//
|
|
|
|
// 1. Definitions
|
|
|
|
// The terms "reproduce," "reproduction," "derivative works," and
|
|
|
|
// "distribution" have the same meaning here as under U.S.
|
|
|
|
// copyright law. A "contribution" is the original software, or
|
|
|
|
// any additions or changes to the software.
|
|
|
|
// A "contributor" is any person or entity that distributes its
|
|
|
|
// contribution under this license.
|
|
|
|
// "Licensed patents" are a contributor's patent claims that read
|
|
|
|
// directly on its contribution.
|
|
|
|
//
|
|
|
|
// 2. Grant of Rights
|
|
|
|
// (A) Copyright Grant- Subject to the terms of this license,
|
|
|
|
// including the license conditions and limitations in section 3,
|
|
|
|
// each contributor grants you a non-exclusive, worldwide,
|
|
|
|
// royalty-free copyright license to reproduce its contribution,
|
|
|
|
// prepare derivative works of its contribution, and distribute
|
|
|
|
// its contribution or any derivative works that you create.
|
|
|
|
// (B) Patent Grant- Subject to the terms of this license,
|
|
|
|
// including the license conditions and limitations in section 3,
|
|
|
|
// each contributor grants you a non-exclusive, worldwide,
|
|
|
|
// royalty-free license under its licensed patents to make, have
|
|
|
|
// made, use, sell, offer for sale, import, and/or otherwise
|
|
|
|
// dispose of its contribution in the software or derivative works
|
|
|
|
// of the contribution in the software.
|
|
|
|
//
|
|
|
|
// 3. Conditions and Limitations
|
|
|
|
// (A) No Trademark License- This license does not grant you
|
|
|
|
// rights to use any contributor's name, logo, or trademarks.
|
|
|
|
// (B) If you bring a patent claim against any contributor over
|
|
|
|
// patents that you claim are infringed by the software, your
|
|
|
|
// patent license from such contributor to the software ends
|
|
|
|
// automatically.
|
|
|
|
// (C) If you distribute any portion of the software, you must
|
|
|
|
// retain all copyright, patent, trademark, and attribution
|
|
|
|
// notices that are present in the software.
|
|
|
|
// (D) If you distribute any portion of the software in source
|
|
|
|
// code form, you may do so only under this license by including a
|
|
|
|
// complete copy of this license with your distribution. If you
|
|
|
|
// distribute any portion of the software in compiled or object
|
|
|
|
// code form, you may only do so under a license that complies
|
|
|
|
// with this license.
|
|
|
|
// (E) The software is licensed "as-is." You bear the risk of
|
|
|
|
// using it. The contributors give no express warranties,
|
|
|
|
// guarantees or conditions. You may have additional consumer
|
|
|
|
// rights under your local laws which this license cannot change.
|
|
|
|
// To the extent permitted under your local laws, the contributors
|
|
|
|
// exclude the implied warranties of merchantability, fitness for
|
|
|
|
// a particular purpose and non-infringement.
|
|
|
|
//
|
|
|
|
|
|
|
|
#ifndef FAR_MULTI_MESH_FACTORY_H
|
|
|
|
#define FAR_MULTI_MESH_FACTORY_H
|
|
|
|
|
|
|
|
#include "../version.h"
|
|
|
|
|
|
|
|
#include "../far/mesh.h"
|
|
|
|
#include "../far/bilinearSubdivisionTablesFactory.h"
|
|
|
|
#include "../far/catmarkSubdivisionTablesFactory.h"
|
|
|
|
#include "../far/loopSubdivisionTablesFactory.h"
|
|
|
|
#include "../far/patchTablesFactory.h"
|
|
|
|
#include "../far/vertexEditTablesFactory.h"
|
|
|
|
|
|
|
|
#include <typeinfo>
|
|
|
|
|
|
|
|
namespace OpenSubdiv {
|
|
|
|
namespace OPENSUBDIV_VERSION {
|
|
|
|
|
2013-05-07 02:05:50 +00:00
|
|
|
/// \brief A specialized factory for batching meshes
|
|
|
|
///
|
|
|
|
/// Because meshes require multiple draw calls in order to process the different
|
|
|
|
/// types of patches, it is useful to have the ability of grouping the tables of
|
|
|
|
/// multiple meshes into a single set of tables. This factory builds upon the
|
|
|
|
/// specialized Far factories in order to provide this batching functionality.
|
|
|
|
///
|
|
|
|
template <class T, class U=T> class FarMultiMeshFactory {
|
2013-03-08 01:50:15 +00:00
|
|
|
|
|
|
|
public:
|
2013-05-07 02:05:50 +00:00
|
|
|
|
|
|
|
typedef std::vector<FarMesh<U> const *> FarMeshVector;
|
|
|
|
|
|
|
|
/// Constructor.
|
2013-03-08 01:50:15 +00:00
|
|
|
FarMultiMeshFactory() {}
|
2013-05-07 02:05:50 +00:00
|
|
|
|
|
|
|
/// Splices a vector of Far meshes into a single Far mesh
|
|
|
|
///
|
|
|
|
/// @param meshes a vector of Far meshes to splice
|
|
|
|
///
|
|
|
|
/// @return the resulting spliced Far mesh
|
|
|
|
///
|
2013-03-08 01:50:15 +00:00
|
|
|
FarMesh<U> * Create(std::vector<FarMesh<U> const *> const &meshes);
|
|
|
|
|
2013-05-10 02:16:51 +00:00
|
|
|
std::vector<FarPatchTables::PatchArrayVector> const & GetMultiPatchArrays() {
|
|
|
|
return _multiPatchArrays;
|
|
|
|
}
|
|
|
|
|
2013-03-08 01:50:15 +00:00
|
|
|
private:
|
2013-05-07 02:05:50 +00:00
|
|
|
|
2013-03-08 01:50:15 +00:00
|
|
|
// splice subdivision tables
|
|
|
|
FarSubdivisionTables<U> * spliceSubdivisionTables(FarMesh<U> *farmesh, FarMeshVector const &meshes);
|
|
|
|
|
|
|
|
// splice patch tables
|
|
|
|
FarPatchTables * splicePatchTables(FarMeshVector const &meshes);
|
|
|
|
|
2013-05-10 02:16:51 +00:00
|
|
|
// splice patch array
|
|
|
|
FarPatchTables::PTable::iterator splicePatch(FarPatchTables::Descriptor desc,
|
|
|
|
FarMeshVector const &meshes,
|
|
|
|
FarPatchTables::PatchArrayVector &result,
|
|
|
|
FarPatchTables::PTable::iterator dstIndexIt,
|
|
|
|
int *voffset, int *poffset, int *qoffset,
|
|
|
|
std::vector<int> const &vertexOffsets);
|
|
|
|
|
2013-03-08 01:50:15 +00:00
|
|
|
// splice hierarchical edit tables
|
|
|
|
FarVertexEditTables<U> * spliceVertexEditTables(FarMesh<U> *farmesh, FarMeshVector const &meshes);
|
|
|
|
|
|
|
|
int _maxlevel;
|
|
|
|
int _maxvalence;
|
2013-05-10 02:16:51 +00:00
|
|
|
|
|
|
|
// patch arrays for each mesh
|
|
|
|
std::vector<FarPatchTables::PatchArrayVector> _multiPatchArrays;
|
2013-03-08 01:50:15 +00:00
|
|
|
};
|
|
|
|
|
2013-05-07 02:05:50 +00:00
|
|
|
|
2013-03-08 01:50:15 +00:00
|
|
|
template <class T, class U> FarMesh<U> *
|
|
|
|
FarMultiMeshFactory<T, U>::Create(std::vector<FarMesh<U> const *> const &meshes) {
|
|
|
|
|
|
|
|
if (meshes.empty()) return NULL;
|
|
|
|
|
|
|
|
bool adaptive = (meshes[0]->GetPatchTables() != NULL);
|
|
|
|
const std::type_info &scheme = typeid(*(meshes[0]->GetSubdivisionTables()));
|
|
|
|
_maxlevel = 0;
|
|
|
|
_maxvalence = 0;
|
|
|
|
|
|
|
|
for (size_t i = 0; i < meshes.size(); ++i) {
|
|
|
|
FarMesh<U> const *mesh = meshes[i];
|
|
|
|
// XXX: once uniform quads are integrated into patch tables,
|
|
|
|
// this restriction can be relaxed so that we can merge adaptive and uniform meshes together.
|
|
|
|
if (adaptive ^ (mesh->GetPatchTables() != NULL)) {
|
|
|
|
assert(false);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
// meshes have to have a same subdivision scheme
|
|
|
|
if (scheme != typeid(*(mesh->GetSubdivisionTables()))) {
|
|
|
|
assert(false);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
_maxlevel = std::max(_maxlevel, mesh->GetSubdivisionTables()->GetMaxLevel()-1);
|
|
|
|
if (mesh->GetPatchTables()) {
|
|
|
|
_maxvalence = std::max(_maxvalence, mesh->GetPatchTables()->GetMaxValence());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
FarMesh<U> * result = new FarMesh<U>();
|
|
|
|
|
|
|
|
// splice subdivision tables
|
|
|
|
result->_subdivisionTables = spliceSubdivisionTables(result, meshes);
|
|
|
|
|
|
|
|
// splice patch/quad index tables
|
2013-05-16 00:53:40 +00:00
|
|
|
result->_patchTables = splicePatchTables(meshes);
|
2013-03-08 01:50:15 +00:00
|
|
|
|
|
|
|
// splice vertex edit tables
|
|
|
|
result->_vertexEditTables = spliceVertexEditTables(result, meshes);
|
|
|
|
|
2013-05-16 23:55:53 +00:00
|
|
|
// count total num vertices, numptex faces
|
2013-05-17 19:49:57 +00:00
|
|
|
int numVertices = 0, numPtexFaces = 0;
|
2013-03-08 01:50:15 +00:00
|
|
|
for (size_t i = 0; i < meshes.size(); ++i) {
|
|
|
|
numVertices += meshes[i]->GetNumVertices();
|
2013-05-16 23:55:53 +00:00
|
|
|
numPtexFaces += meshes[i]->GetNumPtexFaces();
|
2013-03-08 01:50:15 +00:00
|
|
|
}
|
|
|
|
result->_vertices.resize(numVertices);
|
2013-05-16 23:55:53 +00:00
|
|
|
result->_numPtexFaces = numPtexFaces;
|
|
|
|
result->_totalFVarWidth = 0; // XXX: fvar for multimesh hasn't been implemented yet.
|
2013-03-08 01:50:15 +00:00
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename V, typename IT> static IT
|
|
|
|
copyWithOffset(IT dst_iterator, V const &src, int offset) {
|
|
|
|
return std::transform(src.begin(), src.end(), dst_iterator,
|
|
|
|
std::bind2nd(std::plus<typename V::value_type>(), offset));
|
|
|
|
}
|
|
|
|
|
2013-05-10 02:16:51 +00:00
|
|
|
template <typename V, typename IT> static IT
|
|
|
|
copyWithOffset(IT dst_iterator, V const &src, int start, int count, int offset) {
|
|
|
|
return std::transform(src.begin()+start, src.begin()+start+count, dst_iterator,
|
|
|
|
std::bind2nd(std::plus<typename V::value_type>(), offset));
|
|
|
|
}
|
|
|
|
|
2013-05-15 01:10:19 +00:00
|
|
|
template <typename V, typename IT> static IT
|
|
|
|
copyWithPtexFaceOffset(IT dst_iterator, V const &src, int start, int count, int offset) {
|
|
|
|
for (typename V::const_iterator it = src.begin()+start; it != src.begin()+start+count; ++it) {
|
|
|
|
typename V::value_type ptexCoord = *it;
|
|
|
|
ptexCoord.faceIndex += offset;
|
|
|
|
*dst_iterator++ = ptexCoord;
|
|
|
|
}
|
|
|
|
return dst_iterator;
|
|
|
|
}
|
|
|
|
|
2013-03-08 01:50:15 +00:00
|
|
|
template <typename V, typename IT> static IT
|
|
|
|
copyWithOffsetF_ITa(IT dst_iterator, V const &src, int offset) {
|
|
|
|
for (typename V::const_iterator it = src.begin(); it != src.end();) {
|
|
|
|
*dst_iterator++ = *it++ + offset; // offset to F_IT
|
|
|
|
*dst_iterator++ = *it++; // valence
|
|
|
|
}
|
|
|
|
return dst_iterator;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename V, typename IT> static IT
|
|
|
|
copyWithOffsetE_IT(IT dst_iterator, V const &src, int offset) {
|
|
|
|
for (typename V::const_iterator it = src.begin(); it != src.end(); ++it) {
|
|
|
|
*dst_iterator++ = (*it == -1) ? -1 : (*it + offset);
|
|
|
|
}
|
|
|
|
return dst_iterator;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename V, typename IT> static IT
|
|
|
|
copyWithOffsetV_ITa(IT dst_iterator, V const &src, int tableOffset, int vertexOffset) {
|
|
|
|
for (typename V::const_iterator it = src.begin(); it != src.end();) {
|
|
|
|
*dst_iterator++ = *it++ + tableOffset; // offset to V_IT
|
|
|
|
*dst_iterator++ = *it++; // valence
|
|
|
|
*dst_iterator++ = (*it == -1) ? -1 : (*it + vertexOffset); ++it;
|
|
|
|
*dst_iterator++ = (*it == -1) ? -1 : (*it + vertexOffset); ++it;
|
|
|
|
*dst_iterator++ = (*it == -1) ? -1 : (*it + vertexOffset); ++it;
|
|
|
|
}
|
|
|
|
return dst_iterator;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename V, typename IT> static IT
|
|
|
|
copyWithOffsetVertexValence(IT dst_iterator, V const &src, int srcMaxValence, int dstMaxValence, int offset) {
|
|
|
|
for (typename V::const_iterator it = src.begin(); it != src.end(); ) {
|
|
|
|
int valence = *it++;
|
|
|
|
*dst_iterator++ = valence;
|
|
|
|
valence = abs(valence);
|
|
|
|
for (int i = 0; i < 2*dstMaxValence; ++i) {
|
|
|
|
if (i < 2*srcMaxValence) {
|
|
|
|
*dst_iterator++ = (i < 2*valence) ? *it + offset : 0;
|
|
|
|
++it;
|
|
|
|
} else {
|
|
|
|
*dst_iterator++ = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return dst_iterator;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class T, class U> FarSubdivisionTables<U> *
|
|
|
|
FarMultiMeshFactory<T, U>::spliceSubdivisionTables(FarMesh<U> *farMesh, FarMeshVector const &meshes) {
|
|
|
|
|
|
|
|
// count total table size
|
|
|
|
size_t total_F_ITa = 0, total_F_IT = 0;
|
|
|
|
size_t total_E_IT = 0, total_E_W = 0;
|
|
|
|
size_t total_V_ITa = 0, total_V_IT = 0, total_V_W = 0;
|
|
|
|
for (size_t i = 0; i < meshes.size(); ++i) {
|
|
|
|
FarSubdivisionTables<U> const * tables = meshes[i]->GetSubdivisionTables();
|
|
|
|
assert(tables);
|
|
|
|
|
|
|
|
total_F_ITa += tables->Get_F_ITa().size();
|
|
|
|
total_F_IT += tables->Get_F_IT().size();
|
|
|
|
total_E_IT += tables->Get_E_IT().size();
|
|
|
|
total_E_W += tables->Get_E_W().size();
|
|
|
|
total_V_ITa += tables->Get_V_ITa().size();
|
|
|
|
total_V_IT += tables->Get_V_IT().size();
|
|
|
|
total_V_W += tables->Get_V_W().size();
|
|
|
|
}
|
|
|
|
|
|
|
|
FarSubdivisionTables<U> *result = NULL;
|
|
|
|
const std::type_info &scheme = typeid(*(meshes[0]->GetSubdivisionTables()));
|
|
|
|
|
|
|
|
if (scheme == typeid(FarCatmarkSubdivisionTables<U>) ) {
|
|
|
|
result = new FarCatmarkSubdivisionTables<U>(farMesh, _maxlevel);
|
|
|
|
} else if (scheme == typeid(FarBilinearSubdivisionTables<U>) ) {
|
|
|
|
result = new FarBilinearSubdivisionTables<U>(farMesh, _maxlevel);
|
|
|
|
} else if (scheme == typeid(FarLoopSubdivisionTables<U>) ) {
|
|
|
|
result = new FarLoopSubdivisionTables<U>(farMesh, _maxlevel);
|
|
|
|
}
|
|
|
|
|
|
|
|
result->_F_ITa.resize(total_F_ITa);
|
|
|
|
result->_F_IT.resize(total_F_IT);
|
|
|
|
result->_E_IT.resize(total_E_IT);
|
|
|
|
result->_E_W.resize(total_E_W);
|
|
|
|
result->_V_ITa.resize(total_V_ITa);
|
|
|
|
result->_V_IT.resize(total_V_IT);
|
|
|
|
result->_V_W.resize(total_V_W);
|
|
|
|
|
|
|
|
// compute table offsets;
|
|
|
|
std::vector<int> vertexOffsets;
|
|
|
|
std::vector<int> fvOffsets;
|
|
|
|
std::vector<int> evOffsets;
|
|
|
|
std::vector<int> vvOffsets;
|
|
|
|
std::vector<int> F_IToffsets;
|
|
|
|
std::vector<int> V_IToffsets;
|
|
|
|
|
|
|
|
{
|
|
|
|
int vertexOffset = 0;
|
|
|
|
int F_IToffset = 0;
|
|
|
|
int V_IToffset = 0;
|
|
|
|
int fvOffset = 0;
|
|
|
|
int evOffset = 0;
|
|
|
|
int vvOffset = 0;
|
|
|
|
for (size_t i = 0; i < meshes.size(); ++i) {
|
|
|
|
FarSubdivisionTables<U> const * tables = meshes[i]->GetSubdivisionTables();
|
|
|
|
assert(tables);
|
|
|
|
|
|
|
|
vertexOffsets.push_back(vertexOffset);
|
|
|
|
F_IToffsets.push_back(F_IToffset);
|
|
|
|
V_IToffsets.push_back(V_IToffset);
|
|
|
|
fvOffsets.push_back(fvOffset);
|
|
|
|
evOffsets.push_back(evOffset);
|
|
|
|
vvOffsets.push_back(vvOffset);
|
|
|
|
|
|
|
|
vertexOffset += meshes[i]->GetNumVertices();
|
|
|
|
F_IToffset += (int)tables->Get_F_IT().size();
|
|
|
|
fvOffset += (int)tables->Get_F_ITa().size()/2;
|
|
|
|
V_IToffset += (int)tables->Get_V_IT().size();
|
|
|
|
|
|
|
|
if (scheme == typeid(FarCatmarkSubdivisionTables<U>) ||
|
|
|
|
scheme == typeid(FarLoopSubdivisionTables<U>)) {
|
|
|
|
evOffset += (int)tables->Get_E_IT().size()/4;
|
|
|
|
vvOffset += (int)tables->Get_V_ITa().size()/5;
|
|
|
|
} else {
|
|
|
|
evOffset += (int)tables->Get_E_IT().size()/2;
|
|
|
|
vvOffset += (int)tables->Get_V_ITa().size();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// concat F_IT and V_IT
|
|
|
|
std::vector<unsigned int>::iterator F_IT = result->_F_IT.begin();
|
|
|
|
std::vector<unsigned int>::iterator V_IT = result->_V_IT.begin();
|
|
|
|
|
|
|
|
for (size_t i = 0; i < meshes.size(); ++i) {
|
|
|
|
FarSubdivisionTables<U> const * tables = meshes[i]->GetSubdivisionTables();
|
|
|
|
|
|
|
|
int vertexOffset = vertexOffsets[i];
|
|
|
|
// remap F_IT, V_IT tables
|
|
|
|
F_IT = copyWithOffset(F_IT, tables->Get_F_IT(), vertexOffset);
|
|
|
|
V_IT = copyWithOffset(V_IT, tables->Get_V_IT(), vertexOffset);
|
|
|
|
}
|
|
|
|
|
|
|
|
// merge other tables
|
|
|
|
std::vector<int>::iterator F_ITa = result->_F_ITa.begin();
|
|
|
|
std::vector<int>::iterator E_IT = result->_E_IT.begin();
|
|
|
|
std::vector<float>::iterator E_W = result->_E_W.begin();
|
|
|
|
std::vector<float>::iterator V_W = result->_V_W.begin();
|
|
|
|
std::vector<int>::iterator V_ITa = result->_V_ITa.begin();
|
|
|
|
|
|
|
|
for (size_t i = 0; i < meshes.size(); ++i) {
|
|
|
|
FarSubdivisionTables<U> const * tables = meshes[i]->GetSubdivisionTables();
|
|
|
|
|
|
|
|
// copy face tables
|
|
|
|
F_ITa = copyWithOffsetF_ITa(F_ITa, tables->Get_F_ITa(), F_IToffsets[i]);
|
|
|
|
|
|
|
|
// copy edge tables
|
|
|
|
E_IT = copyWithOffsetE_IT(E_IT, tables->Get_E_IT(), vertexOffsets[i]);
|
|
|
|
E_W = copyWithOffset(E_W, tables->Get_E_W(), 0);
|
|
|
|
|
|
|
|
// copy vert tables
|
|
|
|
if (scheme == typeid(FarCatmarkSubdivisionTables<U>) ||
|
|
|
|
scheme == typeid(FarLoopSubdivisionTables<U>)) {
|
|
|
|
V_ITa = copyWithOffsetV_ITa(V_ITa, tables->Get_V_ITa(), V_IToffsets[i], vertexOffsets[i]);
|
|
|
|
} else {
|
|
|
|
V_ITa = copyWithOffset(V_ITa, tables->Get_V_ITa(), vertexOffsets[i]);
|
|
|
|
}
|
|
|
|
V_W = copyWithOffset(V_W, tables->Get_V_W(), 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
// merge batch, model by model
|
|
|
|
FarKernelBatchVector &batches = farMesh->_batches;
|
|
|
|
int editTableIndexOffset = 0;
|
2013-06-10 23:30:25 +00:00
|
|
|
for (int i = 0; i < (int)meshes.size(); ++i) {
|
2013-03-08 01:50:15 +00:00
|
|
|
for (int j = 0; j < (int)meshes[i]->_batches.size(); ++j) {
|
|
|
|
FarKernelBatch batch = meshes[i]->_batches[j];
|
2013-06-06 23:20:40 +00:00
|
|
|
batch._meshIndex = i;
|
2013-05-07 00:50:58 +00:00
|
|
|
batch._vertexOffset += vertexOffsets[i];
|
2013-03-08 01:50:15 +00:00
|
|
|
|
2013-05-07 00:50:58 +00:00
|
|
|
if (batch._kernelType == FarKernelBatch::CATMARK_FACE_VERTEX or
|
|
|
|
batch._kernelType == FarKernelBatch::BILINEAR_FACE_VERTEX) {
|
|
|
|
|
|
|
|
batch._tableOffset += fvOffsets[i];
|
|
|
|
|
|
|
|
} else if (batch._kernelType == FarKernelBatch::CATMARK_EDGE_VERTEX or
|
|
|
|
batch._kernelType == FarKernelBatch::LOOP_EDGE_VERTEX or
|
|
|
|
batch._kernelType == FarKernelBatch::BILINEAR_EDGE_VERTEX) {
|
|
|
|
|
|
|
|
batch._tableOffset += evOffsets[i];
|
|
|
|
|
|
|
|
} else if (batch._kernelType == FarKernelBatch::CATMARK_VERT_VERTEX_A1 or
|
|
|
|
batch._kernelType == FarKernelBatch::CATMARK_VERT_VERTEX_A2 or
|
|
|
|
batch._kernelType == FarKernelBatch::CATMARK_VERT_VERTEX_B or
|
|
|
|
batch._kernelType == FarKernelBatch::LOOP_VERT_VERTEX_A1 or
|
|
|
|
batch._kernelType == FarKernelBatch::LOOP_VERT_VERTEX_A2 or
|
|
|
|
batch._kernelType == FarKernelBatch::LOOP_VERT_VERTEX_B or
|
|
|
|
batch._kernelType == FarKernelBatch::BILINEAR_VERT_VERTEX) {
|
|
|
|
|
|
|
|
batch._tableOffset += vvOffsets[i];
|
|
|
|
|
|
|
|
} else if (batch._kernelType == FarKernelBatch::HIERARCHICAL_EDIT) {
|
|
|
|
|
|
|
|
batch._tableIndex += editTableIndexOffset;
|
2013-03-08 01:50:15 +00:00
|
|
|
}
|
|
|
|
batches.push_back(batch);
|
|
|
|
}
|
|
|
|
editTableIndexOffset += meshes[i]->_vertexEditTables ? meshes[i]->_vertexEditTables->GetNumBatches() : 0;
|
|
|
|
}
|
2013-05-16 23:55:53 +00:00
|
|
|
|
|
|
|
// count verts offsets
|
|
|
|
result->_vertsOffsets.resize(_maxlevel+2);
|
|
|
|
for (size_t i = 0; i < meshes.size(); ++i) {
|
|
|
|
FarSubdivisionTables<U> const * tables = meshes[i]->GetSubdivisionTables();
|
|
|
|
for (size_t j = 0; j < tables->_vertsOffsets.size(); ++j) {
|
|
|
|
result->_vertsOffsets[j] += tables->_vertsOffsets[j];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-03-08 01:50:15 +00:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2013-05-10 02:16:51 +00:00
|
|
|
template <class T, class U> FarPatchTables::PTable::iterator
|
|
|
|
FarMultiMeshFactory<T, U>::splicePatch(FarPatchTables::Descriptor desc,
|
|
|
|
FarMeshVector const &meshes,
|
|
|
|
FarPatchTables::PatchArrayVector &result,
|
|
|
|
FarPatchTables::PTable::iterator dstIndexIt,
|
|
|
|
int *voffset, int *poffset, int *qoffset,
|
|
|
|
std::vector<int> const &vertexOffsets)
|
|
|
|
{
|
|
|
|
for (size_t i = 0; i < meshes.size(); ++i) {
|
|
|
|
FarPatchTables const *patchTables = meshes[i]->GetPatchTables();
|
|
|
|
FarPatchTables::PatchArray const *srcPatchArray = patchTables->GetPatchArray(desc);
|
|
|
|
if (not srcPatchArray) continue;
|
|
|
|
|
|
|
|
// create new patcharray with offset
|
|
|
|
int vindex = srcPatchArray->GetVertIndex();
|
|
|
|
int npatch = srcPatchArray->GetNumPatches();
|
2013-05-16 23:55:53 +00:00
|
|
|
int nvertex = npatch * desc.GetNumControlVertices();
|
2013-05-10 02:16:51 +00:00
|
|
|
|
2013-05-16 23:55:53 +00:00
|
|
|
FarPatchTables::PatchArray patchArray(desc,
|
2013-05-10 02:16:51 +00:00
|
|
|
*voffset,
|
|
|
|
*poffset,
|
|
|
|
npatch,
|
|
|
|
*qoffset);
|
|
|
|
// append patch array
|
|
|
|
result.push_back(patchArray);
|
2013-05-16 23:55:53 +00:00
|
|
|
|
|
|
|
// also store into multiPatchArrays, will be used for partial drawing
|
|
|
|
// XXX: can be stored as indices. revisit here later
|
2013-05-10 02:16:51 +00:00
|
|
|
_multiPatchArrays[i].push_back(patchArray);
|
|
|
|
|
|
|
|
// increment offset
|
|
|
|
*voffset += nvertex;
|
|
|
|
*poffset += npatch;
|
|
|
|
*qoffset += (desc.GetType() == FarPatchTables::GREGORY ||
|
|
|
|
desc.GetType() == FarPatchTables::GREGORY_BOUNDARY) ? npatch * 4 : 0;
|
|
|
|
|
|
|
|
// copy index arrays [vindex, vindex+nvertex]
|
|
|
|
dstIndexIt = copyWithOffset(dstIndexIt,
|
|
|
|
patchTables->GetPatchTable(),
|
|
|
|
vindex,
|
|
|
|
nvertex,
|
|
|
|
vertexOffsets[i]);
|
|
|
|
}
|
|
|
|
return dstIndexIt;
|
|
|
|
}
|
|
|
|
|
2013-03-08 01:50:15 +00:00
|
|
|
template <class T, class U> FarPatchTables *
|
|
|
|
FarMultiMeshFactory<T, U>::splicePatchTables(FarMeshVector const &meshes) {
|
|
|
|
|
|
|
|
FarPatchTables *result = new FarPatchTables(_maxvalence);
|
|
|
|
|
|
|
|
int total_quadOffset0 = 0;
|
|
|
|
int total_quadOffset1 = 0;
|
|
|
|
|
|
|
|
std::vector<int> vertexOffsets;
|
2013-05-10 02:16:51 +00:00
|
|
|
std::vector<int> gregoryQuadOffsets;
|
|
|
|
std::vector<int> numGregoryPatches;
|
2013-03-08 01:50:15 +00:00
|
|
|
int vertexOffset = 0;
|
|
|
|
int maxValence = 0;
|
2013-05-10 02:16:51 +00:00
|
|
|
int numTotalIndices = 0;
|
2013-03-08 01:50:15 +00:00
|
|
|
|
2013-05-10 02:16:51 +00:00
|
|
|
//result->_patchCounts.reserve(meshes.size());
|
|
|
|
//FarPatchCount totalCount;
|
|
|
|
typedef FarPatchTables::Descriptor Descriptor;
|
2013-03-08 01:50:15 +00:00
|
|
|
|
|
|
|
// count how many patches exist on each mesh
|
|
|
|
for (size_t i = 0; i < meshes.size(); ++i) {
|
|
|
|
const FarPatchTables *ptables = meshes[i]->GetPatchTables();
|
|
|
|
assert(ptables);
|
|
|
|
|
|
|
|
vertexOffsets.push_back(vertexOffset);
|
|
|
|
vertexOffset += meshes[i]->GetNumVertices();
|
|
|
|
|
|
|
|
// need to align maxvalence with the highest value
|
|
|
|
maxValence = std::max(maxValence, ptables->_maxValence);
|
2013-05-10 02:16:51 +00:00
|
|
|
|
|
|
|
FarPatchTables::PatchArray const *gregory = ptables->GetPatchArray(Descriptor(FarPatchTables::GREGORY, FarPatchTables::NON_TRANSITION, /*rot*/ 0));
|
|
|
|
FarPatchTables::PatchArray const *gregoryBoundary = ptables->GetPatchArray(Descriptor(FarPatchTables::GREGORY_BOUNDARY, FarPatchTables::NON_TRANSITION, /*rot*/ 0));
|
|
|
|
|
|
|
|
int nGregory = gregory ? gregory->GetNumPatches() : 0;
|
|
|
|
int nGregoryBoundary = gregoryBoundary ? gregoryBoundary->GetNumPatches() : 0;
|
|
|
|
total_quadOffset0 += nGregory * 4;
|
|
|
|
total_quadOffset1 += nGregoryBoundary * 4;
|
|
|
|
numGregoryPatches.push_back(nGregory);
|
|
|
|
gregoryQuadOffsets.push_back(total_quadOffset0);
|
|
|
|
|
|
|
|
numTotalIndices += ptables->GetNumControlVertices();
|
2013-03-08 01:50:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Allocate full patches
|
2013-05-10 02:16:51 +00:00
|
|
|
result->_patches.resize(numTotalIndices);
|
|
|
|
|
2013-03-08 01:50:15 +00:00
|
|
|
// Allocate vertex valence table, quad offset table
|
2013-05-10 02:16:51 +00:00
|
|
|
if (total_quadOffset0 + total_quadOffset1 > 0) {
|
2013-03-08 01:50:15 +00:00
|
|
|
result->_vertexValenceTable.resize((2*maxValence+1) * vertexOffset);
|
|
|
|
result->_quadOffsetTable.resize(total_quadOffset0 + total_quadOffset1);
|
|
|
|
}
|
|
|
|
|
2013-05-10 02:16:51 +00:00
|
|
|
// splice tables
|
|
|
|
// assuming input farmeshes have dense patchtables
|
2013-03-08 01:50:15 +00:00
|
|
|
|
2013-05-10 02:16:51 +00:00
|
|
|
_multiPatchArrays.resize(meshes.size());
|
2013-03-08 01:50:15 +00:00
|
|
|
|
2013-05-10 02:16:51 +00:00
|
|
|
int voffset = 0, poffset = 0, qoffset = 0;
|
|
|
|
FarPatchTables::PTable::iterator dstIndexIt = result->_patches.begin();
|
|
|
|
|
2013-05-17 23:38:15 +00:00
|
|
|
// splice patches : iterate from POINTS
|
|
|
|
for (FarPatchTables::Descriptor::iterator it(FarPatchTables::Descriptor(FarPatchTables::POINTS, FarPatchTables::NON_TRANSITION, 0));
|
|
|
|
it != FarPatchTables::Descriptor::end(); ++it) {
|
2013-05-10 02:16:51 +00:00
|
|
|
dstIndexIt = splicePatch(*it, meshes, result->_patchArrays, dstIndexIt, &voffset, &poffset, &qoffset, vertexOffsets);
|
2013-03-08 01:50:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// merge vertexvalence and quadoffset tables
|
|
|
|
std::vector<unsigned int>::iterator Q0_IT = result->_quadOffsetTable.begin();
|
|
|
|
std::vector<unsigned int>::iterator Q1_IT = Q0_IT + total_quadOffset0;
|
|
|
|
|
|
|
|
std::vector<int>::iterator VV_IT = result->_vertexValenceTable.begin();
|
|
|
|
for (size_t i = 0; i < meshes.size(); ++i) {
|
|
|
|
const FarPatchTables *ptables = meshes[i]->GetPatchTables();
|
|
|
|
|
|
|
|
// merge vertex valence
|
|
|
|
// note: some prims may not have vertex valence table, but still need a space
|
|
|
|
// in order to fill following prim's data at appropriate location.
|
2013-05-10 02:16:51 +00:00
|
|
|
copyWithOffsetVertexValence(VV_IT,
|
|
|
|
ptables->_vertexValenceTable,
|
|
|
|
ptables->_maxValence,
|
|
|
|
maxValence,
|
|
|
|
vertexOffsets[i]);
|
|
|
|
|
2013-03-08 01:50:15 +00:00
|
|
|
VV_IT += meshes[i]->GetNumVertices() * (2 * maxValence + 1);
|
|
|
|
|
|
|
|
// merge quad offsets
|
2013-05-10 02:16:51 +00:00
|
|
|
// int nGregoryQuads = (int)ptables->_full._G_IT.first.size();
|
|
|
|
int nGregoryQuads = numGregoryPatches[i] * 4;
|
|
|
|
if (nGregoryQuads > 0) {
|
|
|
|
Q0_IT = std::copy(ptables->_quadOffsetTable.begin(),
|
|
|
|
ptables->_quadOffsetTable.begin()+nGregoryQuads,
|
|
|
|
Q0_IT);
|
|
|
|
}
|
|
|
|
if (nGregoryQuads < (int)ptables->_quadOffsetTable.size()) {
|
|
|
|
Q1_IT = std::copy(ptables->_quadOffsetTable.begin()+nGregoryQuads,
|
|
|
|
ptables->_quadOffsetTable.end(),
|
|
|
|
Q1_IT);
|
|
|
|
}
|
2013-05-15 01:10:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// merge ptexCoord table
|
2013-05-17 23:38:15 +00:00
|
|
|
for (FarPatchTables::Descriptor::iterator it(FarPatchTables::Descriptor(FarPatchTables::POINTS, FarPatchTables::NON_TRANSITION, 0));
|
|
|
|
it != FarPatchTables::Descriptor::end(); ++it) {
|
2013-05-15 01:10:19 +00:00
|
|
|
int ptexFaceOffset = 0;
|
|
|
|
for (size_t i = 0; i < meshes.size(); ++i) {
|
|
|
|
FarPatchTables const *ptables = meshes[i]->GetPatchTables();
|
|
|
|
FarPatchTables::PatchArray const *parray = ptables->GetPatchArray(*it);
|
2013-06-18 21:07:32 +00:00
|
|
|
if (parray) {
|
|
|
|
copyWithPtexFaceOffset(std::back_inserter(result->_paramTable),
|
|
|
|
ptables->_paramTable,
|
|
|
|
parray->GetPatchIndex(),
|
|
|
|
parray->GetNumPatches(), ptexFaceOffset);
|
|
|
|
}
|
2013-05-15 01:10:19 +00:00
|
|
|
ptexFaceOffset += meshes[i]->GetNumPtexFaces();
|
|
|
|
}
|
2013-03-08 01:50:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class T, class U> FarVertexEditTables<U> *
|
|
|
|
FarMultiMeshFactory<T, U>::spliceVertexEditTables(FarMesh<U> *farMesh, FarMeshVector const &meshes) {
|
|
|
|
|
|
|
|
FarVertexEditTables<U> * result = new FarVertexEditTables<U>(farMesh);
|
|
|
|
|
|
|
|
// at this moment, don't merge vertex edit tables (separate batch)
|
|
|
|
for (size_t i = 0; i < meshes.size(); ++i) {
|
|
|
|
const FarVertexEditTables<U> *vertexEditTables = meshes[i]->GetVertexEdit();
|
|
|
|
if (not vertexEditTables) continue;
|
|
|
|
|
|
|
|
// copy each edit batch XXX:inefficient copy
|
|
|
|
result->_batches.insert(result->_batches.end(),
|
|
|
|
vertexEditTables->_batches.begin(),
|
|
|
|
vertexEditTables->_batches.end());
|
|
|
|
}
|
|
|
|
|
|
|
|
if (result->_batches.empty()) {
|
|
|
|
delete result;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // end namespace OPENSUBDIV_VERSION
|
|
|
|
using namespace OPENSUBDIV_VERSION;
|
|
|
|
|
|
|
|
} // end namespace OpenSubdiv
|
|
|
|
|
|
|
|
#endif /* FAR_MULTI_MESH_FACTORY_H */
|