📄 wmlintpbsplineuniform4.cpp
字号:
// Magic Software, Inc.
// http://www.magic-software.com
// http://www.wild-magic.com
// Copyright (c) 2003. All Rights Reserved
//
// The Wild Magic Library (WML) source code is supplied under the terms of
// the license agreement http://www.magic-software.com/License/WildMagic.pdf
// and may not be copied or disclosed except in accordance with the terms of
// that agreement.
#include "WmlIntpBSplineUniform4.h"
#include "WmlMath.h"
using namespace Wml;
//----------------------------------------------------------------------------
template <class Real>
IntpBSplineUniform4<Real>::IntpBSplineUniform4 (int iDegree, const int* aiDim,
Real* afData)
:
IntpBSplineUniform<Real>(4,iDegree,aiDim,afData)
{
}
//----------------------------------------------------------------------------
template <class Real>
int IntpBSplineUniform4<Real>::Index (int iX, int iY, int iZ, int iW) const
{
return iX + m_aiDim[0]*(iY + m_aiDim[1]*(iZ + m_aiDim[2]*iW));
}
//----------------------------------------------------------------------------
template <class Real>
Real IntpBSplineUniform4<Real>::operator() (Real* afX)
{
return (*this)(afX[0],afX[1],afX[2],afX[3]);
}
//----------------------------------------------------------------------------
template <class Real>
Real IntpBSplineUniform4<Real>::operator() (int* aiDx, Real* afX)
{
return (*this)(aiDx[0],aiDx[1],aiDx[2],aiDx[3],afX[0],afX[1],afX[2],
afX[3]);
}
//----------------------------------------------------------------------------
template <class Real>
Real IntpBSplineUniform4<Real>::operator() (Real fX, Real fY, Real fZ,
Real fW)
{
m_aiBase[0] = (int)Math<Real>::Floor(fX);
m_aiBase[1] = (int)Math<Real>::Floor(fY);
m_aiBase[2] = (int)Math<Real>::Floor(fZ);
m_aiBase[3] = (int)Math<Real>::Floor(fW);
for (int iDim = 0; iDim < 4; iDim++)
{
if ( m_aiOldBase[iDim] != m_aiBase[iDim] )
{
// switch to new local grid
for (int k = 0; k < 4; k++)
{
m_aiOldBase[k] = m_aiBase[k];
m_aiGridMin[k] = m_aiBase[k] - 1;
m_aiGridMax[k] = m_aiGridMin[k] + m_iDegree;
}
// fill in missing grid data if necessary
if ( m_oEvaluateCallback )
EvaluateUnknownData();
ComputeIntermediate();
break;
}
}
SetPolynomial(0,fX-m_aiBase[0],m_aafPoly[0]);
SetPolynomial(0,fY-m_aiBase[1],m_aafPoly[1]);
SetPolynomial(0,fZ-m_aiBase[2],m_aafPoly[2]);
SetPolynomial(0,fW-m_aiBase[3],m_aafPoly[3]);
int aiI[4] = { 0, 0, 0, 0 };
Real fResult = (Real)0.0;
for (int k = 0; k < m_iDp1ToN; k++)
{
fResult += m_aafPoly[0][aiI[0]]*m_aafPoly[1][aiI[1]]*
m_aafPoly[2][aiI[2]]*m_aafPoly[3][aiI[3]]*m_afInter[k];
if ( ++aiI[0] <= m_iDegree )
continue;
aiI[0] = 0;
if ( ++aiI[1] <= m_iDegree )
continue;
aiI[1] = 0;
if ( ++aiI[2] <= m_iDegree )
continue;
aiI[2] = 0;
aiI[3]++;
}
return fResult;
}
//----------------------------------------------------------------------------
template <class Real>
Real IntpBSplineUniform4<Real>::operator() (int iDx, int iDy, int iDz,
int iDw, Real fX, Real fY, Real fZ, Real fW)
{
m_aiBase[0] = (int)Math<Real>::Floor(fX);
m_aiBase[1] = (int)Math<Real>::Floor(fY);
m_aiBase[2] = (int)Math<Real>::Floor(fZ);
m_aiBase[3] = (int)Math<Real>::Floor(fW);
for (int iDim = 0; iDim < 4; iDim++)
{
if ( m_aiOldBase[iDim] != m_aiBase[iDim] )
{
// switch to new local grid
for (int k = 0; k < 4; k++)
{
m_aiOldBase[k] = m_aiBase[k];
m_aiGridMin[k] = m_aiBase[k] - 1;
m_aiGridMax[k] = m_aiGridMin[k] + m_iDegree;
}
// fill in missing grid data if necessary
if ( m_oEvaluateCallback )
EvaluateUnknownData();
ComputeIntermediate();
break;
}
}
SetPolynomial(iDx,fX-m_aiBase[0],m_aafPoly[0]);
SetPolynomial(iDy,fY-m_aiBase[1],m_aafPoly[1]);
SetPolynomial(iDz,fZ-m_aiBase[2],m_aafPoly[2]);
SetPolynomial(iDw,fW-m_aiBase[3],m_aafPoly[3]);
int aiI[4] = { iDx, iDy, iDz, iDw };
int aiDelta[3] = { iDx, m_iDp1*iDy, m_iDp1*m_iDp1*iDz };
Real fResult = (Real)0.0;
int k = aiI[0]+m_iDp1*(aiI[1]+m_iDp1*(aiI[2]+m_iDp1*aiI[3]));
for (/**/; k < m_iDp1ToN; k++)
{
fResult += m_aafPoly[0][aiI[0]]*m_aafPoly[1][aiI[1]]*
m_aafPoly[2][aiI[2]]*m_aafPoly[3][aiI[3]]*m_afInter[k];
if ( ++aiI[0] <= m_iDegree )
continue;
aiI[0] = iDx;
k += aiDelta[0];
if ( ++aiI[1] <= m_iDegree )
continue;
aiI[1] = iDy;
k += aiDelta[1];
if ( ++aiI[2] <= m_iDegree )
continue;
aiI[2] = iDz;
k += aiDelta[2];
aiI[3]++;
}
return fResult;
}
//----------------------------------------------------------------------------
template <class Real>
void IntpBSplineUniform4<Real>::EvaluateUnknownData ()
{
for (int k3 = m_aiGridMin[3]; k3 <= m_aiGridMax[3]; k3++)
{
for (int k2 = m_aiGridMin[2]; k2 <= m_aiGridMax[2]; k2++)
{
for (int k1 = m_aiGridMin[1]; k1 <= m_aiGridMax[1]; k1++)
{
for (int k0 = m_aiGridMin[0]; k0 <= m_aiGridMax[0]; k0++)
{
int iIndex = Index(k0,k1,k2,k3);
if ( m_afData[iIndex] == Math<Real>::MAX_REAL )
m_afData[iIndex] = m_oEvaluateCallback(iIndex);
}
}
}
}
}
//----------------------------------------------------------------------------
template <class Real>
void IntpBSplineUniform4<Real>::ComputeIntermediate ()
{
// fetch subblock of data to cache
int aiDelta[3] =
{
m_aiDim[0]-m_iDp1,
m_aiDim[0]*(m_aiDim[1]-m_iDp1),
m_aiDim[0]*m_aiDim[1]*(m_aiDim[2]-m_iDp1)
};
int aiLoop[4];
for (int iDim = 0; iDim < 4; iDim++)
aiLoop[iDim] = m_aiGridMin[iDim];
int iIndex = Index(aiLoop[0],aiLoop[1],aiLoop[2],aiLoop[3]);
int k;
for (k = 0; k < m_iDp1ToN; k++, iIndex++)
{
m_afCache[k] = m_afData[iIndex];
if ( ++aiLoop[0] <= m_aiGridMax[0] )
continue;
aiLoop[0] = m_aiGridMin[0];
iIndex += aiDelta[0];
if ( ++aiLoop[1] <= m_aiGridMax[1] )
continue;
aiLoop[1] = m_aiGridMin[1];
iIndex += aiDelta[1];
if ( ++aiLoop[2] <= m_aiGridMax[2] )
continue;
aiLoop[2] = m_aiGridMin[2];
iIndex += aiDelta[2];
aiLoop[3]++;
}
// compute and save the intermediate product
for (int i = 0, j = 0; i < m_iDp1ToN; i++)
{
Real fSum = (Real)0.0;
for (k = 0; k < m_iDp1ToN; k += m_aiSkip[j], j += m_aiSkip[j])
fSum += m_afProduct[j]*m_afCache[k];
m_afInter[i] = fSum;
}
}
//----------------------------------------------------------------------------
//----------------------------------------------------------------------------
// explicit instantiation
//----------------------------------------------------------------------------
namespace Wml
{
template class WML_ITEM IntpBSplineUniform4<float>;
template class WML_ITEM IntpBSplineUniform4<double>;
}
//----------------------------------------------------------------------------
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -