2014-09-05 22:07:46 +00:00
|
|
|
//
|
|
|
|
// Copyright 2013 Pixar
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "Apache License")
|
|
|
|
// with the following modification; you may not use this file except in
|
|
|
|
// compliance with the Apache License and the following modification to it:
|
|
|
|
// Section 6. Trademarks. is deleted and replaced with:
|
|
|
|
//
|
|
|
|
// 6. Trademarks. This License does not grant permission to use the trade
|
|
|
|
// names, trademarks, service marks, or product names of the Licensor
|
|
|
|
// and its affiliates, except as required to comply with Section 4(c) of
|
|
|
|
// the License and to reproduce the content of the NOTICE file.
|
|
|
|
//
|
|
|
|
// You may obtain a copy of the Apache License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the Apache License with the above modification is
|
|
|
|
// distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
|
|
|
|
// KIND, either express or implied. See the Apache License for the specific
|
|
|
|
// language governing permissions and limitations under the Apache License.
|
|
|
|
//
|
2015-05-22 18:50:01 +00:00
|
|
|
#include "../far/patchTableFactory.h"
|
2015-04-18 00:36:55 +00:00
|
|
|
#include "../far/error.h"
|
2015-04-20 06:25:43 +00:00
|
|
|
#include "../far/ptexIndices.h"
|
2014-09-05 22:07:46 +00:00
|
|
|
#include "../far/topologyRefiner.h"
|
|
|
|
#include "../vtr/level.h"
|
2015-06-01 07:14:19 +00:00
|
|
|
#include "../vtr/fvarLevel.h"
|
2014-09-05 22:07:46 +00:00
|
|
|
#include "../vtr/refinement.h"
|
2016-07-22 01:47:44 +00:00
|
|
|
#include "../vtr/stackBuffer.h"
|
2015-04-23 23:58:45 +00:00
|
|
|
#include "../far/endCapBSplineBasisPatchFactory.h"
|
|
|
|
#include "../far/endCapGregoryBasisPatchFactory.h"
|
|
|
|
#include "../far/endCapLegacyGregoryPatchFactory.h"
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2014-11-18 18:48:37 +00:00
|
|
|
#include <algorithm>
|
2014-09-05 22:07:46 +00:00
|
|
|
#include <cassert>
|
|
|
|
#include <cstring>
|
|
|
|
|
|
|
|
namespace OpenSubdiv {
|
|
|
|
namespace OPENSUBDIV_VERSION {
|
|
|
|
|
2015-02-26 21:57:47 +00:00
|
|
|
namespace {
|
2015-07-10 23:54:12 +00:00
|
|
|
|
2015-07-21 00:56:00 +00:00
|
|
|
// Helpers for compiler warnings and floating point equality tests
|
|
|
|
#ifdef __INTEL_COMPILER
|
|
|
|
#pragma warning (push)
|
|
|
|
#pragma warning disable 1572
|
|
|
|
#endif
|
|
|
|
|
|
|
|
inline bool isSharpnessEqual(float s1, float s2) { return (s1 == s2); }
|
|
|
|
|
|
|
|
#ifdef __INTEL_COMPILER
|
|
|
|
#pragma warning (pop)
|
|
|
|
#endif
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2015-04-18 00:36:55 +00:00
|
|
|
} // namespace anon
|
2014-09-05 22:07:46 +00:00
|
|
|
|
|
|
|
|
2015-04-18 00:36:55 +00:00
|
|
|
namespace Far {
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2015-04-18 00:36:55 +00:00
|
|
|
void
|
2015-05-22 18:50:01 +00:00
|
|
|
PatchTableFactory::PatchFaceTag::clear() {
|
2015-04-18 00:36:55 +00:00
|
|
|
std::memset(this, 0, sizeof(*this));
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2015-05-22 18:50:01 +00:00
|
|
|
PatchTableFactory::PatchFaceTag::assignBoundaryPropertiesFromEdgeMask(int boundaryEdgeMask) {
|
2015-04-18 00:36:55 +00:00
|
|
|
//
|
|
|
|
// The number of rotations to apply for boundary or corner patches varies on both
|
|
|
|
// where the boundary/corner occurs and whether boundary or corner -- so using a
|
|
|
|
// 4-bit mask should be sufficient to quickly determine all cases:
|
|
|
|
//
|
|
|
|
// Note that we currently expect patches with multiple boundaries to have already
|
|
|
|
// been isolated, so asserts are applied for such unexpected cases.
|
|
|
|
//
|
|
|
|
// Is the compiler going to build the 16-entry lookup table here, or should we do
|
|
|
|
// it ourselves?
|
|
|
|
//
|
|
|
|
_hasBoundaryEdge = true;
|
|
|
|
_boundaryMask = boundaryEdgeMask;
|
|
|
|
|
|
|
|
switch (boundaryEdgeMask) {
|
|
|
|
case 0x0: _boundaryCount = 0, _boundaryIndex = 0, _hasBoundaryEdge = false; break; // no boundaries
|
|
|
|
|
|
|
|
case 0x1: _boundaryCount = 1, _boundaryIndex = 0; break; // boundary edge 0
|
|
|
|
case 0x2: _boundaryCount = 1, _boundaryIndex = 1; break; // boundary edge 1
|
|
|
|
case 0x3: _boundaryCount = 2, _boundaryIndex = 1; break; // corner/crease vertex 1
|
|
|
|
case 0x4: _boundaryCount = 1, _boundaryIndex = 2; break; // boundary edge 2
|
|
|
|
case 0x5: assert(false); break; // N/A - opposite boundary edges
|
|
|
|
case 0x6: _boundaryCount = 2, _boundaryIndex = 2; break; // corner/crease vertex 2
|
|
|
|
case 0x7: assert(false); break; // N/A - three boundary edges
|
|
|
|
case 0x8: _boundaryCount = 1, _boundaryIndex = 3; break; // boundary edge 3
|
|
|
|
case 0x9: _boundaryCount = 2, _boundaryIndex = 0; break; // corner/crease vertex 0
|
|
|
|
case 0xa: assert(false); break; // N/A - opposite boundary edges
|
|
|
|
case 0xb: assert(false); break; // N/A - three boundary edges
|
|
|
|
case 0xc: _boundaryCount = 2, _boundaryIndex = 3; break; // corner/crease vertex 3
|
|
|
|
case 0xd: assert(false); break; // N/A - three boundary edges
|
|
|
|
case 0xe: assert(false); break; // N/A - three boundary edges
|
|
|
|
case 0xf: assert(false); break; // N/A - all boundaries
|
|
|
|
default: assert(false); break;
|
|
|
|
}
|
|
|
|
}
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2015-04-18 00:36:55 +00:00
|
|
|
void
|
2015-05-22 18:50:01 +00:00
|
|
|
PatchTableFactory::PatchFaceTag::assignBoundaryPropertiesFromVertexMask(int boundaryVertexMask) {
|
2015-04-18 00:36:55 +00:00
|
|
|
//
|
|
|
|
// This is strictly needed for the irregular case when a vertex is a boundary in
|
|
|
|
// the presence of no boundary edges -- an extra-ordinary face with only one corner
|
|
|
|
// on the boundary.
|
|
|
|
//
|
|
|
|
// Its unclear at this point if patches with more than one such vertex are supported
|
|
|
|
// (if so, how do we deal with rotations) so for now we only allow one such vertex
|
|
|
|
// and assert for all other cases.
|
|
|
|
//
|
|
|
|
assert(_hasBoundaryEdge == false);
|
|
|
|
_boundaryMask = boundaryVertexMask;
|
|
|
|
|
|
|
|
switch (boundaryVertexMask) {
|
|
|
|
case 0x0: _boundaryCount = 0; break; // no boundaries
|
|
|
|
case 0x1: _boundaryCount = 1, _boundaryIndex = 0; break; // boundary vertex 0
|
|
|
|
case 0x2: _boundaryCount = 1, _boundaryIndex = 1; break; // boundary vertex 1
|
|
|
|
case 0x3: assert(false); break;
|
|
|
|
case 0x4: _boundaryCount = 1, _boundaryIndex = 2; break; // boundary vertex 2
|
|
|
|
case 0x5: assert(false); break;
|
|
|
|
case 0x6: assert(false); break;
|
|
|
|
case 0x7: assert(false); break;
|
|
|
|
case 0x8: _boundaryCount = 1, _boundaryIndex = 3; break; // boundary vertex 3
|
|
|
|
case 0x9: assert(false); break;
|
|
|
|
case 0xa: assert(false); break;
|
|
|
|
case 0xb: assert(false); break;
|
|
|
|
case 0xc: assert(false); break;
|
|
|
|
case 0xd: assert(false); break;
|
|
|
|
case 0xe: assert(false); break;
|
|
|
|
case 0xf: assert(false); break;
|
|
|
|
default: assert(false); break;
|
|
|
|
}
|
|
|
|
}
|
2014-09-05 22:07:46 +00:00
|
|
|
|
|
|
|
//
|
|
|
|
// Trivial anonymous helper functions:
|
|
|
|
//
|
2015-04-18 00:36:55 +00:00
|
|
|
static inline void
|
2015-02-26 21:57:47 +00:00
|
|
|
offsetAndPermuteIndices(Far::Index const indices[], int count,
|
|
|
|
Far::Index offset, int const permutation[],
|
|
|
|
Far::Index result[]) {
|
|
|
|
|
2015-04-24 20:18:03 +00:00
|
|
|
// The patch vertices for boundary and corner patches
|
|
|
|
// are assigned index values even though indices will
|
|
|
|
// be undefined along boundary and corner edges.
|
2015-05-22 18:50:01 +00:00
|
|
|
// When the resulting patch table is going to be used
|
2015-04-24 20:18:03 +00:00
|
|
|
// as indices for drawing, it is convenient for invalid
|
|
|
|
// indices to be replaced with known good values, such
|
|
|
|
// as the first un-permuted index, which is the index
|
|
|
|
// of the first vertex of the patch face.
|
|
|
|
Far::Index knownGoodIndex = indices[0];
|
|
|
|
|
2015-02-26 21:57:47 +00:00
|
|
|
if (permutation) {
|
|
|
|
for (int i = 0; i < count; ++i) {
|
2015-04-27 19:40:18 +00:00
|
|
|
if (permutation[i] < 0) {
|
|
|
|
result[i] = offset + knownGoodIndex;
|
|
|
|
} else {
|
|
|
|
result[i] = offset + indices[permutation[i]];
|
2015-04-17 14:42:53 +00:00
|
|
|
}
|
2015-02-26 21:57:47 +00:00
|
|
|
}
|
2015-04-27 19:40:18 +00:00
|
|
|
} else if (offset) {
|
2015-02-26 21:57:47 +00:00
|
|
|
for (int i = 0; i < count; ++i) {
|
2015-04-27 19:40:18 +00:00
|
|
|
result[i] = offset + indices[i];
|
2015-02-26 21:57:47 +00:00
|
|
|
}
|
2015-04-27 19:40:18 +00:00
|
|
|
} else {
|
|
|
|
std::memcpy(result, indices, count * sizeof(Far::Index));
|
2015-02-26 21:57:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
2016-07-22 01:47:44 +00:00
|
|
|
// Builder Context
|
2015-02-26 21:57:47 +00:00
|
|
|
//
|
2016-07-22 01:47:44 +00:00
|
|
|
// Helper class aggregating transient contextual data structures during
|
|
|
|
// the creation of a patch table.
|
|
|
|
// This helps keeping the factory class stateless.
|
2015-02-26 21:57:47 +00:00
|
|
|
//
|
2016-07-22 01:47:44 +00:00
|
|
|
// Note : struct members are not re-entrant nor are they intended to be !
|
|
|
|
//
|
|
|
|
struct PatchTableFactory::BuilderContext {
|
2015-02-26 21:57:47 +00:00
|
|
|
|
|
|
|
public:
|
2016-07-22 01:47:44 +00:00
|
|
|
BuilderContext(TopologyRefiner const & refiner, Options options);
|
2015-02-26 21:57:47 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
TopologyRefiner const & refiner;
|
|
|
|
|
|
|
|
Options const options;
|
|
|
|
|
|
|
|
PtexIndices const ptexIndices;
|
|
|
|
|
|
|
|
public:
|
|
|
|
struct PatchTuple {
|
|
|
|
PatchTuple()
|
|
|
|
: tag(), faceIndex(-1), levelIndex(-1) { }
|
|
|
|
PatchTuple(PatchTuple const & p)
|
|
|
|
: tag(p.tag), faceIndex(p.faceIndex), levelIndex(p.levelIndex) { }
|
|
|
|
PatchTuple(PatchFaceTag const & tag, int faceIndex, int levelIndex)
|
|
|
|
: tag(tag), faceIndex(faceIndex), levelIndex(levelIndex) { }
|
|
|
|
|
|
|
|
PatchFaceTag tag;
|
|
|
|
int faceIndex;
|
|
|
|
int levelIndex;
|
|
|
|
};
|
2016-08-06 02:00:51 +00:00
|
|
|
typedef std::vector<PatchTuple> PatchTupleVector;
|
2016-07-22 01:47:44 +00:00
|
|
|
|
|
|
|
int gatherBilinearPatchPoints(Index * iptrs,
|
|
|
|
PatchTuple const & patch,
|
|
|
|
int fvarChannel = -1);
|
|
|
|
int gatherRegularPatchPoints(Index * iptrs,
|
|
|
|
PatchTuple const & patch,
|
|
|
|
int fvarChannel = -1);
|
|
|
|
template <class END_CAP_FACTORY_TYPE>
|
|
|
|
int gatherEndCapPatchPoints(END_CAP_FACTORY_TYPE *endCapFactory,
|
|
|
|
Index * iptrs,
|
|
|
|
PatchTuple const & patch,
|
|
|
|
int fvarChannel = -1);
|
|
|
|
|
|
|
|
// True if face-varying patches need to be generated for this topology
|
|
|
|
bool RequiresFVarPatches() const {
|
|
|
|
return (! fvarChannelIndices.empty());
|
|
|
|
}
|
|
|
|
|
|
|
|
// Counters accumulating each type of patch during topology traversal
|
|
|
|
int numRegularPatches;
|
|
|
|
int numIrregularPatches;
|
|
|
|
int numIrregularBoundaryPatches;
|
|
|
|
|
|
|
|
// Tuple for each patch identified during topology traversal
|
2016-08-06 02:00:51 +00:00
|
|
|
PatchTupleVector patches;
|
2016-07-22 01:47:44 +00:00
|
|
|
|
|
|
|
std::vector<int> levelVertOffsets;
|
|
|
|
std::vector< std::vector<int> > levelFVarValueOffsets;
|
|
|
|
|
|
|
|
// These are the indices of face-varying channels in the refiner
|
|
|
|
// or empty if we are not populating face-varying data.
|
|
|
|
std::vector<int> fvarChannelIndices;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Constructor
|
|
|
|
PatchTableFactory::BuilderContext::BuilderContext(
|
|
|
|
TopologyRefiner const & ref, Options opts) :
|
|
|
|
refiner(ref), options(opts), ptexIndices(refiner),
|
|
|
|
numRegularPatches(0), numIrregularPatches(0),
|
|
|
|
numIrregularBoundaryPatches(0) {
|
|
|
|
|
|
|
|
if (options.generateFVarTables) {
|
|
|
|
// If client-code does not select specific channels, default to all
|
|
|
|
// the channels in the refiner.
|
|
|
|
if (options.numFVarChannels==-1) {
|
|
|
|
fvarChannelIndices.resize(refiner.GetNumFVarChannels());
|
|
|
|
for (int fvc=0;fvc<(int)fvarChannelIndices.size(); ++fvc) {
|
|
|
|
fvarChannelIndices[fvc] = fvc; // std::iota
|
2014-09-05 22:07:46 +00:00
|
|
|
}
|
|
|
|
} else {
|
2016-07-22 01:47:44 +00:00
|
|
|
fvarChannelIndices.assign(
|
|
|
|
options.fvarChannelIndices,
|
|
|
|
options.fvarChannelIndices+options.numFVarChannels);
|
2014-09-05 22:07:46 +00:00
|
|
|
}
|
|
|
|
}
|
2016-07-22 01:47:44 +00:00
|
|
|
}
|
2015-02-26 21:57:47 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
int
|
|
|
|
PatchTableFactory::BuilderContext::gatherBilinearPatchPoints(
|
|
|
|
Index * iptrs, PatchTuple const & patch, int fvarChannel) {
|
2015-02-26 21:57:47 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
Vtr::internal::Level const * level = &refiner.getLevel(patch.levelIndex);
|
|
|
|
int levelVertOffset = (fvarChannel < 0)
|
|
|
|
? levelVertOffsets[patch.levelIndex]
|
|
|
|
: levelFVarValueOffsets[fvarChannel][patch.levelIndex];
|
2015-02-26 21:57:47 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
ConstIndexArray cvs = (fvarChannel < 0)
|
|
|
|
? level->getFaceVertices(patch.faceIndex)
|
|
|
|
: level->getFaceFVarValues(patch.faceIndex,
|
|
|
|
fvarChannelIndices[fvarChannel]);
|
2015-02-26 21:57:47 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
for (int i = 0; i < cvs.size(); ++i) iptrs[i] = levelVertOffset + cvs[i];
|
|
|
|
return cvs.size();
|
|
|
|
}
|
2015-02-26 21:57:47 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
int
|
|
|
|
PatchTableFactory::BuilderContext::gatherRegularPatchPoints(
|
|
|
|
Index * iptrs, PatchTuple const & patch, int fvarChannel) {
|
|
|
|
|
|
|
|
Vtr::internal::Level const * level = &refiner.getLevel(patch.levelIndex);
|
|
|
|
int levelVertOffset = (fvarChannel < 0)
|
|
|
|
? levelVertOffsets[patch.levelIndex]
|
|
|
|
: levelFVarValueOffsets[fvarChannel][patch.levelIndex];
|
|
|
|
int refinerChannel = (fvarChannel < 0)
|
|
|
|
? fvarChannel
|
|
|
|
: fvarChannelIndices[fvarChannel];
|
|
|
|
|
|
|
|
Index patchVerts[16];
|
|
|
|
|
|
|
|
int bIndex = patch.tag._boundaryIndex;
|
|
|
|
|
|
|
|
int const * permutation = 0;
|
|
|
|
|
|
|
|
if (patch.tag._boundaryCount == 0) {
|
|
|
|
static int const permuteRegular[16] =
|
|
|
|
{ 5, 6, 7, 8, 4, 0, 1, 9, 15, 3, 2, 10, 14, 13, 12, 11 };
|
|
|
|
permutation = permuteRegular;
|
|
|
|
level->gatherQuadRegularInteriorPatchPoints(
|
|
|
|
patch.faceIndex, patchVerts, /*rotation=*/0, refinerChannel);
|
|
|
|
} else if (patch.tag._boundaryCount == 1) {
|
|
|
|
// Expand boundary patch vertices and rotate to
|
|
|
|
// restore correct orientation.
|
|
|
|
static int const permuteBoundary[4][16] = {
|
|
|
|
{ -1, -1, -1, -1, 11, 3, 0, 4, 10, 2, 1, 5, 9, 8, 7, 6 },
|
|
|
|
{ 9, 10, 11, -1, 8, 2, 3, -1, 7, 1, 0, -1, 6, 5, 4, -1 },
|
|
|
|
{ 6, 7, 8, 9, 5, 1, 2, 10, 4, 0, 3, 11, -1, -1, -1, -1 },
|
|
|
|
{ -1, 4, 5, 6, -1, 0, 1, 7, -1, 3, 2, 8, -1, 11, 10, 9 } };
|
|
|
|
permutation = permuteBoundary[bIndex];
|
|
|
|
level->gatherQuadRegularBoundaryPatchPoints(
|
|
|
|
patch.faceIndex, patchVerts, bIndex, refinerChannel);
|
|
|
|
} else if (patch.tag._boundaryCount == 2) {
|
|
|
|
// Expand corner patch vertices and rotate to
|
|
|
|
// restore correct orientation.
|
|
|
|
static int const permuteCorner[4][16] = {
|
|
|
|
{ -1, -1, -1, -1, -1, 0, 1, 4, -1, 3, 2, 5, -1, 8, 7, 6 },
|
|
|
|
{ -1, -1, -1, -1, 8, 3, 0, -1, 7, 2, 1, -1, 6, 5, 4, -1 },
|
|
|
|
{ 6, 7, 8, -1, 5, 2, 3, -1, 4, 1, 0, -1, -1, -1, -1, -1 },
|
|
|
|
{ -1, 4, 5, 6, -1, 1, 2, 7, -1, 0, 3, 8, -1, -1, -1, -1 } };
|
|
|
|
permutation = permuteCorner[bIndex];
|
|
|
|
level->gatherQuadRegularCornerPatchPoints(
|
|
|
|
patch.faceIndex, patchVerts, bIndex, refinerChannel);
|
|
|
|
} else {
|
|
|
|
assert(patch.tag._boundaryCount <= 2);
|
|
|
|
}
|
2015-02-26 21:57:47 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
offsetAndPermuteIndices(
|
|
|
|
patchVerts, 16, levelVertOffset, permutation, iptrs);
|
|
|
|
return 16;
|
|
|
|
}
|
2015-02-26 21:57:47 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
template <class END_CAP_FACTORY_TYPE>
|
|
|
|
int
|
|
|
|
PatchTableFactory::BuilderContext::
|
|
|
|
gatherEndCapPatchPoints(
|
|
|
|
END_CAP_FACTORY_TYPE *endCapFactory,
|
|
|
|
Index * iptrs, PatchTuple const & patch, int fvarChannel) {
|
|
|
|
|
|
|
|
Vtr::internal::Level const * level = &refiner.getLevel(patch.levelIndex);
|
|
|
|
int levelVertOffset = (fvarChannel < 0)
|
|
|
|
? levelVertOffsets[patch.levelIndex]
|
|
|
|
: levelFVarValueOffsets[fvarChannel][patch.levelIndex];
|
2016-08-05 22:10:33 +00:00
|
|
|
int refinerChannel = (fvarChannel < 0)
|
|
|
|
? fvarChannel
|
|
|
|
: fvarChannelIndices[fvarChannel];
|
2016-07-22 01:47:44 +00:00
|
|
|
|
|
|
|
// identify relevant spans around the corner vertices for the irregular patches
|
|
|
|
// (this is just a stub for now -- leaving the span "size" to zero, as constructed,
|
|
|
|
// indicates to use the full neighborhood)...
|
|
|
|
Vtr::internal::Level::VSpan cornerSpans[4];
|
|
|
|
|
|
|
|
ConstIndexArray cvs = endCapFactory->GetPatchPoints(
|
2016-08-05 22:10:33 +00:00
|
|
|
level, patch.faceIndex, cornerSpans, levelVertOffset, refinerChannel);
|
2016-07-22 01:47:44 +00:00
|
|
|
|
|
|
|
for (int i = 0; i < cvs.size(); ++i) iptrs[i] = cvs[i];
|
|
|
|
return cvs.size();
|
|
|
|
}
|
2015-02-26 21:57:47 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
bool
|
|
|
|
PatchTableFactory::computePatchTag(
|
|
|
|
BuilderContext & context,
|
|
|
|
Index const levelIndex, Index const faceIndex,
|
|
|
|
PatchTableFactory::PatchFaceTag &patchTag) {
|
2015-02-26 21:57:47 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
TopologyRefiner const & refiner = context.refiner;
|
2015-02-26 21:57:47 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
Vtr::internal::Level const * level = &refiner.getLevel(levelIndex);
|
2015-02-26 21:57:47 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
if (level->isFaceHole(faceIndex)) {
|
|
|
|
return false;
|
|
|
|
}
|
2015-02-26 21:57:47 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
//
|
|
|
|
// Given components at Level[i], we need to be looking at Refinement[i] -- and not
|
|
|
|
// [i-1] -- because the Refinement has transitional information for its parent edges
|
|
|
|
// and faces.
|
|
|
|
//
|
|
|
|
// For components in this level, we want to determine:
|
|
|
|
// - what Edges are "transitional" (already done in Refinement for parent)
|
|
|
|
// - what Faces are "transitional" (already done in Refinement for parent)
|
|
|
|
// - what Faces are "complete" (applied to this Level in previous refinement)
|
|
|
|
//
|
|
|
|
Vtr::internal::Refinement const * refinement =
|
|
|
|
(levelIndex < refiner.GetMaxLevel())
|
|
|
|
? refinement = &refiner.getRefinement(levelIndex) : 0;
|
2015-02-26 21:57:47 +00:00
|
|
|
|
|
|
|
//
|
2016-07-22 01:47:44 +00:00
|
|
|
// This face does not warrant a patch under the following conditions:
|
|
|
|
//
|
|
|
|
// - the face was fully refined into child faces
|
|
|
|
// - the face is not a quad (should have been refined, so assert)
|
|
|
|
// - the face is not "complete"
|
2015-02-26 21:57:47 +00:00
|
|
|
//
|
2016-07-22 01:47:44 +00:00
|
|
|
// The first is trivially determined, and the second is really redundant. The
|
|
|
|
// last -- "incompleteness" -- indicates a face that exists to support the limit
|
|
|
|
// of some neighboring component, and which does not have its own neighborhood
|
|
|
|
// fully defined for its limit. If any child vertex of a vertex of this face is
|
|
|
|
// "incomplete" (and all are tagged) the face must be "incomplete", so get the
|
|
|
|
// "composite" tag which combines bits for all vertices:
|
|
|
|
//
|
|
|
|
Vtr::internal::Refinement::SparseTag refinedFaceTag =
|
|
|
|
refinement
|
|
|
|
? refinement->getParentFaceSparseTag(faceIndex)
|
|
|
|
: Vtr::internal::Refinement::SparseTag();
|
2015-02-26 21:57:47 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
if (refinedFaceTag._selected) {
|
|
|
|
return false;
|
|
|
|
}
|
2015-02-26 21:57:47 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
Vtr::ConstIndexArray fVerts = level->getFaceVertices(faceIndex);
|
|
|
|
assert(fVerts.size() == 4);
|
2015-02-26 21:57:47 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
Vtr::internal::Level::VTag compFaceVertTag = level->getFaceCompositeVTag(fVerts);
|
|
|
|
if (compFaceVertTag._incomplete) {
|
|
|
|
return false;
|
|
|
|
}
|
2015-02-26 21:57:47 +00:00
|
|
|
|
|
|
|
//
|
2016-07-22 01:47:44 +00:00
|
|
|
// We have a quad that will be represented as a B-spline or end cap patch. Use
|
|
|
|
// the "composite" tag again to quickly determine if any vertex is irregular, on
|
|
|
|
// a boundary, non-manifold, etc.
|
|
|
|
//
|
|
|
|
// Inspect the edges for boundaries and transitional edges and pack results into
|
|
|
|
// 4-bit masks. We detect boundary edges rather than vertices as we hope to
|
|
|
|
// replace the mask in future with one for infinitely sharp edges -- allowing
|
|
|
|
// us to detect regular patches and avoid isolation. We still need to account
|
|
|
|
// for the irregular/xordinary case when a corner vertex is a boundary but there
|
|
|
|
// are no boundary edges.
|
|
|
|
//
|
|
|
|
// As for transition detection, assign the transition properties (even if 0).
|
2015-02-26 21:57:47 +00:00
|
|
|
//
|
2016-07-22 01:47:44 +00:00
|
|
|
// NOTE on patches around non-manifold vertices:
|
|
|
|
// In most cases the use of regular boundary or corner patches is what we want,
|
|
|
|
// but in some, i.e. when a non-manifold vertex is infinitely sharp, using
|
|
|
|
// such patches will create some discontinuities. At this point non-manifold
|
|
|
|
// support is still evolving and is not strictly defined, so this is left to
|
|
|
|
// a later date to resolve.
|
|
|
|
//
|
|
|
|
// NOTE on infinitely sharp (hard) edges:
|
|
|
|
// We should be able to adapt this later to detect hard (inf-sharp) edges
|
|
|
|
// rather than just boundary edges -- there is a similar tag per edge. That
|
|
|
|
// should allow us to generate regular patches for interior hard features.
|
|
|
|
//
|
|
|
|
bool hasBoundaryVertex = compFaceVertTag._boundary;
|
|
|
|
bool hasNonManifoldVertex = compFaceVertTag._nonManifold;
|
|
|
|
bool hasXOrdinaryVertex = compFaceVertTag._xordinary;
|
|
|
|
|
|
|
|
patchTag._isRegular = ! hasXOrdinaryVertex || hasNonManifoldVertex;
|
|
|
|
|
|
|
|
// single crease patch optimization
|
|
|
|
if (context.options.useSingleCreasePatch &&
|
|
|
|
! hasXOrdinaryVertex && ! hasBoundaryVertex && ! hasNonManifoldVertex) {
|
|
|
|
|
|
|
|
Vtr::ConstIndexArray fEdges = level->getFaceEdges(faceIndex);
|
|
|
|
Vtr::internal::Level::ETag compFaceETag = level->getFaceCompositeETag(fEdges);
|
|
|
|
|
|
|
|
if (compFaceETag._semiSharp || compFaceETag._infSharp) {
|
|
|
|
float sharpness = 0;
|
|
|
|
int rotation = 0;
|
|
|
|
if (level->isSingleCreasePatch(faceIndex, &sharpness, &rotation)) {
|
|
|
|
|
|
|
|
// cap sharpness to the max isolation level
|
|
|
|
float cappedSharpness =
|
|
|
|
std::min(sharpness, (float)(context.options.maxIsolationLevel - levelIndex));
|
|
|
|
if (cappedSharpness > 0) {
|
|
|
|
patchTag._isSingleCrease = true;
|
|
|
|
patchTag._boundaryIndex = rotation;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-02-26 21:57:47 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
// Identify boundaries for both regular and xordinary patches -- non-manifold
|
|
|
|
// (infinitely sharp) edges and vertices are currently interpreted as boundaries
|
|
|
|
// for regular patches, though an irregular patch or extrapolated boundary patch
|
|
|
|
// is really necessary in future for some non-manifold cases.
|
|
|
|
//
|
|
|
|
if (hasBoundaryVertex || hasNonManifoldVertex) {
|
|
|
|
Vtr::ConstIndexArray fEdges = level->getFaceEdges(faceIndex);
|
|
|
|
|
|
|
|
int boundaryEdgeMask = ((level->getEdgeTag(fEdges[0])._boundary) << 0) |
|
|
|
|
((level->getEdgeTag(fEdges[1])._boundary) << 1) |
|
|
|
|
((level->getEdgeTag(fEdges[2])._boundary) << 2) |
|
|
|
|
((level->getEdgeTag(fEdges[3])._boundary) << 3);
|
|
|
|
if (hasNonManifoldVertex) {
|
|
|
|
int nonManEdgeMask = ((level->getEdgeTag(fEdges[0])._nonManifold) << 0) |
|
|
|
|
((level->getEdgeTag(fEdges[1])._nonManifold) << 1) |
|
|
|
|
((level->getEdgeTag(fEdges[2])._nonManifold) << 2) |
|
|
|
|
((level->getEdgeTag(fEdges[3])._nonManifold) << 3);
|
|
|
|
|
|
|
|
// Other than non-manifold edges, non-manifold vertices that were made
|
|
|
|
// sharp should also trigger new "boundary" edges for the sharp corner
|
|
|
|
// patches introduced in these cases.
|
|
|
|
//
|
|
|
|
if (level->getVertexTag(fVerts[0])._nonManifold &&
|
|
|
|
level->getVertexTag(fVerts[0])._infSharp) {
|
|
|
|
nonManEdgeMask |= (1 << 0) | (1 << 3);
|
|
|
|
}
|
|
|
|
if (level->getVertexTag(fVerts[1])._nonManifold &&
|
|
|
|
level->getVertexTag(fVerts[1])._infSharp) {
|
|
|
|
nonManEdgeMask |= (1 << 1) | (1 << 0);
|
|
|
|
}
|
|
|
|
if (level->getVertexTag(fVerts[2])._nonManifold &&
|
|
|
|
level->getVertexTag(fVerts[2])._infSharp) {
|
|
|
|
nonManEdgeMask |= (1 << 2) | (1 << 1);
|
|
|
|
}
|
|
|
|
if (level->getVertexTag(fVerts[3])._nonManifold &&
|
|
|
|
level->getVertexTag(fVerts[3])._infSharp) {
|
|
|
|
nonManEdgeMask |= (1 << 3) | (1 << 2);
|
|
|
|
}
|
|
|
|
boundaryEdgeMask |= nonManEdgeMask;
|
|
|
|
}
|
2015-02-26 21:57:47 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
if (boundaryEdgeMask) {
|
|
|
|
patchTag.assignBoundaryPropertiesFromEdgeMask(boundaryEdgeMask);
|
|
|
|
} else {
|
|
|
|
int boundaryVertMask = ((level->getVertexTag(fVerts[0])._boundary) << 0) |
|
|
|
|
((level->getVertexTag(fVerts[1])._boundary) << 1) |
|
|
|
|
((level->getVertexTag(fVerts[2])._boundary) << 2) |
|
|
|
|
((level->getVertexTag(fVerts[3])._boundary) << 3);
|
|
|
|
|
|
|
|
if (hasNonManifoldVertex) {
|
|
|
|
int nonManVertMask = ((level->getVertexTag(fVerts[0])._nonManifold) << 0) |
|
|
|
|
((level->getVertexTag(fVerts[1])._nonManifold) << 1) |
|
|
|
|
((level->getVertexTag(fVerts[2])._nonManifold) << 2) |
|
|
|
|
((level->getVertexTag(fVerts[3])._nonManifold) << 3);
|
|
|
|
boundaryVertMask |= nonManVertMask;
|
|
|
|
}
|
|
|
|
patchTag.assignBoundaryPropertiesFromVertexMask(boundaryVertMask);
|
|
|
|
}
|
|
|
|
}
|
2015-02-26 21:57:47 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
// XXXX (barfowl) -- why are we approximating a smooth x-ordinary corner with
|
|
|
|
// a sharp corner patch? The boundary/corner points of the regular patch are
|
|
|
|
// not even made colinear to make it smoother. Something historical here...
|
|
|
|
//
|
|
|
|
// So this treatment may become optional in future and is bracketed with a
|
|
|
|
// condition now for that reason. We approximate x-ordinary smooth corners
|
|
|
|
// with regular B-spline patches instead of using a Gregory patch. The smooth
|
|
|
|
// corner must be properly isolated from any other irregular vertices (as it
|
|
|
|
// will be at any level > 1) otherwise the Gregory patch is necessary.
|
|
|
|
//
|
|
|
|
// This flag to be initialized with a future option... ?
|
|
|
|
bool approxSmoothCornerWithRegularPatch = true;
|
|
|
|
|
|
|
|
if (approxSmoothCornerWithRegularPatch) {
|
|
|
|
if (!patchTag._isRegular && (patchTag._boundaryCount == 2)) {
|
|
|
|
// We may have a sharp corner opposite/adjacent an xordinary vertex --
|
|
|
|
// need to make sure there is only one xordinary vertex and that it
|
|
|
|
// is the corner vertex.
|
|
|
|
if (levelIndex > 1) {
|
|
|
|
patchTag._isRegular = true;
|
|
|
|
} else {
|
|
|
|
int xordVertex = 0;
|
|
|
|
int xordCount = 0;
|
|
|
|
if (level->getVertexTag(fVerts[0])._xordinary) { xordCount++; xordVertex = 0; }
|
|
|
|
if (level->getVertexTag(fVerts[1])._xordinary) { xordCount++; xordVertex = 1; }
|
|
|
|
if (level->getVertexTag(fVerts[2])._xordinary) { xordCount++; xordVertex = 2; }
|
|
|
|
if (level->getVertexTag(fVerts[3])._xordinary) { xordCount++; xordVertex = 3; }
|
|
|
|
|
|
|
|
if (xordCount == 1) {
|
|
|
|
// We require the vertex opposite the xordinary vertex be interior:
|
|
|
|
if (! level->getVertexTag(fVerts[(xordVertex + 2) % 4])._boundary) {
|
|
|
|
patchTag._isRegular = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-02-26 21:57:47 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
//
|
|
|
|
// Now that all boundary features have have been identified and tagged, assign
|
|
|
|
// the transition type for the patch before taking inventory.
|
|
|
|
//
|
|
|
|
// Identify and increment counts for regular patches (both non-transitional and
|
|
|
|
// transitional) and extra-ordinary patches (always non-transitional):
|
|
|
|
//
|
|
|
|
patchTag._transitionMask = refinedFaceTag._transitional;
|
|
|
|
|
|
|
|
return true;
|
2015-02-26 21:57:47 +00:00
|
|
|
}
|
2014-09-05 22:07:46 +00:00
|
|
|
|
|
|
|
//
|
2015-05-22 18:50:01 +00:00
|
|
|
// Reserves tables based on the contents of the PatchArrayVector in the PatchTable:
|
2014-09-05 22:07:46 +00:00
|
|
|
//
|
|
|
|
void
|
2016-07-22 01:47:44 +00:00
|
|
|
PatchTableFactory::allocateVertexTables(
|
|
|
|
PatchTable * table, bool hasSharpness) {
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2014-11-18 18:48:37 +00:00
|
|
|
int ncvs = 0, npatches = 0;
|
2015-05-22 18:50:01 +00:00
|
|
|
for (int i=0; i<table->GetNumPatchArrays(); ++i) {
|
|
|
|
npatches += table->GetNumPatches(i);
|
|
|
|
ncvs += table->GetNumControlVertices(i);
|
2014-09-05 22:07:46 +00:00
|
|
|
}
|
|
|
|
|
2016-02-19 09:02:07 +00:00
|
|
|
if (ncvs==0 || npatches==0)
|
2014-09-05 22:07:46 +00:00
|
|
|
return;
|
|
|
|
|
2015-05-22 18:50:01 +00:00
|
|
|
table->_patchVerts.resize( ncvs );
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2015-05-22 18:50:01 +00:00
|
|
|
table->_paramTable.resize( npatches );
|
2014-10-13 15:52:09 +00:00
|
|
|
|
|
|
|
if (hasSharpness) {
|
2015-05-22 18:50:01 +00:00
|
|
|
table->_sharpnessIndices.resize( npatches, Vtr::INDEX_INVALID );
|
2014-10-13 15:52:09 +00:00
|
|
|
}
|
2014-09-05 22:07:46 +00:00
|
|
|
}
|
|
|
|
|
2015-02-26 21:57:47 +00:00
|
|
|
//
|
|
|
|
// Allocate face-varying tables
|
|
|
|
//
|
|
|
|
void
|
2016-07-22 01:47:44 +00:00
|
|
|
PatchTableFactory::allocateFVarChannels(
|
|
|
|
BuilderContext const & context, PatchTable * table) {
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
TopologyRefiner const & refiner = context.refiner;
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
int npatches = table->GetNumPatchesTotal();
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
table->allocateFVarPatchChannels((int)context.fvarChannelIndices.size());
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
// Initialize each channel
|
|
|
|
for (int fvc=0; fvc<(int)context.fvarChannelIndices.size(); ++fvc) {
|
|
|
|
int refinerChannel = context.fvarChannelIndices[fvc];
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2015-02-26 21:57:47 +00:00
|
|
|
Sdc::Options::FVarLinearInterpolation interpolation =
|
2016-07-22 01:47:44 +00:00
|
|
|
refiner.GetFVarLinearInterpolation(refinerChannel);
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
table->setFVarPatchChannelLinearInterpolation(interpolation, fvc);
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
PatchDescriptor::Type type = context.options.triangulateQuads ?
|
2015-06-02 21:58:35 +00:00
|
|
|
PatchDescriptor::TRIANGLES : PatchDescriptor::QUADS;
|
|
|
|
|
2016-07-19 23:39:33 +00:00
|
|
|
table->allocateFVarPatchChannelValues(
|
2016-07-22 01:47:44 +00:00
|
|
|
PatchDescriptor(type), npatches, fvc);
|
2015-02-26 21:57:47 +00:00
|
|
|
}
|
2014-09-05 22:07:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//
|
2015-06-11 20:18:25 +00:00
|
|
|
// Populates the PatchParam for the given face, returning
|
|
|
|
// a pointer to the next entry
|
2014-09-05 22:07:46 +00:00
|
|
|
//
|
2016-07-22 01:47:44 +00:00
|
|
|
PatchParam
|
2015-05-22 18:50:01 +00:00
|
|
|
PatchTableFactory::computePatchParam(
|
2016-07-22 01:47:44 +00:00
|
|
|
BuilderContext const & context,
|
2015-04-24 20:18:03 +00:00
|
|
|
int depth, Vtr::Index faceIndex, int boundaryMask,
|
2016-07-22 01:47:44 +00:00
|
|
|
int transitionMask) {
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
TopologyRefiner const & refiner = context.refiner;
|
2014-09-05 22:07:46 +00:00
|
|
|
|
|
|
|
// Move up the hierarchy accumulating u,v indices to the coarse level:
|
|
|
|
int childIndexInParent = 0,
|
|
|
|
u = 0,
|
|
|
|
v = 0,
|
|
|
|
ofs = 1;
|
|
|
|
|
2015-05-22 04:26:41 +00:00
|
|
|
bool nonquad = (refiner.GetLevel(depth).GetFaceVertices(faceIndex).size() != 4);
|
2014-09-05 22:07:46 +00:00
|
|
|
|
|
|
|
for (int i = depth; i > 0; --i) {
|
2015-05-26 03:34:50 +00:00
|
|
|
Vtr::internal::Refinement const& refinement = refiner.getRefinement(i-1);
|
|
|
|
Vtr::internal::Level const& parentLevel = refiner.getLevel(i-1);
|
2014-09-05 22:07:46 +00:00
|
|
|
|
|
|
|
Vtr::Index parentFaceIndex = refinement.getChildFaceParentFace(faceIndex);
|
|
|
|
childIndexInParent = refinement.getChildFaceInParentFace(faceIndex);
|
|
|
|
|
|
|
|
if (parentLevel.getFaceVertices(parentFaceIndex).size() == 4) {
|
|
|
|
switch ( childIndexInParent ) {
|
|
|
|
case 0 : break;
|
|
|
|
case 1 : { u+=ofs; } break;
|
|
|
|
case 2 : { u+=ofs; v+=ofs; } break;
|
|
|
|
case 3 : { v+=ofs; } break;
|
|
|
|
}
|
|
|
|
ofs = (unsigned short)(ofs << 1);
|
|
|
|
} else {
|
|
|
|
nonquad = true;
|
|
|
|
// If the root face is not a quad, we need to offset the ptex index
|
|
|
|
// CCW to match the correct child face
|
2014-12-15 18:23:13 +00:00
|
|
|
Vtr::ConstIndexArray children = refinement.getFaceChildFaces(parentFaceIndex);
|
2014-09-05 22:07:46 +00:00
|
|
|
for (int j=0; j<children.size(); ++j) {
|
|
|
|
if (children[j]==faceIndex) {
|
|
|
|
childIndexInParent = j;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
faceIndex = parentFaceIndex;
|
|
|
|
}
|
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
Vtr::Index ptexIndex = context.ptexIndices.GetFaceId(faceIndex);
|
2014-09-05 22:07:46 +00:00
|
|
|
assert(ptexIndex!=-1);
|
|
|
|
|
|
|
|
if (nonquad) {
|
|
|
|
ptexIndex+=childIndexInParent;
|
|
|
|
--depth;
|
|
|
|
}
|
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
PatchParam param;
|
|
|
|
param.Set(ptexIndex, (short)u, (short)v, (unsigned short) depth, nonquad,
|
2015-04-17 14:42:53 +00:00
|
|
|
(unsigned short) boundaryMask, (unsigned short) transitionMask);
|
2016-07-22 01:47:44 +00:00
|
|
|
return param;
|
2014-09-05 22:07:46 +00:00
|
|
|
}
|
|
|
|
|
2015-02-26 21:57:47 +00:00
|
|
|
//
|
|
|
|
// Indexing sharpnesses
|
|
|
|
//
|
|
|
|
inline int
|
|
|
|
assignSharpnessIndex(float sharpness, std::vector<float> & sharpnessValues) {
|
|
|
|
|
|
|
|
// linear search
|
|
|
|
for (int i=0; i<(int)sharpnessValues.size(); ++i) {
|
2015-07-21 00:56:00 +00:00
|
|
|
if (isSharpnessEqual(sharpnessValues[i], sharpness)) {
|
2015-02-26 21:57:47 +00:00
|
|
|
return i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
sharpnessValues.push_back(sharpness);
|
|
|
|
return (int)sharpnessValues.size()-1;
|
|
|
|
}
|
|
|
|
|
2015-04-23 23:58:45 +00:00
|
|
|
//
|
|
|
|
// We should be able to use a single Create() method for both the adaptive and uniform
|
|
|
|
// cases. In the past, more additional arguments were passed to the uniform version,
|
|
|
|
// but that may no longer be necessary (see notes in the uniform version below)...
|
|
|
|
//
|
2015-05-22 18:50:01 +00:00
|
|
|
PatchTable *
|
|
|
|
PatchTableFactory::Create(TopologyRefiner const & refiner, Options options) {
|
2014-09-05 22:07:46 +00:00
|
|
|
|
|
|
|
if (refiner.IsUniform()) {
|
|
|
|
return createUniform(refiner, options);
|
|
|
|
} else {
|
2015-04-23 23:58:45 +00:00
|
|
|
return createAdaptive(refiner, options);
|
2014-09-05 22:07:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-05-22 18:50:01 +00:00
|
|
|
PatchTable *
|
|
|
|
PatchTableFactory::createUniform(TopologyRefiner const & refiner, Options options) {
|
2014-09-05 22:07:46 +00:00
|
|
|
|
|
|
|
assert(refiner.IsUniform());
|
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
BuilderContext context(refiner, options);
|
|
|
|
|
2014-09-17 18:17:25 +00:00
|
|
|
// ensure that triangulateQuads is only set for quadrilateral schemes
|
2016-02-19 09:02:07 +00:00
|
|
|
options.triangulateQuads &= (refiner.GetSchemeType()==Sdc::SCHEME_BILINEAR ||
|
2015-01-07 01:40:11 +00:00
|
|
|
refiner.GetSchemeType()==Sdc::SCHEME_CATMARK);
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2015-07-29 21:42:03 +00:00
|
|
|
// level=0 may contain n-gons, which are not supported in PatchTable.
|
|
|
|
// even if generateAllLevels = true, we start from level 1.
|
|
|
|
|
2015-04-02 03:44:33 +00:00
|
|
|
int maxvalence = refiner.GetMaxValence(),
|
2014-09-05 22:07:46 +00:00
|
|
|
maxlevel = refiner.GetMaxLevel(),
|
2015-07-29 21:42:03 +00:00
|
|
|
firstlevel = options.generateAllLevels ? 1 : maxlevel,
|
2014-11-25 20:41:19 +00:00
|
|
|
nlevels = maxlevel-firstlevel+1;
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2014-11-25 20:41:19 +00:00
|
|
|
PatchDescriptor::Type ptype = PatchDescriptor::NON_PATCH;
|
2014-09-11 23:55:39 +00:00
|
|
|
if (options.triangulateQuads) {
|
2014-11-25 20:41:19 +00:00
|
|
|
ptype = PatchDescriptor::TRIANGLES;
|
2014-09-11 23:55:39 +00:00
|
|
|
} else {
|
|
|
|
switch (refiner.GetSchemeType()) {
|
2015-01-07 01:40:11 +00:00
|
|
|
case Sdc::SCHEME_BILINEAR :
|
|
|
|
case Sdc::SCHEME_CATMARK : ptype = PatchDescriptor::QUADS; break;
|
|
|
|
case Sdc::SCHEME_LOOP : ptype = PatchDescriptor::TRIANGLES; break;
|
2014-09-11 23:55:39 +00:00
|
|
|
}
|
2014-09-05 22:07:46 +00:00
|
|
|
}
|
2014-11-25 20:41:19 +00:00
|
|
|
assert(ptype!=PatchDescriptor::NON_PATCH);
|
2014-09-05 22:07:46 +00:00
|
|
|
|
|
|
|
//
|
2015-05-22 18:50:01 +00:00
|
|
|
// Create the instance of the table and allocate and initialize its members.
|
|
|
|
PatchTable * table = new PatchTable(maxvalence);
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
table->_numPtexFaces = context.ptexIndices.GetNumFaces();
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2015-05-22 18:50:01 +00:00
|
|
|
table->reservePatchArrays(nlevels);
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2015-04-17 14:42:53 +00:00
|
|
|
PatchDescriptor desc(ptype);
|
2014-09-05 22:07:46 +00:00
|
|
|
|
|
|
|
// generate patch arrays
|
|
|
|
for (int level=firstlevel, poffset=0, voffset=0; level<=maxlevel; ++level) {
|
|
|
|
|
2015-05-22 04:26:41 +00:00
|
|
|
TopologyLevel const & refLevel = refiner.GetLevel(level);
|
|
|
|
|
|
|
|
int npatches = refLevel.GetNumFaces();
|
2014-10-24 20:52:40 +00:00
|
|
|
if (refiner.HasHoles()) {
|
2015-05-22 04:26:41 +00:00
|
|
|
for (int i = npatches - 1; i >= 0; --i) {
|
|
|
|
npatches -= refLevel.IsFaceHole(i);
|
|
|
|
}
|
2014-10-24 20:52:40 +00:00
|
|
|
}
|
|
|
|
assert(npatches>=0);
|
2014-11-11 19:27:25 +00:00
|
|
|
|
2014-09-11 23:55:39 +00:00
|
|
|
if (options.triangulateQuads)
|
2014-09-05 22:07:46 +00:00
|
|
|
npatches *= 2;
|
2014-09-11 23:55:39 +00:00
|
|
|
|
2015-07-29 21:42:03 +00:00
|
|
|
table->pushPatchArray(desc, npatches, &voffset, &poffset, 0);
|
2014-09-05 22:07:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Allocate various tables
|
2016-07-22 01:47:44 +00:00
|
|
|
allocateVertexTables( table, /*hasSharpness=*/false );
|
|
|
|
|
|
|
|
if (context.RequiresFVarPatches()) {
|
|
|
|
allocateFVarChannels(context, table);
|
2014-09-05 22:07:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Now populate the patches:
|
|
|
|
//
|
2015-02-26 21:57:47 +00:00
|
|
|
|
2015-05-22 18:50:01 +00:00
|
|
|
Index * iptr = &table->_patchVerts[0];
|
|
|
|
PatchParam * pptr = &table->_paramTable[0];
|
2014-10-09 21:48:50 +00:00
|
|
|
Index ** fptr = 0;
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2015-07-29 21:42:03 +00:00
|
|
|
// we always skip level=0 vertices (control cages)
|
|
|
|
Index levelVertOffset = refiner.GetLevel(0).GetNumVertices();
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2014-10-09 21:48:50 +00:00
|
|
|
Index * levelFVarVertOffsets = 0;
|
2016-07-22 01:47:44 +00:00
|
|
|
if (context.RequiresFVarPatches()) {
|
2015-02-26 21:57:47 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
levelFVarVertOffsets = (Index *)alloca(context.fvarChannelIndices.size()*sizeof(Index));
|
|
|
|
memset(levelFVarVertOffsets, 0, context.fvarChannelIndices.size()*sizeof(Index));
|
2015-02-26 21:57:47 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
fptr = (Index **)alloca(context.fvarChannelIndices.size()*sizeof(Index *));
|
|
|
|
for (int fvc=0; fvc<(int)context.fvarChannelIndices.size(); ++fvc) {
|
|
|
|
fptr[fvc] = table->getFVarValues(fvc).begin();
|
2015-02-26 21:57:47 +00:00
|
|
|
}
|
2014-09-05 22:07:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for (int level=1; level<=maxlevel; ++level) {
|
|
|
|
|
2015-05-22 04:26:41 +00:00
|
|
|
TopologyLevel const & refLevel = refiner.GetLevel(level);
|
|
|
|
|
|
|
|
int nfaces = refLevel.GetNumFaces();
|
2014-09-05 22:07:46 +00:00
|
|
|
if (level>=firstlevel) {
|
|
|
|
for (int face=0; face<nfaces; ++face) {
|
2014-11-11 19:27:25 +00:00
|
|
|
|
2016-02-19 09:02:07 +00:00
|
|
|
if (refiner.HasHoles() && refLevel.IsFaceHole(face)) {
|
2014-10-24 20:52:40 +00:00
|
|
|
continue;
|
|
|
|
}
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2015-05-22 04:26:41 +00:00
|
|
|
ConstIndexArray fverts = refLevel.GetFaceVertices(face);
|
2014-09-05 22:07:46 +00:00
|
|
|
for (int vert=0; vert<fverts.size(); ++vert) {
|
|
|
|
*iptr++ = levelVertOffset + fverts[vert];
|
|
|
|
}
|
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
*pptr++ = computePatchParam(context, level, face, /*boundary*/0, /*transition*/0);
|
|
|
|
|
|
|
|
if (context.RequiresFVarPatches()) {
|
|
|
|
for (int fvc=0; fvc<(int)context.fvarChannelIndices.size(); ++fvc) {
|
|
|
|
int refinerChannel = context.fvarChannelIndices[fvc];
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
ConstIndexArray fvalues = refLevel.GetFaceFVarValues(face, refinerChannel);
|
2015-03-05 21:09:27 +00:00
|
|
|
for (int vert=0; vert<fvalues.size(); ++vert) {
|
2016-07-22 01:47:44 +00:00
|
|
|
assert((levelVertOffset + fvalues[vert]) < (int)table->getFVarValues(fvc).size());
|
|
|
|
fptr[fvc][vert] = levelFVarVertOffsets[fvc] + fvalues[vert];
|
2015-02-26 21:57:47 +00:00
|
|
|
}
|
2016-07-22 01:47:44 +00:00
|
|
|
fptr[fvc]+=fvalues.size();
|
2015-02-26 21:57:47 +00:00
|
|
|
}
|
2014-09-05 22:07:46 +00:00
|
|
|
}
|
|
|
|
|
2014-09-11 23:55:39 +00:00
|
|
|
if (options.triangulateQuads) {
|
2014-09-05 22:07:46 +00:00
|
|
|
// Triangulate the quadrilateral: {v0,v1,v2,v3} -> {v0,v1,v2},{v3,v0,v2}.
|
|
|
|
*iptr = *(iptr - 4); // copy v0 index
|
|
|
|
++iptr;
|
|
|
|
*iptr = *(iptr - 3); // copy v2 index
|
|
|
|
++iptr;
|
|
|
|
|
|
|
|
*pptr = *(pptr - 1); // copy first patch param
|
|
|
|
++pptr;
|
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
if (context.RequiresFVarPatches()) {
|
|
|
|
for (int fvc=0; fvc<(int)context.fvarChannelIndices.size(); ++fvc) {
|
|
|
|
*fptr[fvc] = *(fptr[fvc]-4); // copy fv0 index
|
|
|
|
++fptr[fvc];
|
|
|
|
*fptr[fvc] = *(fptr[fvc]-3); // copy fv2 index
|
|
|
|
++fptr[fvc];
|
2014-09-05 22:07:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (options.generateAllLevels) {
|
2015-05-22 04:26:41 +00:00
|
|
|
levelVertOffset += refiner.GetLevel(level).GetNumVertices();
|
2016-07-22 01:47:44 +00:00
|
|
|
for (int fvc=0; fvc<(int)context.fvarChannelIndices.size(); ++fvc) {
|
|
|
|
int refinerChannel = context.fvarChannelIndices[fvc];
|
|
|
|
levelFVarVertOffsets[fvc] += refiner.GetLevel(level).GetNumFVarValues(refinerChannel);
|
2014-09-05 22:07:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-05-22 18:50:01 +00:00
|
|
|
return table;
|
2014-09-05 22:07:46 +00:00
|
|
|
}
|
|
|
|
|
2015-05-22 18:50:01 +00:00
|
|
|
PatchTable *
|
|
|
|
PatchTableFactory::createAdaptive(TopologyRefiner const & refiner, Options options) {
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2016-02-19 09:02:07 +00:00
|
|
|
assert(! refiner.IsUniform());
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
BuilderContext context(refiner, options);
|
2015-02-26 21:57:47 +00:00
|
|
|
|
2014-09-05 22:07:46 +00:00
|
|
|
//
|
2016-07-22 01:47:44 +00:00
|
|
|
// First identify the patches -- accumulating an inventory of
|
|
|
|
// information about each resulting patch:
|
2014-09-05 22:07:46 +00:00
|
|
|
//
|
2015-04-23 23:58:45 +00:00
|
|
|
identifyAdaptivePatches(context);
|
2014-09-05 22:07:46 +00:00
|
|
|
|
|
|
|
//
|
2016-07-22 01:47:44 +00:00
|
|
|
// Create and initialize the instance of the table:
|
2014-09-05 22:07:46 +00:00
|
|
|
//
|
2015-04-02 03:44:33 +00:00
|
|
|
int maxValence = refiner.GetMaxValence();
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
PatchTable * table = new PatchTable(maxValence);
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
table->_numPtexFaces = context.ptexIndices.GetNumFaces();
|
2014-09-05 22:07:46 +00:00
|
|
|
|
|
|
|
//
|
|
|
|
// Now populate the patches:
|
|
|
|
//
|
2016-07-22 01:47:44 +00:00
|
|
|
populateAdaptivePatches(context, table);
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
return table;
|
2014-09-05 22:07:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Identify all patches required for faces at all levels -- accumulating the number of patches
|
|
|
|
// for each type, and retaining enough information for the patch for each face to populate it
|
|
|
|
// later with no additional analysis.
|
|
|
|
//
|
|
|
|
void
|
2016-07-22 01:47:44 +00:00
|
|
|
PatchTableFactory::identifyAdaptivePatches(BuilderContext & context) {
|
2015-02-26 21:57:47 +00:00
|
|
|
|
|
|
|
TopologyRefiner const & refiner = context.refiner;
|
|
|
|
|
2014-09-05 22:07:46 +00:00
|
|
|
//
|
|
|
|
// Iterate through the levels of refinement to inspect and tag components with information
|
|
|
|
// relative to patch generation. We allocate all of the tags locally and use them to
|
|
|
|
// populate the patches once a complete inventory has been taken and all tables appropriately
|
|
|
|
// allocated and initialized:
|
|
|
|
//
|
|
|
|
// The first Level may have no Refinement if it is the only level -- similarly the last Level
|
|
|
|
// has no Refinement, so a single level is effectively the last, but with less information
|
|
|
|
// available in some cases, as it was not generated by refinement.
|
|
|
|
//
|
2016-07-22 01:47:44 +00:00
|
|
|
int reservePatches = refiner.GetNumFacesTotal();
|
|
|
|
context.patches.reserve(reservePatches);
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
context.levelVertOffsets.push_back(0);
|
|
|
|
context.levelFVarValueOffsets.resize(context.fvarChannelIndices.size());
|
|
|
|
for (int fvc=0; fvc<(int)context.fvarChannelIndices.size(); ++fvc) {
|
|
|
|
context.levelFVarValueOffsets[fvc].push_back(0);
|
|
|
|
}
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
for (int levelIndex=0; levelIndex<refiner.GetNumLevels(); ++levelIndex) {
|
2015-05-26 03:34:50 +00:00
|
|
|
Vtr::internal::Level const * level = &refiner.getLevel(levelIndex);
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
context.levelVertOffsets.push_back(
|
|
|
|
context.levelVertOffsets.back() + level->getNumVertices());
|
|
|
|
|
|
|
|
for (int fvc=0; fvc<(int)context.fvarChannelIndices.size(); ++fvc) {
|
|
|
|
int refinerChannel = context.fvarChannelIndices[fvc];
|
|
|
|
context.levelFVarValueOffsets[fvc].push_back(
|
|
|
|
context.levelFVarValueOffsets[fvc].back()
|
|
|
|
+ level->getNumFVarValues(refinerChannel));
|
2015-03-26 17:42:32 +00:00
|
|
|
}
|
2014-09-05 22:07:46 +00:00
|
|
|
|
|
|
|
for (int faceIndex = 0; faceIndex < level->getNumFaces(); ++faceIndex) {
|
2014-11-11 19:27:25 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
PatchFaceTag patchTag;
|
2014-09-05 22:07:46 +00:00
|
|
|
patchTag.clear();
|
2015-03-26 17:42:32 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
if (! computePatchTag(context, levelIndex, faceIndex, patchTag)) {
|
2014-09-05 22:07:46 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
context.patches.push_back(
|
|
|
|
BuilderContext::PatchTuple(patchTag, faceIndex, levelIndex));
|
2015-03-26 17:42:32 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
// Count the patches here to simplify subsequent allocation.
|
2014-09-05 22:07:46 +00:00
|
|
|
if (patchTag._isRegular) {
|
2016-07-22 01:47:44 +00:00
|
|
|
++context.numRegularPatches;
|
2014-09-05 22:07:46 +00:00
|
|
|
} else {
|
2016-07-22 01:47:44 +00:00
|
|
|
++context.numIrregularPatches;
|
|
|
|
// For legacy gregory patches we need to know how many
|
|
|
|
// irregular patches are also boundary patches.
|
|
|
|
if (patchTag._boundaryCount > 0) {
|
|
|
|
++context.numIrregularBoundaryPatches;
|
2014-09-05 22:07:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
2016-07-22 01:47:44 +00:00
|
|
|
// Populate adaptive patches that we've previously identified.
|
2014-09-05 22:07:46 +00:00
|
|
|
//
|
|
|
|
void
|
2015-05-22 18:50:01 +00:00
|
|
|
PatchTableFactory::populateAdaptivePatches(
|
2016-07-22 01:47:44 +00:00
|
|
|
BuilderContext & context, PatchTable * table) {
|
2015-02-26 21:57:47 +00:00
|
|
|
|
|
|
|
TopologyRefiner const & refiner = context.refiner;
|
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
// State needed to populate an array in the patch table.
|
|
|
|
// Pointers in this structure are initialized after the patch array
|
|
|
|
// data buffers have been allocated and are then incremented as we
|
|
|
|
// populate data into the patch table. Currently, we'll have at
|
|
|
|
// most 3 patch arrays: Regular, Irregular, and IrregularBoundary.
|
|
|
|
struct PatchArrayBuilder {
|
|
|
|
PatchArrayBuilder()
|
|
|
|
: patchType(PatchDescriptor::REGULAR), numPatches(0)
|
|
|
|
, iptr(NULL), pptr(NULL), sptr(NULL) { }
|
|
|
|
|
|
|
|
PatchDescriptor::Type patchType;
|
|
|
|
int numPatches;
|
|
|
|
|
|
|
|
Far::Index *iptr;
|
|
|
|
Far::PatchParam *pptr;
|
|
|
|
Far::Index *sptr;
|
|
|
|
Vtr::internal::StackBuffer<Far::Index*,1> fptr;
|
|
|
|
|
|
|
|
private:
|
|
|
|
// Non-copyable
|
|
|
|
PatchArrayBuilder(PatchArrayBuilder const &) {}
|
|
|
|
PatchArrayBuilder & operator=(PatchArrayBuilder const &) {return *this;}
|
|
|
|
|
|
|
|
} arrayBuilders[3];
|
|
|
|
int R = 0, IR = 1, IRB = 2; // Regular, Irregular, IrregularBoundary
|
|
|
|
|
2016-08-06 02:00:51 +00:00
|
|
|
// Regular patches patches will be packed into the first patch array
|
2016-07-22 01:47:44 +00:00
|
|
|
arrayBuilders[R].patchType = PatchDescriptor::REGULAR;
|
|
|
|
arrayBuilders[R].numPatches = context.numRegularPatches;
|
2016-08-06 02:00:51 +00:00
|
|
|
int numPatchArrays = (arrayBuilders[R].numPatches > 0);
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
switch(context.options.GetEndCapType()) {
|
|
|
|
case Options::ENDCAP_BSPLINE_BASIS:
|
|
|
|
// Irregular patches are converted to bspline basis and
|
|
|
|
// will be packed into the same patch array as regular patches
|
|
|
|
IR = IRB = R;
|
|
|
|
arrayBuilders[R].numPatches += context.numIrregularPatches;
|
2016-08-06 02:00:51 +00:00
|
|
|
// Make sure we've counted this array even when
|
|
|
|
// there are no regular patches.
|
|
|
|
numPatchArrays = (arrayBuilders[R].numPatches > 0);
|
2016-07-22 01:47:44 +00:00
|
|
|
break;
|
|
|
|
case Options::ENDCAP_GREGORY_BASIS:
|
|
|
|
// Irregular patches (both interior and boundary) are converted
|
|
|
|
// to Gregory basis and will be packed into an additional patch array
|
|
|
|
IR = IRB = numPatchArrays;
|
|
|
|
arrayBuilders[IR].patchType = PatchDescriptor::GREGORY_BASIS;
|
|
|
|
arrayBuilders[IR].numPatches += context.numIrregularPatches;
|
|
|
|
numPatchArrays += (arrayBuilders[IR].numPatches > 0);
|
|
|
|
break;
|
|
|
|
case Options::ENDCAP_LEGACY_GREGORY:
|
|
|
|
// Irregular interior and irregular boundary patches each will be
|
|
|
|
// packed into separate additional patch arrays.
|
|
|
|
IR = numPatchArrays;
|
|
|
|
arrayBuilders[IR].patchType = PatchDescriptor::GREGORY;
|
|
|
|
arrayBuilders[IR].numPatches = context.numIrregularPatches
|
|
|
|
- context.numIrregularBoundaryPatches;
|
|
|
|
numPatchArrays += (arrayBuilders[IR].numPatches > 0);
|
|
|
|
|
|
|
|
IRB = numPatchArrays;
|
|
|
|
arrayBuilders[IRB].patchType = PatchDescriptor::GREGORY_BOUNDARY;
|
|
|
|
arrayBuilders[IRB].numPatches = context.numIrregularBoundaryPatches;
|
|
|
|
numPatchArrays += (arrayBuilders[IRB].numPatches > 0);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
// Create patch arrays
|
|
|
|
table->reservePatchArrays(numPatchArrays);
|
2014-10-09 21:48:50 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
int voffset=0, poffset=0, qoffset=0;
|
|
|
|
for (int arrayIndex=0; arrayIndex<numPatchArrays; ++arrayIndex) {
|
|
|
|
PatchArrayBuilder & arrayBuilder = arrayBuilders[arrayIndex];
|
|
|
|
table->pushPatchArray(PatchDescriptor(arrayBuilder.patchType),
|
|
|
|
arrayBuilder.numPatches, &voffset, &poffset, &qoffset );
|
|
|
|
}
|
2014-10-09 21:48:50 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
// Allocate patch array data buffers
|
|
|
|
bool hasSharpness = context.options.useSingleCreasePatch;
|
|
|
|
allocateVertexTables(table, hasSharpness);
|
2014-10-09 21:48:50 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
if (context.RequiresFVarPatches()) {
|
|
|
|
allocateFVarChannels(context, table);
|
|
|
|
}
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
// Initialize pointers used while populating patch array data buffers
|
|
|
|
for (int arrayIndex=0; arrayIndex<numPatchArrays; ++arrayIndex) {
|
|
|
|
PatchArrayBuilder & arrayBuilder = arrayBuilders[arrayIndex];
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
arrayBuilder.iptr = table->getPatchArrayVertices(arrayIndex).begin();
|
|
|
|
arrayBuilder.pptr = table->getPatchParams(arrayIndex).begin();
|
|
|
|
if (hasSharpness) {
|
|
|
|
arrayBuilder.sptr = table->getSharpnessIndices(arrayIndex);
|
2014-10-13 15:52:09 +00:00
|
|
|
}
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2015-02-26 21:57:47 +00:00
|
|
|
if (context.RequiresFVarPatches()) {
|
2016-07-22 01:47:44 +00:00
|
|
|
arrayBuilder.fptr.SetSize((int)context.fvarChannelIndices.size());
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
for (int fvc=0; fvc<(int)context.fvarChannelIndices.size(); ++fvc) {
|
2015-02-26 21:57:47 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
PatchDescriptor desc = table->GetFVarChannelPatchDescriptor(fvc);
|
2015-05-22 18:50:01 +00:00
|
|
|
Index pidx = table->getPatchIndex(arrayIndex, 0);
|
2016-07-22 01:47:44 +00:00
|
|
|
int ofs = pidx * desc.GetNumControlVertices();
|
|
|
|
arrayBuilder.fptr[fvc] = &table->getFVarValues(fvc)[ofs];
|
2014-09-05 22:07:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-04-23 23:58:45 +00:00
|
|
|
// endcap factories
|
|
|
|
// XXX
|
|
|
|
EndCapBSplineBasisPatchFactory *endCapBSpline = NULL;
|
|
|
|
EndCapGregoryBasisPatchFactory *endCapGregoryBasis = NULL;
|
|
|
|
EndCapLegacyGregoryPatchFactory *endCapLegacyGregory = NULL;
|
2015-10-08 01:53:02 +00:00
|
|
|
StencilTable *localPointStencils = NULL;
|
|
|
|
StencilTable *localPointVaryingStencils = NULL;
|
2015-04-23 23:58:45 +00:00
|
|
|
|
|
|
|
switch(context.options.GetEndCapType()) {
|
|
|
|
case Options::ENDCAP_GREGORY_BASIS:
|
2015-10-08 01:53:02 +00:00
|
|
|
localPointStencils = new StencilTable(0);
|
|
|
|
localPointVaryingStencils = new StencilTable(0);
|
2015-04-23 23:58:45 +00:00
|
|
|
endCapGregoryBasis = new EndCapGregoryBasisPatchFactory(
|
2015-10-08 01:53:02 +00:00
|
|
|
refiner,
|
|
|
|
localPointStencils,
|
|
|
|
localPointVaryingStencils,
|
|
|
|
context.options.shareEndCapPatchPoints);
|
2015-04-23 23:58:45 +00:00
|
|
|
break;
|
|
|
|
case Options::ENDCAP_BSPLINE_BASIS:
|
2015-10-08 01:53:02 +00:00
|
|
|
localPointStencils = new StencilTable(0);
|
|
|
|
localPointVaryingStencils = new StencilTable(0);
|
|
|
|
endCapBSpline = new EndCapBSplineBasisPatchFactory(
|
|
|
|
refiner,
|
|
|
|
localPointStencils,
|
|
|
|
localPointVaryingStencils);
|
2015-04-23 23:58:45 +00:00
|
|
|
break;
|
|
|
|
case Options::ENDCAP_LEGACY_GREGORY:
|
|
|
|
endCapLegacyGregory = new EndCapLegacyGregoryPatchFactory(refiner);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
// Populate patch data buffers
|
|
|
|
for (int patchIndex=0; patchIndex<(int)context.patches.size(); ++patchIndex) {
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
BuilderContext::PatchTuple const & patch = context.patches[patchIndex];
|
|
|
|
int boundaryMask = patch.tag._boundaryMask;
|
|
|
|
int transitionMask = patch.tag._transitionMask;
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
float sharpness = 0;
|
|
|
|
if (hasSharpness && patch.tag._isSingleCrease) {
|
|
|
|
Vtr::internal::Level const & level = refiner.getLevel(patch.levelIndex);
|
|
|
|
int bIndex = patch.tag._boundaryIndex;
|
|
|
|
boundaryMask = (1<<bIndex);
|
|
|
|
sharpness = level.getEdgeSharpness(
|
|
|
|
(level.getFaceEdges(patch.faceIndex)[bIndex]));
|
|
|
|
sharpness = std::min(sharpness,
|
|
|
|
(float)(context.options.maxIsolationLevel-patch.levelIndex));
|
|
|
|
}
|
2015-04-27 19:40:18 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
// Most patches will be packed into the regular patch array
|
|
|
|
PatchArrayBuilder * arrayBuilder = &arrayBuilders[R];
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
if (patch.tag._isRegular) {
|
|
|
|
arrayBuilder->iptr +=
|
|
|
|
context.gatherRegularPatchPoints(arrayBuilder->iptr, patch);
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
} else {
|
|
|
|
// Switch to building the irregular patch array
|
|
|
|
arrayBuilder = &arrayBuilders[IR];
|
|
|
|
|
|
|
|
boundaryMask = 0;
|
|
|
|
transitionMask = 0;
|
|
|
|
|
|
|
|
// switch endcap patchtype by option
|
|
|
|
switch(context.options.GetEndCapType()) {
|
|
|
|
case Options::ENDCAP_GREGORY_BASIS:
|
|
|
|
arrayBuilder->iptr +=
|
|
|
|
context.gatherEndCapPatchPoints(
|
|
|
|
endCapGregoryBasis, arrayBuilder->iptr, patch);
|
|
|
|
break;
|
|
|
|
case Options::ENDCAP_BSPLINE_BASIS:
|
|
|
|
arrayBuilder->iptr +=
|
|
|
|
context.gatherEndCapPatchPoints(
|
|
|
|
endCapBSpline, arrayBuilder->iptr, patch);
|
|
|
|
break;
|
|
|
|
case Options::ENDCAP_LEGACY_GREGORY:
|
|
|
|
// For legacy gregory patches we may need to switch to
|
|
|
|
// the irregular boundary patch array.
|
|
|
|
if (patch.tag._boundaryCount == 0) {
|
|
|
|
arrayBuilder->iptr +=
|
|
|
|
context.gatherEndCapPatchPoints(
|
|
|
|
endCapLegacyGregory, arrayBuilder->iptr, patch);
|
2014-09-05 22:07:46 +00:00
|
|
|
} else {
|
2016-07-22 01:47:44 +00:00
|
|
|
arrayBuilder = &arrayBuilders[IRB];
|
|
|
|
arrayBuilder->iptr +=
|
|
|
|
context.gatherEndCapPatchPoints(
|
|
|
|
endCapLegacyGregory, arrayBuilder->iptr, patch);
|
2015-06-02 04:18:35 +00:00
|
|
|
}
|
2016-07-22 01:47:44 +00:00
|
|
|
break;
|
|
|
|
case Options::ENDCAP_BILINEAR_BASIS:
|
|
|
|
// not implemented yet
|
|
|
|
assert(false);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
// no endcap
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
*arrayBuilder->pptr++ =
|
|
|
|
computePatchParam(context,
|
|
|
|
patch.levelIndex, patch.faceIndex,
|
|
|
|
boundaryMask, transitionMask);
|
2015-04-17 23:26:57 +00:00
|
|
|
|
2016-07-22 01:47:44 +00:00
|
|
|
if (hasSharpness) {
|
|
|
|
*arrayBuilder->sptr++ =
|
|
|
|
assignSharpnessIndex(sharpness, table->_sharpnessValues);
|
2014-09-05 22:07:46 +00:00
|
|
|
}
|
2016-07-22 01:47:44 +00:00
|
|
|
|
2015-02-26 21:57:47 +00:00
|
|
|
if (context.RequiresFVarPatches()) {
|
2016-07-22 01:47:44 +00:00
|
|
|
for (int fvc=0; fvc<(int)context.fvarChannelIndices.size(); ++fvc) {
|
|
|
|
// For now, fvar data is always bilinear. Eventually, we will
|
|
|
|
// inspect the fvar topology and emit patches accordingly.
|
|
|
|
BuilderContext::PatchTuple fvarPatch(patch);
|
|
|
|
arrayBuilder->fptr[fvc] +=
|
|
|
|
context.gatherBilinearPatchPoints(
|
|
|
|
arrayBuilder->fptr[fvc], fvarPatch, fvc);
|
2014-09-05 22:07:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-04-23 23:58:45 +00:00
|
|
|
// finalize end patches
|
2016-02-19 09:02:07 +00:00
|
|
|
if (localPointStencils && localPointStencils->GetNumStencils() > 0) {
|
2016-03-15 18:36:34 +00:00
|
|
|
localPointStencils->finalize();
|
2015-12-11 22:33:31 +00:00
|
|
|
} else {
|
|
|
|
delete localPointStencils;
|
|
|
|
localPointStencils = NULL;
|
|
|
|
}
|
|
|
|
|
2016-02-19 09:02:07 +00:00
|
|
|
if (localPointVaryingStencils && localPointVaryingStencils->GetNumStencils() > 0) {
|
2016-03-15 18:36:34 +00:00
|
|
|
localPointVaryingStencils->finalize();
|
2015-12-11 22:33:31 +00:00
|
|
|
} else {
|
|
|
|
delete localPointVaryingStencils;
|
|
|
|
localPointVaryingStencils = NULL;
|
|
|
|
}
|
2015-10-08 01:53:02 +00:00
|
|
|
|
2015-04-23 23:58:45 +00:00
|
|
|
switch(context.options.GetEndCapType()) {
|
|
|
|
case Options::ENDCAP_GREGORY_BASIS:
|
2015-10-08 01:53:02 +00:00
|
|
|
table->_localPointStencils = localPointStencils;
|
|
|
|
table->_localPointVaryingStencils = localPointVaryingStencils;
|
2015-04-23 23:58:45 +00:00
|
|
|
delete endCapGregoryBasis;
|
|
|
|
break;
|
|
|
|
case Options::ENDCAP_BSPLINE_BASIS:
|
2015-10-08 01:53:02 +00:00
|
|
|
table->_localPointStencils = localPointStencils;
|
|
|
|
table->_localPointVaryingStencils = localPointVaryingStencils;
|
2015-04-23 23:58:45 +00:00
|
|
|
delete endCapBSpline;
|
|
|
|
break;
|
|
|
|
case Options::ENDCAP_LEGACY_GREGORY:
|
2015-05-21 02:44:43 +00:00
|
|
|
endCapLegacyGregory->Finalize(
|
2015-05-22 18:50:01 +00:00
|
|
|
table->GetMaxValence(),
|
|
|
|
&table->_quadOffsetsTable,
|
|
|
|
&table->_vertexValenceTable);
|
2015-04-23 23:58:45 +00:00
|
|
|
delete endCapLegacyGregory;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2014-09-05 22:07:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
} // end namespace Far
|
|
|
|
|
|
|
|
} // end namespace OPENSUBDIV_VERSION
|
|
|
|
} // end namespace OpenSubdiv
|
2015-04-18 00:36:55 +00:00
|
|
|
|