mirror of
https://github.com/PixarAnimationStudios/OpenSubdiv
synced 2025-01-07 15:30:14 +00:00
e8ac41dbc8
this change fixes a crash when selecting catmark_hole_test1 in glStencilViewer. If there's a hole limit stencils may not be found for given limit location, resulting invalid stencil entries.
574 lines
20 KiB
C++
574 lines
20 KiB
C++
//
|
|
// 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.
|
|
//
|
|
|
|
#include "../far/stencilTablesFactory.h"
|
|
#include "../far/endCapGregoryBasisPatchFactory.h"
|
|
#include "../far/patchTables.h"
|
|
#include "../far/patchTablesFactory.h"
|
|
#include "../far/patchMap.h"
|
|
#include "../far/protoStencil.h"
|
|
#include "../far/topologyRefiner.h"
|
|
|
|
#include <cassert>
|
|
#include <algorithm>
|
|
|
|
namespace OpenSubdiv {
|
|
namespace OPENSUBDIV_VERSION {
|
|
|
|
namespace Far {
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
void
|
|
StencilTablesFactory::generateControlVertStencils(
|
|
int numControlVerts, Stencil & dst) {
|
|
|
|
// Control vertices contribute a single index with a weight of 1.0
|
|
for (int i=0; i<numControlVerts; ++i) {
|
|
*dst._size = 1;
|
|
*dst._indices = i;
|
|
*dst._weights = 1.0f;
|
|
dst.Next();
|
|
}
|
|
}
|
|
|
|
//
|
|
// StencilTables factory
|
|
//
|
|
StencilTables const *
|
|
StencilTablesFactory::Create(TopologyRefiner const & refiner,
|
|
Options options) {
|
|
|
|
StencilTables * result = new StencilTables;
|
|
|
|
// always initialize numControlVertices (useful for torus case)
|
|
result->_numControlVertices = refiner.GetNumVertices(0);
|
|
|
|
int maxlevel = std::min(int(options.maxLevel), refiner.GetMaxLevel());
|
|
if (maxlevel==0 and (not options.generateControlVerts)) {
|
|
return result;
|
|
}
|
|
|
|
// 'maxsize' reflects the size of the default supporting basis factorized
|
|
// in the stencils, with a little bit of head-room. Each subdivision scheme
|
|
// has a set valence for 'regular' vertices, which drives the size of the
|
|
// supporting basis of control-vertices. The goal is to reduce the number
|
|
// of incidences where the pool allocator has to switch to dynamically
|
|
// allocated heap memory when encountering extraordinary vertices that
|
|
// require a larger supporting basis.
|
|
//
|
|
// The maxsize settings we use follow the assumption that the vast
|
|
// majority of the vertices in a mesh are regular, and that the valence
|
|
// of the extraordinary vertices is only higher by 1 edge.
|
|
int maxsize = 0;
|
|
bool interpolateVarying = false;
|
|
switch (options.interpolationMode) {
|
|
case INTERPOLATE_VERTEX: {
|
|
Sdc::SchemeType type = refiner.GetSchemeType();
|
|
switch (type) {
|
|
case Sdc::SCHEME_BILINEAR : maxsize = 5; break;
|
|
case Sdc::SCHEME_CATMARK : maxsize = 17; break;
|
|
case Sdc::SCHEME_LOOP : maxsize = 10; break;
|
|
default:
|
|
assert(0);
|
|
}
|
|
} break;
|
|
case INTERPOLATE_VARYING: maxsize = 5; interpolateVarying=true; break;
|
|
default:
|
|
assert(0);
|
|
}
|
|
|
|
std::vector<StencilAllocator> allocators(
|
|
options.generateIntermediateLevels ? maxlevel+1 : 2,
|
|
StencilAllocator(maxsize, interpolateVarying));
|
|
|
|
StencilAllocator * srcAlloc = &allocators[0],
|
|
* dstAlloc = &allocators[1];
|
|
|
|
//
|
|
// Interpolate stencils for each refinement level using
|
|
// TopologyRefiner::InterpolateLevel<>()
|
|
//
|
|
for (int level=1;level<=maxlevel; ++level) {
|
|
|
|
dstAlloc->Resize(refiner.GetNumVertices(level));
|
|
|
|
if (options.interpolationMode==INTERPOLATE_VERTEX) {
|
|
refiner.Interpolate(level, *srcAlloc, *dstAlloc);
|
|
} else {
|
|
refiner.InterpolateVarying(level, *srcAlloc, *dstAlloc);
|
|
}
|
|
|
|
if (options.generateIntermediateLevels) {
|
|
if (level<maxlevel) {
|
|
if (options.factorizeIntermediateLevels) {
|
|
srcAlloc = &allocators[level];
|
|
} else {
|
|
// if the stencils are dependent on the previous level of
|
|
// subdivision, pass an empty allocator to treat all parent
|
|
// vertices as control vertices
|
|
assert(allocators[0].GetNumStencils()==0);
|
|
}
|
|
dstAlloc = &allocators[level+1];
|
|
}
|
|
} else {
|
|
std::swap(srcAlloc, dstAlloc);
|
|
}
|
|
}
|
|
|
|
// Copy stencils from the pool allocator into the tables
|
|
{
|
|
// Add total number of stencils, weights & indices
|
|
int nelems = 0, nstencils=0;
|
|
if (options.generateIntermediateLevels) {
|
|
for (int level=0; level<=maxlevel; ++level) {
|
|
nstencils += allocators[level].GetNumStencils();
|
|
nelems += allocators[level].GetNumVerticesTotal();
|
|
}
|
|
} else {
|
|
nstencils = (int)srcAlloc->GetNumStencils();
|
|
nelems = srcAlloc->GetNumVerticesTotal();
|
|
}
|
|
|
|
// Allocate
|
|
result->_numControlVertices = refiner.GetNumVertices(0);
|
|
|
|
if (options.generateControlVerts) {
|
|
nstencils += result->_numControlVertices;
|
|
nelems += result->_numControlVertices;
|
|
}
|
|
result->resize(nstencils, nelems);
|
|
|
|
// Copy stencils
|
|
Stencil dst(&result->_sizes.at(0),
|
|
&result->_indices.at(0), &result->_weights.at(0));
|
|
|
|
if (options.generateControlVerts) {
|
|
generateControlVertStencils(result->_numControlVertices, dst);
|
|
}
|
|
|
|
if (options.generateIntermediateLevels) {
|
|
for (int level=1; level<=maxlevel; ++level) {
|
|
for (int i=0; i<allocators[level].GetNumStencils(); ++i) {
|
|
*dst._size = allocators[level].CopyStencil(i, dst._indices, dst._weights);
|
|
dst.Next();
|
|
}
|
|
}
|
|
} else {
|
|
for (int i=0; i<srcAlloc->GetNumStencils(); ++i) {
|
|
*dst._size = srcAlloc->CopyStencil(i, dst._indices, dst._weights);
|
|
dst.Next();
|
|
}
|
|
}
|
|
|
|
if (options.generateOffsets) {
|
|
result->generateOffsets();
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
StencilTables const *
|
|
StencilTablesFactory::Create(int numTables, StencilTables const ** tables) {
|
|
|
|
// XXXtakahito:
|
|
// This function returns NULL for empty inputs or erroneous condition.
|
|
// It's convenient for skipping varying stencils etc, however,
|
|
// other Create() API returns an empty stencil instead of NULL.
|
|
// They need to be consistent.
|
|
|
|
if ( (numTables<=0) or (not tables)) {
|
|
return NULL;
|
|
}
|
|
|
|
int ncvs = -1,
|
|
nstencils = 0,
|
|
nelems = 0;
|
|
|
|
for (int i=0; i<numTables; ++i) {
|
|
|
|
StencilTables const * st = tables[i];
|
|
// allow the tables could have a null entry.
|
|
if (!st) continue;
|
|
|
|
if (ncvs >= 0 and st->GetNumControlVertices() != ncvs) {
|
|
return NULL;
|
|
}
|
|
ncvs = st->GetNumControlVertices();
|
|
nstencils += st->GetNumStencils();
|
|
nelems += (int)st->GetControlIndices().size();
|
|
}
|
|
|
|
if (ncvs == -1) {
|
|
return NULL;
|
|
}
|
|
|
|
StencilTables * result = new StencilTables;
|
|
result->resize(nstencils, nelems);
|
|
|
|
int * sizes = &result->_sizes[0];
|
|
Index * indices = &result->_indices[0];
|
|
float * weights = &result->_weights[0];
|
|
for (int i=0; i<numTables; ++i) {
|
|
StencilTables const * st = tables[i];
|
|
if (!st) continue;
|
|
|
|
int st_nstencils = st->GetNumStencils(),
|
|
st_nelems = (int)st->_indices.size();
|
|
memcpy(sizes, &st->_sizes[0], st_nstencils*sizeof(int));
|
|
memcpy(indices, &st->_indices[0], st_nelems*sizeof(Index));
|
|
memcpy(weights, &st->_weights[0], st_nelems*sizeof(float));
|
|
|
|
sizes += st_nstencils;
|
|
indices += st_nelems;
|
|
weights += st_nelems;
|
|
}
|
|
|
|
result->_numControlVertices = ncvs;
|
|
|
|
// have to re-generate offsets from scratch
|
|
result->generateOffsets();
|
|
|
|
return result;
|
|
}
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
StencilTables const *
|
|
StencilTablesFactory::AppendEndCapStencilTables(
|
|
TopologyRefiner const &refiner,
|
|
StencilTables const *baseStencilTables,
|
|
StencilTables const *endCapStencilTables,
|
|
bool factorize) {
|
|
|
|
// factorize and append.
|
|
if (baseStencilTables == NULL or
|
|
endCapStencilTables == NULL) return NULL;
|
|
|
|
// endcap stencils have indices that are relative to the level
|
|
// (maxlevel) of subdivision. These indices need to be offset to match
|
|
// the indices from the multi-level adaptive stencil tables.
|
|
// In addition: stencil tables can be built with singular stencils
|
|
// (single weight of 1.0f) as place-holders for coarse mesh vertices,
|
|
// which also needs to be accounted for.
|
|
|
|
int stencilsIndexOffset = 0;
|
|
int controlVertsIndexOffset = 0;
|
|
int nBaseStencils = baseStencilTables->GetNumStencils();
|
|
int nBaseStencilsElements = (int)baseStencilTables->_indices.size();
|
|
{
|
|
int maxlevel = refiner.GetMaxLevel();
|
|
int nverts = refiner.GetNumVerticesTotal();
|
|
if (nBaseStencils == nverts) {
|
|
|
|
// the table contain stencils for the control vertices
|
|
//
|
|
// <----------------- nverts ------------------>
|
|
//
|
|
// +---------------+----------------------------+-----------------+
|
|
// | control verts | refined verts : (max lv) | endcap points |
|
|
// +---------------+----------------------------+-----------------+
|
|
// | base stencil tables | endcap stencils |
|
|
// +--------------------------------------------+-----------------+
|
|
// : ^ /
|
|
// : \_________/
|
|
// <-------------------------------->
|
|
// stencilsIndexOffset
|
|
//
|
|
//
|
|
stencilsIndexOffset = nverts - refiner.GetNumVertices(maxlevel);
|
|
controlVertsIndexOffset = stencilsIndexOffset;
|
|
|
|
} else if (nBaseStencils == (nverts -refiner.GetNumVertices(0))) {
|
|
|
|
// the table does not contain stencils for the control vertices
|
|
//
|
|
// <----------------- nverts ------------------>
|
|
// <------ nBaseStencils ------->
|
|
// +---------------+----------------------------+-----------------+
|
|
// | control verts | refined verts : (max lv) | endcap points |
|
|
// +---------------+----------------------------+-----------------+
|
|
// | base stencil tables | endcap stencils |
|
|
// +----------------------------+-----------------+
|
|
// : ^ /
|
|
// : \_________/
|
|
// <---------------->
|
|
// stencilsIndexOffset
|
|
// <-------------------------------->
|
|
// controlVertsIndexOffset
|
|
//
|
|
stencilsIndexOffset = nBaseStencils - refiner.GetNumVertices(maxlevel);
|
|
controlVertsIndexOffset = nverts - refiner.GetNumVertices(maxlevel);
|
|
|
|
} else {
|
|
// these are not the stencils you are looking for.
|
|
assert(0);
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
// copy all endcap stencils to proto stencils, and factorize if needed.
|
|
int nEndCapStencils = endCapStencilTables->GetNumStencils();
|
|
int nEndCapStencilsElements = 0;
|
|
|
|
// we exclude zero weight stencils. the resulting number of
|
|
// stencils of endcap may be different from input.
|
|
StencilAllocator allocator(16);
|
|
allocator.Resize(nEndCapStencils);
|
|
for (int i = 0 ; i < nEndCapStencils; ++i) {
|
|
Stencil src = endCapStencilTables->GetStencil(i);
|
|
allocator[i].Clear();
|
|
for (int j = 0; j < src.GetSize(); ++j) {
|
|
Index index = src.GetVertexIndices()[j];
|
|
float weight = src.GetWeights()[j];
|
|
if (weight == 0.0) continue;
|
|
|
|
if (factorize) {
|
|
allocator[i].AddWithWeight(*baseStencilTables,
|
|
index + stencilsIndexOffset,
|
|
weight);
|
|
} else {
|
|
allocator.PushBackVertex(i,
|
|
index + controlVertsIndexOffset,
|
|
weight);
|
|
}
|
|
}
|
|
nEndCapStencilsElements += allocator.GetSize(i);
|
|
}
|
|
|
|
// create new stencil tables
|
|
StencilTables * result = new StencilTables;
|
|
result->_numControlVertices = refiner.GetNumVertices(0);
|
|
result->resize(nBaseStencils + nEndCapStencils,
|
|
nBaseStencilsElements + nEndCapStencilsElements);
|
|
|
|
int* sizes = &result->_sizes[0];
|
|
Index * indices = &result->_indices[0];
|
|
float * weights = &result->_weights[0];
|
|
|
|
// put base stencils first
|
|
memcpy(sizes, &baseStencilTables->_sizes[0],
|
|
nBaseStencils*sizeof(int));
|
|
memcpy(indices, &baseStencilTables->_indices[0],
|
|
nBaseStencilsElements*sizeof(Index));
|
|
memcpy(weights, &baseStencilTables->_weights[0],
|
|
nBaseStencilsElements*sizeof(float));
|
|
|
|
sizes += nBaseStencils;
|
|
indices += nBaseStencilsElements;
|
|
weights += nBaseStencilsElements;
|
|
|
|
// endcap stencils second
|
|
for (int i = 0 ; i < nEndCapStencils; ++i) {
|
|
int size = allocator.GetSize(i);
|
|
for (int j = 0; j < size; ++j) {
|
|
*indices++ = allocator.GetIndices(i)[j];
|
|
*weights++ = allocator.GetWeights(i)[j];
|
|
}
|
|
*sizes++ = size;
|
|
}
|
|
|
|
// have to re-generate offsets from scratch
|
|
result->generateOffsets();
|
|
|
|
return result;
|
|
}
|
|
|
|
//------------------------------------------------------------------------------
|
|
LimitStencilTables const *
|
|
LimitStencilTablesFactory::Create(TopologyRefiner const & refiner,
|
|
LocationArrayVec const & locationArrays, StencilTables const * cvStencils,
|
|
PatchTables const * patchTables) {
|
|
|
|
// Compute the total number of stencils to generate
|
|
int numStencils=0, numLimitStencils=0;
|
|
for (int i=0; i<(int)locationArrays.size(); ++i) {
|
|
assert(locationArrays[i].numLocations>=0);
|
|
numStencils += locationArrays[i].numLocations;
|
|
}
|
|
if (numStencils<=0) {
|
|
return 0;
|
|
}
|
|
|
|
bool uniform = refiner.IsUniform();
|
|
|
|
int maxlevel = refiner.GetMaxLevel(), maxsize=17;
|
|
|
|
StencilTables const * cvstencils = cvStencils;
|
|
if (not cvstencils) {
|
|
// Generate stencils for the control vertices - this is necessary to
|
|
// properly factorize patches with control vertices at level 0 (natural
|
|
// regular patches, such as in a torus)
|
|
// note: the control vertices of the mesh are added as single-index
|
|
// stencils of weight 1.0f
|
|
StencilTablesFactory::Options options;
|
|
options.generateIntermediateLevels = uniform ? false :true;
|
|
options.generateControlVerts = true;
|
|
options.generateOffsets = true;
|
|
|
|
// XXXX (manuelk) We could potentially save some mem-copies by not
|
|
// instanciating the stencil tables and work directly off the pool
|
|
// allocators.
|
|
cvstencils = StencilTablesFactory::Create(refiner, options);
|
|
} else {
|
|
// Sanity checks
|
|
if (cvstencils->GetNumStencils() != (uniform ?
|
|
refiner.GetNumVertices(maxlevel) :
|
|
refiner.GetNumVerticesTotal())) {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
// If a stencil table was given, use it, otherwise, create a new one
|
|
PatchTables const * patchtables = patchTables;
|
|
|
|
if (not patchTables) {
|
|
// XXXX (manuelk) If no patch-tables was passed, we should be able to
|
|
// infer the patches fairly easily from the refiner. Once more tags
|
|
// have been added to the refiner, maybe we can remove the need for the
|
|
// patch tables.
|
|
|
|
PatchTablesFactory::Options options;
|
|
options.SetEndCapType(
|
|
Far::PatchTablesFactory::Options::ENDCAP_GREGORY_BASIS);
|
|
|
|
patchtables = PatchTablesFactory::Create(refiner, options);
|
|
|
|
if (not cvStencils) {
|
|
// if cvstencils is just created above, append endcap stencils
|
|
if (StencilTables const *endCapStencilTables =
|
|
patchtables->GetEndCapVertexStencilTables()) {
|
|
StencilTables const *tables =
|
|
StencilTablesFactory::AppendEndCapStencilTables(
|
|
refiner, cvstencils, endCapStencilTables);
|
|
delete cvstencils;
|
|
cvstencils = tables;
|
|
}
|
|
}
|
|
} else {
|
|
// Sanity checks
|
|
if (patchTables->IsFeatureAdaptive()==uniform) {
|
|
if (not cvStencils) {
|
|
assert(cvstencils and cvstencils!=cvStencils);
|
|
delete cvstencils;
|
|
}
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
assert(patchtables and cvstencils);
|
|
|
|
// Create a patch-map to locate sub-patches faster
|
|
PatchMap patchmap( *patchtables );
|
|
|
|
//
|
|
// Generate limit stencils for locations
|
|
//
|
|
|
|
// Create a pool allocator to accumulate ProtoLimitStencils
|
|
LimitStencilAllocator alloc(maxsize);
|
|
alloc.Resize(numStencils);
|
|
|
|
// XXXX (manuelk) we can make uniform (bilinear) stencils faster with a
|
|
// dedicated code path that does not use PatchTables or the PatchMap
|
|
float wP[20], wDs[20], wDt[20];
|
|
|
|
for (int i=0; i<(int)locationArrays.size(); ++i) {
|
|
|
|
LocationArray const & array = locationArrays[i];
|
|
|
|
assert(array.ptexIdx>=0);
|
|
|
|
for (int j=0; j<array.numLocations; ++j) {
|
|
|
|
float s = array.s[j],
|
|
t = array.t[j];
|
|
|
|
PatchMap::Handle const * handle = patchmap.FindPatch(array.ptexIdx, s, t);
|
|
|
|
if (handle) {
|
|
|
|
ConstIndexArray cvs = patchtables->GetPatchVertices(*handle);
|
|
|
|
patchtables->EvaluateBasis(*handle, s, t, wP, wDs, wDt);
|
|
|
|
StencilTables const & src = *cvstencils;
|
|
ProtoLimitStencil dst = alloc[numLimitStencils];
|
|
|
|
dst.Clear();
|
|
for (int k = 0; k < cvs.size(); ++k) {
|
|
dst.AddWithWeight(src[cvs[k]], wP[k], wDs[k], wDt[k]);
|
|
}
|
|
|
|
++numLimitStencils;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (not cvStencils) {
|
|
delete cvstencils;
|
|
}
|
|
|
|
if (not patchTables) {
|
|
delete patchtables;
|
|
}
|
|
|
|
//
|
|
// Copy the proto-stencils into the limit stencil tables
|
|
//
|
|
LimitStencilTables * result = new LimitStencilTables;
|
|
|
|
int nelems = alloc.GetNumVerticesTotal();
|
|
if (nelems>0) {
|
|
|
|
// Allocate
|
|
result->resize(numLimitStencils, nelems);
|
|
|
|
// Copy stencils
|
|
LimitStencil dst(&result->_sizes.at(0), &result->_indices.at(0),
|
|
&result->_weights.at(0), &result->_duWeights.at(0),
|
|
&result->_dvWeights.at(0));
|
|
|
|
for (int i = 0; i < numLimitStencils; ++i) {
|
|
*dst._size = alloc.CopyLimitStencil(i, dst._indices, dst._weights,
|
|
dst._duWeights, dst._dvWeights);
|
|
dst.Next();
|
|
}
|
|
|
|
// XXXX manuelk should offset creation be optional ?
|
|
result->generateOffsets();
|
|
}
|
|
result->_numControlVertices = refiner.GetNumVertices(0);
|
|
|
|
return result;
|
|
}
|
|
|
|
} // end namespace Far
|
|
|
|
} // end namespace OPENSUBDIV_VERSION
|
|
} // end namespace OpenSubdiv
|