2013-04-19 02:55:05 +00:00
|
|
|
//
|
2013-07-18 21:19:50 +00:00
|
|
|
// Copyright 2013 Pixar
|
2013-04-19 02:55:05 +00:00
|
|
|
//
|
2013-07-18 21:19:50 +00:00
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License
|
|
|
|
// and the following modification to it: Section 6 Trademarks.
|
|
|
|
// deleted and replaced with:
|
2013-04-19 02:55:05 +00:00
|
|
|
//
|
2013-07-18 21:19:50 +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 for reproducing
|
|
|
|
// the content of the NOTICE file.
|
2013-04-19 02:55:05 +00:00
|
|
|
//
|
2013-07-18 21:19:50 +00:00
|
|
|
// You may obtain a copy of the License at
|
2013-04-19 02:55:05 +00:00
|
|
|
//
|
2013-07-18 21:19:50 +00:00
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing,
|
|
|
|
// software distributed under the License is distributed on an
|
|
|
|
// "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
|
|
|
|
// either express or implied. See the License for the specific
|
|
|
|
// language governing permissions and limitations under the
|
|
|
|
// License.
|
2013-04-19 02:55:05 +00:00
|
|
|
//
|
|
|
|
|
|
|
|
#include "../osd/cpuEvalLimitController.h"
|
|
|
|
#include "../osd/cpuEvalLimitKernel.h"
|
2013-05-17 16:06:40 +00:00
|
|
|
#include "../far/patchTables.h"
|
2013-04-19 02:55:05 +00:00
|
|
|
|
|
|
|
namespace OpenSubdiv {
|
|
|
|
namespace OPENSUBDIV_VERSION {
|
|
|
|
|
|
|
|
OsdCpuEvalLimitController::OsdCpuEvalLimitController() {
|
|
|
|
}
|
|
|
|
|
|
|
|
OsdCpuEvalLimitController::~OsdCpuEvalLimitController() {
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
OsdCpuEvalLimitController::_EvalLimitSample( OpenSubdiv::OsdEvalCoords const & coords,
|
2013-06-18 01:13:13 +00:00
|
|
|
OsdCpuEvalLimitContext * context,
|
2013-04-19 02:55:05 +00:00
|
|
|
unsigned int index ) {
|
2013-06-13 20:58:23 +00:00
|
|
|
float u=coords.u,
|
|
|
|
v=coords.v;
|
|
|
|
|
|
|
|
FarPatchMap::Handle const * handle = context->GetPatchMap().FindPatch( coords.face, u, v );
|
|
|
|
|
|
|
|
// the map may not be able to return a handle if there is a hole or the face
|
|
|
|
// index is incorrect
|
|
|
|
if (not handle)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
FarPatchParam::BitField bits = context->GetPatchBitFields()[ handle->patchIdx ];
|
2013-04-19 02:55:05 +00:00
|
|
|
|
2013-06-13 20:58:23 +00:00
|
|
|
bits.Normalize( u, v );
|
|
|
|
|
|
|
|
bits.Rotate( u, v );
|
|
|
|
|
|
|
|
FarPatchTables::PatchArray const & parray = context->GetPatchArrayVector()[ handle->patchArrayIdx ];
|
2013-04-19 02:55:05 +00:00
|
|
|
|
2013-06-13 20:58:23 +00:00
|
|
|
unsigned int const * cvs = &context->GetControlVertices()[ parray.GetVertIndex() + handle->vertexOffset ];
|
2013-04-19 02:55:05 +00:00
|
|
|
|
2013-06-18 01:13:13 +00:00
|
|
|
OsdCpuEvalLimitContext::VertexData & vertexData = context->GetVertexData();
|
|
|
|
|
|
|
|
if (vertexData.IsBound()) {
|
|
|
|
|
|
|
|
int offset = vertexData.outDesc.stride * index;
|
|
|
|
|
|
|
|
// Based on patch type - go execute interpolation
|
|
|
|
switch( parray.GetDescriptor().GetType() ) {
|
|
|
|
|
|
|
|
case FarPatchTables::REGULAR : if (vertexData.IsBound()) {
|
|
|
|
evalBSpline( v, u, cvs,
|
|
|
|
vertexData.inDesc,
|
|
|
|
vertexData.in.GetData(),
|
|
|
|
vertexData.outDesc,
|
|
|
|
vertexData.out.GetData()+offset,
|
|
|
|
vertexData.outDu.GetData()+offset,
|
|
|
|
vertexData.outDv.GetData()+offset );
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case FarPatchTables::BOUNDARY : if (vertexData.IsBound()) {
|
|
|
|
evalBoundary( v, u, cvs,
|
|
|
|
vertexData.inDesc,
|
|
|
|
vertexData.in.GetData(),
|
|
|
|
vertexData.outDesc,
|
|
|
|
vertexData.out.GetData()+offset,
|
|
|
|
vertexData.outDu.GetData()+offset,
|
|
|
|
vertexData.outDv.GetData()+offset );
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case FarPatchTables::CORNER : if (vertexData.IsBound()) {
|
|
|
|
evalCorner( v, u, cvs,
|
|
|
|
vertexData.inDesc,
|
|
|
|
vertexData.in.GetData(),
|
|
|
|
vertexData.outDesc,
|
|
|
|
vertexData.out.GetData()+offset,
|
|
|
|
vertexData.outDu.GetData()+offset,
|
|
|
|
vertexData.outDv.GetData()+offset );
|
|
|
|
} break;
|
|
|
|
|
|
|
|
|
|
|
|
case FarPatchTables::GREGORY : if (vertexData.IsBound()) {
|
|
|
|
evalGregory( v, u, cvs,
|
|
|
|
&context->GetVertexValenceTable()[0],
|
|
|
|
&context->GetQuadOffsetTable()[ parray.GetQuadOffsetIndex() + handle->vertexOffset ],
|
|
|
|
context->GetMaxValence(),
|
|
|
|
vertexData.inDesc,
|
|
|
|
vertexData.in.GetData(),
|
|
|
|
vertexData.outDesc,
|
|
|
|
vertexData.out.GetData()+offset,
|
|
|
|
vertexData.outDu.GetData()+offset,
|
|
|
|
vertexData.outDv.GetData()+offset );
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case FarPatchTables::GREGORY_BOUNDARY :
|
|
|
|
if (vertexData.IsBound()) {
|
|
|
|
evalGregoryBoundary(v, u, cvs,
|
|
|
|
&context->GetVertexValenceTable()[0],
|
|
|
|
&context->GetQuadOffsetTable()[ parray.GetQuadOffsetIndex() + handle->vertexOffset ],
|
|
|
|
context->GetMaxValence(),
|
|
|
|
vertexData.inDesc,
|
|
|
|
vertexData.in.GetData(),
|
|
|
|
vertexData.outDesc,
|
|
|
|
vertexData.out.GetData()+offset,
|
|
|
|
vertexData.outDu.GetData()+offset,
|
|
|
|
vertexData.outDv.GetData()+offset );
|
|
|
|
} break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
assert(0);
|
|
|
|
}
|
2013-06-13 20:58:23 +00:00
|
|
|
}
|
|
|
|
|
2013-06-18 01:13:13 +00:00
|
|
|
OsdCpuEvalLimitContext::VaryingData & varyingData = context->GetVaryingData();
|
|
|
|
|
2013-06-13 20:58:23 +00:00
|
|
|
if (varyingData.IsBound()) {
|
|
|
|
|
|
|
|
static int indices[5][4] = { {5, 6,10, 9}, // regular
|
|
|
|
{1, 2, 6, 5}, // boundary
|
|
|
|
{1, 2, 5, 4}, // corner
|
|
|
|
{0, 1, 2, 3}, // gregory
|
|
|
|
{0, 1, 2, 3} };// gregory boundary
|
|
|
|
|
|
|
|
int type = (int)(parray.GetDescriptor().GetType() - FarPatchTables::REGULAR);
|
|
|
|
|
2013-06-18 01:13:13 +00:00
|
|
|
int offset = varyingData.outDesc.stride * index;
|
|
|
|
|
2013-06-13 20:58:23 +00:00
|
|
|
unsigned int zeroRing[4] = { cvs[indices[type][0]],
|
|
|
|
cvs[indices[type][1]],
|
|
|
|
cvs[indices[type][2]],
|
|
|
|
cvs[indices[type][3]] };
|
2013-06-11 22:59:43 +00:00
|
|
|
|
2013-06-13 20:58:23 +00:00
|
|
|
evalBilinear( v, u, zeroRing,
|
2013-06-18 01:13:13 +00:00
|
|
|
varyingData.inDesc,
|
|
|
|
varyingData.in.GetData(),
|
|
|
|
varyingData.outDesc,
|
|
|
|
varyingData.out.GetData()+offset);
|
2013-06-13 20:58:23 +00:00
|
|
|
|
|
|
|
}
|
2013-04-19 02:55:05 +00:00
|
|
|
|
2013-06-13 20:58:23 +00:00
|
|
|
// Note : currently we only support bilinear boundary interpolation rules
|
|
|
|
// for face-varying data. Although Hbr supports 3 additional smooth rule
|
|
|
|
// sets, the feature-adaptive patch interpolation code currently does not
|
|
|
|
// support them, and neither does this EvalContext.
|
2013-06-18 01:13:13 +00:00
|
|
|
OsdCpuEvalLimitContext::FaceVaryingData & faceVaryingData = context->GetFaceVaryingData();
|
|
|
|
if (faceVaryingData.IsBound()) {
|
2013-06-07 01:07:46 +00:00
|
|
|
|
2013-06-13 20:58:23 +00:00
|
|
|
FarPatchTables::FVarDataTable const & fvarData = context->GetFVarData();
|
2013-06-05 18:44:30 +00:00
|
|
|
|
2013-06-13 20:58:23 +00:00
|
|
|
if (not fvarData.empty()) {
|
2013-06-07 01:07:46 +00:00
|
|
|
|
2013-06-18 01:13:13 +00:00
|
|
|
int offset = faceVaryingData.outDesc.stride * index;
|
2013-06-13 20:58:23 +00:00
|
|
|
|
|
|
|
static unsigned int zeroRing[4] = {0,1,2,3};
|
2013-06-07 01:07:46 +00:00
|
|
|
|
2013-06-13 20:58:23 +00:00
|
|
|
evalBilinear( v, u, zeroRing,
|
2013-06-18 01:13:13 +00:00
|
|
|
faceVaryingData.inDesc,
|
|
|
|
&fvarData[ handle->patchIdx * 4 * context->GetFVarWidth() ],
|
|
|
|
faceVaryingData.outDesc,
|
|
|
|
faceVaryingData.out.GetData()+offset);
|
2013-06-13 20:58:23 +00:00
|
|
|
}
|
2013-04-19 02:55:05 +00:00
|
|
|
}
|
2013-06-05 18:44:30 +00:00
|
|
|
|
2013-06-13 20:58:23 +00:00
|
|
|
return 1;
|
2013-04-19 02:55:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
} // end namespace OPENSUBDIV_VERSION
|
|
|
|
} // end namespace OpenSubdiv
|