2013-09-11 23:05:14 +00:00
|
|
|
//
|
2013-09-26 19:04:57 +00:00
|
|
|
// Copyright 2013 Pixar
|
2013-09-11 23:05:14 +00:00
|
|
|
//
|
2013-09-26 19:04:57 +00:00
|
|
|
// 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:
|
2013-09-11 23:05:14 +00:00
|
|
|
//
|
2013-09-26 19:04:57 +00:00
|
|
|
// 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.
|
2013-09-11 23:05:14 +00:00
|
|
|
//
|
2013-09-26 19:04:57 +00:00
|
|
|
// You may obtain a copy of the Apache License at
|
2013-09-11 23:05:14 +00:00
|
|
|
//
|
2013-09-26 19:04:57 +00:00
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
2013-09-11 23:05:14 +00:00
|
|
|
//
|
2013-09-26 19:04:57 +00:00
|
|
|
// 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.
|
2013-09-11 23:05:14 +00:00
|
|
|
//
|
|
|
|
|
|
|
|
#ifndef FAR_STENCILTABLES_H
|
|
|
|
#define FAR_STENCILTABLES_H
|
|
|
|
|
|
|
|
#include "../version.h"
|
|
|
|
|
2014-09-05 22:07:46 +00:00
|
|
|
#include <cassert>
|
2013-09-11 23:05:14 +00:00
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
namespace OpenSubdiv {
|
|
|
|
namespace OPENSUBDIV_VERSION {
|
|
|
|
|
2014-09-05 22:07:46 +00:00
|
|
|
namespace Far {
|
2013-09-11 23:05:14 +00:00
|
|
|
|
2014-09-05 22:07:46 +00:00
|
|
|
/// \brief Vertex stencil descriptor
|
2013-09-11 23:05:14 +00:00
|
|
|
///
|
2014-09-05 22:07:46 +00:00
|
|
|
/// Allows access and manipulation of a single stencil in a StencilTables.
|
2013-09-11 23:05:14 +00:00
|
|
|
///
|
2014-09-05 22:07:46 +00:00
|
|
|
class Stencil {
|
|
|
|
|
2013-09-11 23:05:14 +00:00
|
|
|
public:
|
|
|
|
|
2014-09-05 22:07:46 +00:00
|
|
|
/// \brief Default constructor
|
|
|
|
Stencil() {}
|
|
|
|
|
|
|
|
/// \brief Constructor
|
|
|
|
///
|
|
|
|
/// @param size Table pointer to the size of the stencil
|
|
|
|
///
|
|
|
|
/// @param indices Table pointer to the vertex indices of the stencil
|
|
|
|
///
|
|
|
|
/// @param weights Table pointer to the vertex weights of the stencil
|
|
|
|
///
|
|
|
|
Stencil(unsigned char * size,
|
|
|
|
int * indices,
|
|
|
|
float * weights)
|
|
|
|
: _size(size),
|
|
|
|
_indices(indices),
|
|
|
|
_weights(weights) {
|
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief Copy constructor
|
|
|
|
Stencil(Stencil const & other) {
|
|
|
|
_size = other._size;
|
|
|
|
_indices = other._indices;
|
|
|
|
_weights = other._weights;
|
|
|
|
}
|
|
|
|
|
2014-09-19 00:48:34 +00:00
|
|
|
/// \brief Returns the size of the stencil
|
2013-09-11 23:05:14 +00:00
|
|
|
int GetSize() const {
|
|
|
|
return *_size;
|
|
|
|
}
|
|
|
|
|
2014-09-19 00:48:34 +00:00
|
|
|
/// \brief Returns the size of the stencil as a pointer
|
2014-09-12 23:59:16 +00:00
|
|
|
unsigned char * GetSizePtr() const {
|
|
|
|
return _size;
|
|
|
|
}
|
|
|
|
|
2013-12-20 19:01:58 +00:00
|
|
|
/// \brief Returns the control vertices indices
|
2013-09-11 23:05:14 +00:00
|
|
|
int const * GetVertexIndices() const {
|
|
|
|
return _indices;
|
|
|
|
}
|
|
|
|
|
2013-12-20 19:01:58 +00:00
|
|
|
/// \brief Returns the interpolation weights
|
2014-09-05 22:07:46 +00:00
|
|
|
float const * GetWeights() const {
|
|
|
|
return _weights;
|
2013-09-11 23:05:14 +00:00
|
|
|
}
|
|
|
|
|
2014-09-05 22:07:46 +00:00
|
|
|
/// \brief Advance to the next stencil in the table
|
|
|
|
void Next() {
|
2013-09-11 23:05:14 +00:00
|
|
|
int stride = *_size;
|
|
|
|
++_size;
|
|
|
|
_indices += stride;
|
2014-09-05 22:07:46 +00:00
|
|
|
_weights += stride;
|
2013-09-11 23:05:14 +00:00
|
|
|
}
|
|
|
|
|
2014-09-05 22:07:46 +00:00
|
|
|
protected:
|
|
|
|
friend class StencilTablesFactory;
|
2014-09-12 23:59:16 +00:00
|
|
|
friend class LimitStencilTablesFactory;
|
2013-09-11 23:05:14 +00:00
|
|
|
|
2014-09-05 22:07:46 +00:00
|
|
|
unsigned char * _size;
|
|
|
|
int * _indices;
|
|
|
|
float * _weights;
|
2013-09-11 23:05:14 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/// \brief Table of subdivision stencils.
|
|
|
|
///
|
|
|
|
/// Stencils are the most direct methods of evaluation of locations on the limit
|
|
|
|
/// of a surface. Every point of a limit surface can be computed by linearly
|
|
|
|
/// blending a collection of coarse control vertices.
|
|
|
|
///
|
|
|
|
/// A stencil assigns a series of control vertex indices with a blending weight
|
|
|
|
/// that corresponds to a unique parametric location of the limit surface. When
|
|
|
|
/// the control vertices move in space, the limit location can be very efficiently
|
|
|
|
/// recomputed simply by applying the blending weights to the series of coarse
|
|
|
|
/// control vertices.
|
|
|
|
///
|
2014-09-05 22:07:46 +00:00
|
|
|
class StencilTables {
|
2013-09-11 23:05:14 +00:00
|
|
|
|
|
|
|
public:
|
|
|
|
|
2014-09-05 22:07:46 +00:00
|
|
|
/// \brief Returns the number of stencils in the table
|
2013-09-11 23:05:14 +00:00
|
|
|
int GetNumStencils() const {
|
|
|
|
return (int)_sizes.size();
|
|
|
|
}
|
|
|
|
|
2014-09-19 00:48:34 +00:00
|
|
|
/// \brief Returns the number of control vertices indexed in the table
|
2014-09-05 22:07:46 +00:00
|
|
|
int GetNumControlVertices() const {
|
|
|
|
return _numControlVertices;
|
2013-09-11 23:05:14 +00:00
|
|
|
}
|
|
|
|
|
2014-09-05 22:07:46 +00:00
|
|
|
/// \brief Returns a Stencil at index i in the tables
|
|
|
|
Stencil GetStencil(int i) const;
|
2013-09-11 23:05:14 +00:00
|
|
|
|
|
|
|
/// \brief Returns the number of control vertices of each stencil in the table
|
2014-09-05 22:07:46 +00:00
|
|
|
std::vector<unsigned char> const & GetSizes() const {
|
2013-09-11 23:05:14 +00:00
|
|
|
return _sizes;
|
|
|
|
}
|
|
|
|
|
2014-09-05 22:07:46 +00:00
|
|
|
/// \brief Returns the offset to a given stencil (factory may leave empty)
|
2013-09-11 23:05:14 +00:00
|
|
|
std::vector<int> const & GetOffsets() const {
|
|
|
|
return _offsets;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief Returns the indices of the control vertices
|
|
|
|
std::vector<int> const & GetControlIndices() const {
|
|
|
|
return _indices;
|
|
|
|
}
|
|
|
|
|
2014-09-05 22:07:46 +00:00
|
|
|
/// \brief Returns the stencil interpolation weights
|
2013-09-11 23:05:14 +00:00
|
|
|
std::vector<float> const & GetWeights() const {
|
2014-09-05 22:07:46 +00:00
|
|
|
return _weights;
|
2013-09-11 23:05:14 +00:00
|
|
|
}
|
|
|
|
|
2014-09-19 00:48:34 +00:00
|
|
|
/// \brief Returns the stencil at index i in the tables
|
2014-09-24 21:06:18 +00:00
|
|
|
Stencil operator[] (int index) const;
|
2014-09-19 00:48:34 +00:00
|
|
|
|
2014-09-05 22:07:46 +00:00
|
|
|
/// \brief Updates point values based on the control values
|
|
|
|
///
|
|
|
|
/// \note The destination buffers ('uderivs' & 'vderivs') are assumed to
|
|
|
|
/// have allocated at least \c GetNumStencils() elements.
|
|
|
|
///
|
|
|
|
/// @param controlValues Buffer with primvar data for the control vertices
|
|
|
|
///
|
|
|
|
/// @param values Destination buffer for the interpolated primvar
|
|
|
|
/// data
|
|
|
|
///
|
|
|
|
/// @param start (skip to )index of first value to update
|
|
|
|
///
|
|
|
|
/// @param end Index of last value to update
|
2014-09-12 23:59:16 +00:00
|
|
|
///
|
2014-09-05 22:07:46 +00:00
|
|
|
template <class T>
|
|
|
|
void UpdateValues(T const *controlValues, T *values, int start=-1, int end=-1) const {
|
|
|
|
|
|
|
|
_Update(controlValues, values, _weights, start, end);
|
2013-09-11 23:05:14 +00:00
|
|
|
}
|
|
|
|
|
2014-09-12 23:59:16 +00:00
|
|
|
/// \brief Clears the stencils from the table
|
|
|
|
void Clear() {
|
|
|
|
_numControlVertices=0;
|
|
|
|
_sizes.clear();
|
|
|
|
_offsets.clear();
|
|
|
|
_indices.clear();
|
|
|
|
_weights.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
protected:
|
2014-09-05 22:07:46 +00:00
|
|
|
|
|
|
|
// Update values by appling cached stencil weights to new control values
|
|
|
|
template <class T> void _Update( T const *controlValues, T *values,
|
|
|
|
std::vector<float> const & valueWeights, int start, int end) const;
|
|
|
|
|
2014-09-12 23:59:16 +00:00
|
|
|
protected:
|
2014-09-05 22:07:46 +00:00
|
|
|
|
|
|
|
friend class StencilTablesFactory;
|
|
|
|
|
|
|
|
int _numControlVertices; // number of control vertices
|
|
|
|
|
|
|
|
std::vector<unsigned char> _sizes; // number of coeffiecient for each stencil
|
|
|
|
std::vector<int> _offsets, // offset to the start of each stencil
|
|
|
|
_indices; // indices of contributing coarse vertices
|
|
|
|
std::vector<float> _weights; // stencil weight coefficients
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/// \brief Limit point stencil descriptor
|
|
|
|
///
|
|
|
|
class LimitStencil : public Stencil {
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
/// \brief Constructor
|
|
|
|
///
|
|
|
|
/// @param size Table pointer to the size of the stencil
|
|
|
|
///
|
|
|
|
/// @param indices Table pointer to the vertex indices of the stencil
|
|
|
|
///
|
|
|
|
/// @param weights Table pointer to the vertex weights of the stencil
|
|
|
|
///
|
|
|
|
/// @param duWeights Table pointer to the 'u' derivative weights
|
|
|
|
///
|
|
|
|
/// @param dvWeights Table pointer to the 'v' derivative weights
|
|
|
|
///
|
|
|
|
LimitStencil( unsigned char * size,
|
|
|
|
int * indices,
|
|
|
|
float * weights,
|
|
|
|
float * duWeights,
|
|
|
|
float * dvWeights )
|
|
|
|
: Stencil(size, indices, weights),
|
|
|
|
_duWeights(duWeights),
|
|
|
|
_dvWeights(dvWeights) {
|
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief
|
|
|
|
float const * GetDuWeights() const {
|
|
|
|
return _duWeights;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief
|
|
|
|
float const * GetDvWeights() const {
|
|
|
|
return _dvWeights;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief Advance to the next stencil in the table
|
|
|
|
void Next() {
|
|
|
|
int stride = *_size;
|
|
|
|
++_size;
|
|
|
|
_indices += stride;
|
|
|
|
_weights += stride;
|
|
|
|
_duWeights += stride;
|
|
|
|
_dvWeights += stride;
|
2013-09-11 23:05:14 +00:00
|
|
|
}
|
2014-09-05 22:07:46 +00:00
|
|
|
|
2013-09-11 23:05:14 +00:00
|
|
|
private:
|
2014-09-12 23:59:16 +00:00
|
|
|
|
|
|
|
friend class StencilTablesFactory;
|
|
|
|
friend class LimitStencilTablesFactory;
|
|
|
|
|
2014-09-05 22:07:46 +00:00
|
|
|
float * _duWeights, // pointer to stencil u derivative limit weights
|
|
|
|
* _dvWeights; // pointer to stencil v derivative limit weights
|
|
|
|
};
|
2013-09-11 23:05:14 +00:00
|
|
|
|
2014-09-05 22:07:46 +00:00
|
|
|
/// \brief Table of limit subdivision stencils.
|
|
|
|
///
|
|
|
|
///
|
|
|
|
class LimitStencilTables : public StencilTables {
|
2013-09-11 23:05:14 +00:00
|
|
|
|
2014-09-05 22:07:46 +00:00
|
|
|
public:
|
|
|
|
|
|
|
|
/// \brief Returns the 'u' derivative stencil interpolation weights
|
|
|
|
std::vector<float> const & GetDuWeights() const {
|
|
|
|
return _duWeights;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief Returns the 'v' derivative stencil interpolation weights
|
|
|
|
std::vector<float> const & GetDvWeights() const {
|
|
|
|
return _dvWeights;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief Updates derivative values based on the control values
|
|
|
|
///
|
|
|
|
/// \note The destination buffers ('uderivs' & 'vderivs') are assumed to
|
|
|
|
/// have allocated at least \c GetNumStencils() elements.
|
|
|
|
///
|
|
|
|
/// @param controlValues Buffer with primvar data for the control vertices
|
|
|
|
///
|
|
|
|
/// @param uderivs Destination buffer for the interpolated 'u'
|
|
|
|
/// derivative primvar data
|
|
|
|
///
|
|
|
|
/// @param vderivs Destination buffer for the interpolated 'v'
|
|
|
|
/// derivative primvar data
|
|
|
|
///
|
|
|
|
/// @param start (skip to )index of first value to update
|
|
|
|
///
|
|
|
|
/// @param end Index of last value to update
|
|
|
|
///
|
|
|
|
template <class T>
|
|
|
|
void UpdateDerivs(T const *controlValues, T *uderivs, T *vderivs,
|
|
|
|
int start=-1, int end=-1) const {
|
2013-09-11 23:05:14 +00:00
|
|
|
|
2014-09-05 22:07:46 +00:00
|
|
|
_Update(controlValues, uderivs, _duWeights, start, end);
|
|
|
|
_Update(controlValues, vderivs, _dvWeights, start, end);
|
|
|
|
}
|
2013-09-11 23:05:14 +00:00
|
|
|
|
2014-09-12 23:59:16 +00:00
|
|
|
/// \brief Clears the stencils from the table
|
|
|
|
void Clear() {
|
|
|
|
StencilTables::Clear();
|
|
|
|
_duWeights.clear();
|
|
|
|
_dvWeights.clear();
|
|
|
|
}
|
2013-09-11 23:05:14 +00:00
|
|
|
|
2014-09-05 22:07:46 +00:00
|
|
|
private:
|
2014-09-12 23:59:16 +00:00
|
|
|
friend class LimitStencilTablesFactory;
|
|
|
|
|
2014-09-05 22:07:46 +00:00
|
|
|
std::vector<float> _duWeights, // u derivative limit stencil weights
|
|
|
|
_dvWeights; // v derivative limit stencil weights
|
2013-09-11 23:05:14 +00:00
|
|
|
};
|
|
|
|
|
2014-09-05 22:07:46 +00:00
|
|
|
|
|
|
|
// Update values by appling cached stencil weights to new control values
|
2013-09-11 23:05:14 +00:00
|
|
|
template <class T> void
|
2014-09-05 22:07:46 +00:00
|
|
|
StencilTables::_Update(T const *controlValues, T *values,
|
|
|
|
std::vector<float> const &valueWeights, int start, int end) const {
|
|
|
|
|
|
|
|
int const * indices = &_indices.at(0);
|
|
|
|
float const * weights = &valueWeights.at(0);
|
2013-09-11 23:05:14 +00:00
|
|
|
|
2014-09-05 22:07:46 +00:00
|
|
|
if (start>0) {
|
|
|
|
assert(start<(int)_offsets.size());
|
|
|
|
indices += _offsets[start];
|
|
|
|
weights += _offsets[start];
|
|
|
|
values += start;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (end<start or end<0) {
|
|
|
|
end = GetNumStencils();
|
|
|
|
}
|
2013-09-11 23:05:14 +00:00
|
|
|
|
2014-09-05 22:07:46 +00:00
|
|
|
int nstencils = end - std::max(0, start);
|
|
|
|
for (int i=0; i<nstencils; ++i) {
|
2013-09-11 23:05:14 +00:00
|
|
|
|
|
|
|
// Zero out the result accumulators
|
|
|
|
values[i].Clear();
|
|
|
|
|
|
|
|
// For each element in the array, add the coefs contribution
|
2014-09-05 22:07:46 +00:00
|
|
|
for (int j=0; j<_sizes[i]; ++j, ++indices, ++weights) {
|
|
|
|
values[i].AddWithWeight( controlValues[*indices], *weights );
|
2013-09-11 23:05:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-05 22:07:46 +00:00
|
|
|
// Returns a Stencil at index i in the table
|
|
|
|
inline Stencil
|
|
|
|
StencilTables::GetStencil(int i) const {
|
2013-09-11 23:05:14 +00:00
|
|
|
|
2014-09-19 00:48:34 +00:00
|
|
|
assert((not _offsets.empty()) and i<(int)_offsets.size());
|
|
|
|
|
2013-09-11 23:05:14 +00:00
|
|
|
int ofs = _offsets[i];
|
|
|
|
|
2014-09-05 22:07:46 +00:00
|
|
|
return Stencil( const_cast<unsigned char *>(&_sizes[i]),
|
2013-09-11 23:05:14 +00:00
|
|
|
const_cast<int *>(&_indices[ofs]),
|
2014-09-05 22:07:46 +00:00
|
|
|
const_cast<float *>(&_weights[ofs]) );
|
2013-09-11 23:05:14 +00:00
|
|
|
}
|
|
|
|
|
2014-09-24 21:06:18 +00:00
|
|
|
inline Stencil
|
|
|
|
StencilTables::operator[] (int index) const {
|
|
|
|
|
|
|
|
return GetStencil(index);
|
|
|
|
}
|
|
|
|
|
2014-09-05 22:07:46 +00:00
|
|
|
} // end namespace Far
|
2013-09-11 23:05:14 +00:00
|
|
|
|
|
|
|
} // end namespace OPENSUBDIV_VERSION
|
|
|
|
using namespace OPENSUBDIV_VERSION;
|
|
|
|
|
|
|
|
} // end namespace OpenSubdiv
|
|
|
|
|
|
|
|
#endif // FAR_STENCILTABLES_H
|