| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| /////////////////////////// |
| // BSplineEvaluationData // |
| /////////////////////////// |
| template< int Degree > |
| double BSplineEvaluationData< Degree >::Value( int depth , int off , double s , bool dirichlet , bool derivative ) |
| { |
| if( s<0 || s>1 ) return 0.; |
|
|
| int dim = Dimension(depth) , res = 1<<depth; |
| if( off<0 || off>=dim ) return 0; |
|
|
| BSplineComponents components = BSplineComponents( depth , off , dirichlet ); |
|
|
| // [NOTE] This is an ugly way to ensure that when s=1 we evaluate using a B-Spline component within the valid range. |
| int ii = std::max< int >( 0 , std::min< int >( res-1 , (int)floor( s * res ) ) ) - off; |
|
|
| if( ii<SupportStart || ii>SupportEnd ) return 0; |
| if( derivative ) return components[ii-SupportStart].derivative()(s); |
| else return components[ii-SupportStart](s); |
| } |
| template< int Degree > |
| void BSplineEvaluationData< Degree >::SetCenterEvaluator( typename CenterEvaluator::Evaluator& evaluator , int depth , bool dirichlet ) |
| { |
| evaluator._depth = depth; |
| int dim = BSplineEvaluationData< Degree >::Dimension( depth ) , res = 1<<depth; |
| for( int i=0 ; i<CenterEvaluator::Size ; i++ ) for( int j=SupportStart ; j<=SupportEnd ; j++ ) |
| { |
| int ii = ( i<=CenterEvaluator::Start ? i : ( dim - CenterEvaluator::Size + i ) ); |
| double s = 0.5 + ii + j; |
| for( int d1=0 ; d1<2 ; d1++ ) evaluator._ccValues[d1][i][j-SupportStart] = Value( depth , ii , s/res , dirichlet , d1!=0 ); |
| } |
| } |
| template< int Degree > |
| void BSplineEvaluationData< Degree >::SetChildCenterEvaluator( typename CenterEvaluator::ChildEvaluator& evaluator , int parentDepth , bool dirichlet ) |
| { |
| evaluator._parentDepth = parentDepth; |
| int dim = BSplineEvaluationData< Degree >::Dimension( parentDepth ) , res = 1<<(parentDepth+1); |
| for( int i=0 ; i<CenterEvaluator::Size ; i++ ) for( int j=ChildSupportStart ; j<=ChildSupportEnd ; j++ ) |
| { |
| int ii = ( i<=CenterEvaluator::Start ? i : ( dim - CenterEvaluator::Size + i ) ); |
| double s = 0.5 + 2*ii + j; |
| for( int d1=0 ; d1<2 ; d1++ ) evaluator._pcValues[d1][i][j-ChildSupportStart] = Value( parentDepth , ii , s/res , dirichlet , d1!=0 ); |
| } |
| } |
| template< int Degree > |
| double BSplineEvaluationData< Degree >::CenterEvaluator::Evaluator::value( int fIdx , int cIdx , bool d ) const |
| { |
| int dd = cIdx-fIdx , res = 1<<(_depth) , dim = Dimension(_depth); |
| if( cIdx<0 || fIdx<0 || cIdx>=res || fIdx>=dim || dd<SupportStart || dd>SupportEnd ) return 0; |
| return _ccValues[d?1:0][ CenterEvaluator::Index( _depth , fIdx ) ][dd-SupportStart]; |
| } |
| template< int Degree > |
| double BSplineEvaluationData< Degree >::CenterEvaluator::ChildEvaluator::value( int fIdx , int cIdx , bool d ) const |
| { |
| int dd = cIdx-2*fIdx , res = 1<<(_parentDepth+1) , dim = Dimension(_parentDepth); |
| if( cIdx<0 || fIdx<0 || cIdx>=res || fIdx>=dim || dd<ChildSupportStart || dd>ChildSupportEnd ) return 0; |
| return _pcValues[d?1:0][ CenterEvaluator::Index( _parentDepth , fIdx ) ][dd-ChildSupportStart]; |
| } |
| template< int Degree > |
| void BSplineEvaluationData< Degree >::SetCornerEvaluator( typename CornerEvaluator::Evaluator& evaluator , int depth , bool dirichlet ) |
| { |
| evaluator._depth = depth; |
| int dim = BSplineEvaluationData< Degree >::Dimension( depth ) , res = 1<<depth; |
| for( int i=0 ; i<CornerEvaluator::Size ; i++ ) for( int j=CornerStart ; j<=CornerEnd ; j++ ) |
| { |
| int ii = ( i<=CornerEvaluator::Start ? i : ( dim - CornerEvaluator::Size + i ) ); |
| double s = ii + j; |
| for( int d1=0 ; d1<2 ; d1++ ) evaluator._ccValues[d1][i][j-CornerStart] = Value( depth , ii , s/res , dirichlet , d1!=0 ); |
| } |
| } |
| template< int Degree > |
| void BSplineEvaluationData< Degree >::SetChildCornerEvaluator( typename CornerEvaluator::ChildEvaluator& evaluator , int parentDepth , bool dirichlet ) |
| { |
| evaluator._parentDepth = parentDepth; |
| int dim = BSplineEvaluationData< Degree >::Dimension( parentDepth ) , res = 1<<(parentDepth+1); |
| for( int i=0 ; i<CornerEvaluator::Size ; i++ ) for( int j=ChildCornerStart ; j<=ChildCornerEnd ; j++ ) |
| { |
| int ii = ( i<=CornerEvaluator::Start ? i : ( dim - CornerEvaluator::Size + i ) ); |
| double s = 2*ii + j; |
| for( int d1=0 ; d1<2 ; d1++ ) evaluator._pcValues[d1][i][j-ChildCornerStart] = Value( parentDepth , ii , s/res , dirichlet , d1!=0 ); |
| } |
| } |
| template< int Degree > |
| void BSplineEvaluationData< Degree >::SetUpSampleEvaluator( UpSampleEvaluator& evaluator , int lowDepth , bool dirichlet ) |
| { |
| evaluator._lowDepth = lowDepth; |
| int lowDim = Dimension(lowDepth); |
| for( int i=0 ; i<UpSampleEvaluator::Size ; i++ ) |
| { |
| int ii = ( i<=UpSampleEvaluator::Start ? i : ( lowDim - UpSampleEvaluator::Size + i ) ); |
| BSplineUpSamplingCoefficients b( lowDepth , ii , dirichlet ); |
| for( int j=0 ; j<UpSampleSize ; j++ ) evaluator._pcValues[i][j] = b[j]; |
| } |
| } |
| template< int Degree > |
| double BSplineEvaluationData< Degree >::CornerEvaluator::Evaluator::value( int fIdx , int cIdx , bool d ) const |
| { |
| int dd = cIdx-fIdx , res = ( 1<<_depth ) + 1 , dim = Dimension(_depth); |
| if( cIdx<0 || fIdx<0 || cIdx>=res || fIdx>=dim || dd<CornerStart || dd>CornerEnd ) return 0; |
| return _ccValues[d?1:0][ CornerEvaluator::Index( _depth , fIdx ) ][dd-CornerStart]; |
| } |
| template< int Degree > |
| double BSplineEvaluationData< Degree >::CornerEvaluator::ChildEvaluator::value( int fIdx , int cIdx , bool d ) const |
| { |
| int dd = cIdx-2*fIdx , res = ( 1<<(_parentDepth+1) ) + 1 , dim = Dimension(_parentDepth); |
| if( cIdx<0 || fIdx<0 || cIdx>=res || fIdx>=dim || dd<ChildCornerStart || dd>ChildCornerEnd ) return 0; |
| return _pcValues[d?1:0][ CornerEvaluator::Index( _parentDepth , fIdx ) ][dd-ChildCornerStart]; |
| } |
| template< int Degree > |
| double BSplineEvaluationData< Degree >::UpSampleEvaluator::value( int pIdx , int cIdx ) const |
| { |
| int dd = cIdx-2*pIdx , pDim = Dimension( _lowDepth ) , cDim = Dimension( _lowDepth+1 ); |
| if( cIdx<0 || pIdx<0 || cIdx>=cDim || pIdx>=pDim || dd<UpSampleStart || dd>UpSampleEnd ) return 0; |
| return _pcValues[ UpSampleEvaluator::Index( _lowDepth , pIdx ) ][dd-UpSampleStart]; |
| } |
|
|
| ////////////////////////////////////////////// |
| // BSplineEvaluationData::BSplineComponents // |
| ////////////////////////////////////////////// |
| template< int Degree > |
| BSplineEvaluationData< Degree >::BSplineComponents::BSplineComponents( int depth , int offset , bool dirichlet ) |
| { |
| int res = 1<<depth; |
| BSplineElements< Degree > elements( res , offset , dirichlet ); |
|
|
| // The first index is the position, the second is the element type |
| Polynomial< Degree > components[Degree+1][Degree+1]; |
| // Generate the elements that can appear in the base function corresponding to the base function at (depth,offset) = (0,0) |
| for( int d=0 ; d<=Degree ; d++ ) for( int dd=0 ; dd<=Degree ; dd++ ) components[d][dd] = Polynomial< Degree >::BSplineComponent( Degree-dd ).shift( -( (Degree+1)/2 ) + d ); |
|
|
| // Now adjust to the desired depth and offset |
| double width = 1. / res; |
| for( int d=0 ; d<=Degree ; d++ ) for( int dd=0 ; dd<=Degree ; dd++ ) components[d][dd] = components[d][dd].scale( width ).shift( width*offset ); |
|
|
| // Now write in the polynomials |
| for( int d=0 ; d<=Degree ; d++ ) |
| { |
| int idx = offset + SupportStart + d; |
| _polys[d] = Polynomial< Degree >(); |
|
|
| if( idx>=0 && idx<res ) for( int dd=0 ; dd<=Degree ; dd++ ) _polys[d] += components[d][dd] * ( ( double )( elements[idx][dd] ) ) / elements.denominator; |
| } |
| } |
|
|
| ////////////////////////////////////////////////////////// |
| // BSplineEvaluationData::BSplineUpSamplingCoefficients // |
| ////////////////////////////////////////////////////////// |
| template< int Degree > |
| BSplineEvaluationData< Degree >::BSplineUpSamplingCoefficients::BSplineUpSamplingCoefficients( int depth , int offset , bool dirichlet ) |
| { |
| // [ 1/8 1/2 3/4 1/2 1/8] |
| // [ 1 , 1 ] -> [ 3/4 , 1/2 , 1/8 ] + [ 1/8 , 1/2 , 3/4 ] = [ 7/8 , 1 , 7/8 ] |
| int dim = Dimension(depth) , _dim = Dimension(depth+1); |
| bool reflect; |
| offset = BSplineData< Degree >::RemapOffset( depth , offset , reflect ); |
| int multiplier = ( dirichlet && reflect ) ? -1 : 1; |
| bool useReflected = Inset || ( offset % ( dim-1 ) ); |
| int b[ UpSampleSize ]; |
| Polynomial< Degree+1 >::BinomialCoefficients( b ); |
|
|
| // Clear the values |
| memset( _coefficients , 0 , sizeof(int) * UpSampleSize ); |
|
|
| // Get the array of coefficients, relative to the origin |
| int* coefficients = _coefficients - ( 2*offset + UpSampleStart ); |
| for( int i=UpSampleStart ; i<=UpSampleEnd ; i++ ) |
| { |
| int _offset = 2*offset+i; |
| _offset = BSplineData< Degree >::RemapOffset( depth+1 , _offset , reflect ); |
| if( useReflected || !reflect ) |
| { |
| int _multiplier = multiplier * ( ( dirichlet && reflect ) ? -1 : 1 ); |
| coefficients[ _offset ] += b[ i-UpSampleStart ] * _multiplier; |
| } |
| // If we are not inset and we are at the boundary, use the reflection as well |
| if( !Inset && ( offset % (dim-1) ) && !( _offset % (_dim-1) ) ) |
| { |
| _offset = BSplineData< Degree >::RemapOffset( depth+1 , _offset , reflect ); |
| int _multiplier = multiplier * ( ( dirichlet && reflect ) ? -1 : 1 ); |
| if( dirichlet ) _multiplier *= -1; |
| coefficients[ _offset ] += b[ i-UpSampleStart ] * _multiplier; |
| } |
| } |
| } |
|
|
| //////////////////////////// |
| // BSplineIntegrationData // |
| //////////////////////////// |
| template< int Degree1 , int Degree2 > |
| double BSplineIntegrationData< Degree1 , Degree2 >::Dot( int depth1 , int off1 , bool dirichlet1 , bool d1 , int depth2 , int off2 , bool dirichlet2 , bool d2 ) |
| { |
| const int _Degree1 = (d1 ? (Degree1-1) : Degree1) , _Degree2 = (d2 ? (Degree2-1) : Degree2); |
| int sums[ Degree1+1 ][ Degree2+1 ]; |
|
|
| int depth = std::max< int >( depth1 , depth2 ); |
|
|
| BSplineElements< Degree1 > b1( 1<<depth1 , off1 , dirichlet1 ); |
| BSplineElements< Degree2 > b2( 1<<depth2 , off2 , dirichlet2 ); |
|
|
| { |
| BSplineElements< Degree1 > b; |
| while( depth1<depth ) b=b1 , b.upSample( b1 ) , depth1++; |
| } |
| { |
| BSplineElements< Degree2 > b; |
| while( depth2<depth ) b=b2 , b.upSample( b2 ) , depth2++; |
| } |
|
|
| BSplineElements< Degree1-1 > db1; |
| BSplineElements< Degree2-1 > db2; |
| b1.differentiate( db1 ) , b2.differentiate( db2 ); |
|
|
| int start1=-1 , end1=-1 , start2=-1 , end2=-1; |
| for( int i=0 ; i<int( b1.size() ) ; i++ ) |
| { |
| for( int j=0 ; j<=Degree1 ; j++ ) |
| { |
| if( b1[i][j] && start1==-1 ) start1 = i; |
| if( b1[i][j] ) end1 = i+1; |
| } |
| for( int j=0 ; j<=Degree2 ; j++ ) |
| { |
| if( b2[i][j] && start2==-1 ) start2 = i; |
| if( b2[i][j] ) end2 = i+1; |
| } |
| } |
| if( start1==end1 || start2==end2 || start1>=end2 || start2>=end1 ) return 0.; |
| int start = std::max< int >( start1 , start2 ) , end = std::min< int >( end1 , end2 ); |
| memset( sums , 0 , sizeof( sums ) ); |
|
|
| // Iterate over the support |
| for( int i=start ; i<end ; i++ ) |
| // Iterate over all pairs of elements within a node |
| for( int j=0 ; j<=_Degree1 ; j++ ) for( int k=0 ; k<=_Degree2 ; k++ ) |
| // Accumulate the product of the coefficients |
| sums[j][k] += ( d1 ? db1[i][j] : b1[i][j] ) * ( d2 ? db2[i][k] : b2[i][k] ); |
|
|
| double _dot = 0; |
| if( d1 && d2 ) |
| { |
| double integrals[ Degree1 ][ Degree2 ]; |
| SetBSplineElementIntegrals< Degree1-1 , Degree2-1 >( integrals ); |
| for( int j=0 ; j<=_Degree1 ; j++ ) for( int k=0 ; k<=_Degree2 ; k++ ) _dot += integrals[j][k] * sums[j][k]; |
| } |
| else if( d1 ) |
| { |
| double integrals[ Degree1 ][ Degree2+1 ]; |
| SetBSplineElementIntegrals< Degree1-1 , Degree2 >( integrals ); |
| for( int j=0 ; j<=_Degree1 ; j++ ) for( int k=0 ; k<=_Degree2 ; k++ ) _dot += integrals[j][k] * sums[j][k]; |
| } |
| else if( d2 ) |
| { |
| double integrals[ Degree1+1 ][ Degree2 ]; |
| SetBSplineElementIntegrals< Degree1 , Degree2-1 >( integrals ); |
| for( int j=0 ; j<=_Degree1 ; j++ ) for( int k=0 ; k<=_Degree2 ; k++ ) _dot += integrals[j][k] * sums[j][k]; |
| } |
| else |
| { |
| double integrals[ Degree1+1 ][ Degree2+1 ]; |
| SetBSplineElementIntegrals< Degree1 , Degree2 >( integrals ); |
| for( int j=0 ; j<=_Degree1 ; j++ ) for( int k=0 ; k<=_Degree2 ; k++ ) _dot += integrals[j][k] * sums[j][k]; |
| } |
|
|
| _dot /= b1.denominator; |
| _dot /= b2.denominator; |
| if ( d1 && d2 ) return _dot * (1<<depth); |
| else if( d1 || d2 ) return _dot; |
| else return _dot / (1<<depth); |
| } |
| template< int Degree1 , int Degree2 > |
| void BSplineIntegrationData< Degree1, Degree2 >::SetIntegrator( typename FunctionIntegrator::Integrator& integrator , int depth , bool dirichlet1 , bool dirichlet2 ) |
| { |
| integrator._depth = depth; |
| int dim = BSplineEvaluationData< Degree2 >::Dimension( depth ); |
| for( int i=0 ; i<FunctionIntegrator::Size ; i++ ) for( int j=OverlapStart ; j<=OverlapEnd ; j++ ) |
| { |
| int ii = ( i<=FunctionIntegrator::Start ? i : ( dim - FunctionIntegrator::Size + i ) ); |
| for( int d1=0 ; d1<2 ; d1++ ) for( int d2=0 ; d2<2 ; d2++ ) integrator._ccIntegrals[d1][d2][i][j-OverlapStart] = Dot( depth , ii , dirichlet1 , d1!=0 , depth , ii+j , dirichlet2 , d2!=0 ); |
| } |
| } |
| template< int Degree1 , int Degree2 > |
| void BSplineIntegrationData< Degree1, Degree2 >::SetChildIntegrator( typename FunctionIntegrator::ChildIntegrator& integrator , int parentDepth , bool dirichlet1 , bool dirichlet2 ) |
| { |
| integrator._parentDepth = parentDepth; |
| int dim = BSplineEvaluationData< Degree2 >::Dimension( parentDepth ); |
| for( int i=0 ; i<FunctionIntegrator::Size ; i++ ) for( int j=ChildOverlapStart ; j<=ChildOverlapEnd ; j++ ) |
| { |
| int ii = ( i<=FunctionIntegrator::Start ? i : ( dim - FunctionIntegrator::Size + i ) ); |
| for( int d1=0 ; d1<2 ; d1++ ) for( int d2=0 ; d2<2 ; d2++ ) integrator._pcIntegrals[d1][d2][i][j-ChildOverlapStart] = Dot( parentDepth , ii , dirichlet1 , d1!=0 , parentDepth+1 , 2*ii+j , dirichlet2 , d2!=0 ); |
| } |
| } |
| template< int Degree1 , int Degree2 > |
| double BSplineIntegrationData< Degree1 , Degree2 >::FunctionIntegrator::Integrator::dot( int off1 , int off2 , bool d1 , bool d2 ) const |
| { |
| int d = off2-off1 , dim1 = BSplineEvaluationData< Degree1 >::Dimension( _depth ) , dim2 = BSplineEvaluationData< Degree2 >::Dimension( _depth ); |
| if( off1<0 || off2<0 || off1>=dim1 || off2>=dim2 || d<OverlapStart || d>OverlapEnd ) return 0; |
| return _ccIntegrals[d1?1:0][d2?1:0][ FunctionIntegrator::Index( _depth , off1 ) ][d-OverlapStart]; |
| } |
| template< int Degree1 , int Degree2 > |
| double BSplineIntegrationData< Degree1 , Degree2 >::FunctionIntegrator::ChildIntegrator::dot( int off1 , int off2 , bool d1 , bool d2 ) const |
| { |
| int d = off2-2*off1 , dim1 = BSplineEvaluationData< Degree1 >::Dimension( _parentDepth ) , dim2 = BSplineEvaluationData< Degree2 >::Dimension( _parentDepth+1 ); |
| if( off1<0 || off2<0 || off1>=dim1 || off2>=dim2 || d<ChildOverlapStart || d>ChildOverlapEnd ) return 0; |
| return _pcIntegrals[d1?1:0][d2?1:0][ FunctionIntegrator::Index( _parentDepth , off1 ) ][d-ChildOverlapStart]; |
| } |
| ///////////////// |
| // BSplineData // |
| ///////////////// |
| #define MODULO( A , B ) ( (A)<0 ? ( (B)-((-(A))%(B)) ) % (B) : (A) % (B) ) |
| template< int Degree > |
| int BSplineData< Degree >::RemapOffset( int depth , int offset , bool& reflect ) |
| { |
| const int I = ( Degree&1 ) ? 0 : 1; |
| int dim = Dimension( depth ); |
| offset = MODULO( offset , 2*(dim-1+I) ); |
| reflect = offset>=dim; |
| if( reflect ) return 2*(dim-1+I) - (offset+I); |
| else return offset; |
| } |
| #undef MODULO |
|
|
| template< int Degree > BSplineData< Degree >::BSplineData( void ){ functionCount = sampleCount = 0; } |
|
|
|
|
| template< int Degree > |
| void BSplineData< Degree >::set( int maxDepth , bool dirichlet ) |
| { |
| _dirichlet = dirichlet; |
|
|
| depth = maxDepth; |
| functionCount = TotalFunctionCount( depth ); |
| sampleCount = TotalSampleCount( depth ); |
| baseBSplines = NewPointer< typename BSplineEvaluationData< Degree >::BSplineComponents >( functionCount ); |
|
|
| for( size_t i=0 ; i<functionCount ; i++ ) |
| { |
| int d , off; |
| FactorFunctionIndex( (int)i , d , off ); |
| baseBSplines[i] = typename BSplineEvaluationData< Degree >::BSplineComponents( d , off , _dirichlet ); |
| } |
| } |
|
|
| ///////////////////// |
| // BSplineElements // |
| ///////////////////// |
| template< int Degree > |
| BSplineElements< Degree >::BSplineElements( int res , int offset , bool dirichlet ) |
| { |
| denominator = 1; |
| std::vector< BSplineElementCoefficients< Degree > >::resize( res , BSplineElementCoefficients< Degree >() ); |
|
|
| // If we have primal dirichlet constraints, the boundary functions are necessarily zero |
| if( _Primal && dirichlet && !(offset%res) ) return; |
|
|
| // Construct the B-Spline |
| for( int i=0 ; i<=Degree ; i++ ) |
| { |
| int idx = -_Off + offset + i; |
| if( idx>=0 && idx<res ) (*this)[idx][i] = 1; |
| } |
| // Fold in the periodic instances (which cancels the negation) |
| _addPeriodic< true >( _RotateLeft ( offset , res ) , false ) , _addPeriodic< false >( _RotateRight( offset , res ) , false ); |
|
|
| // Recursively fold in the boundaries |
| if( _Primal && !(offset%res) ) return; |
|
|
| // Fold in the reflected instance (which may require negation) |
| _addPeriodic< true >( _ReflectLeft( offset , res ) , dirichlet ) , _addPeriodic< false >( _ReflectRight( offset , res ) , dirichlet ); |
| } |
| template< int Degree > int BSplineElements< Degree >::_ReflectLeft ( int offset , int res ){ return (Degree&1) ? -offset : -1-offset; } |
| template< int Degree > int BSplineElements< Degree >::_ReflectRight( int offset , int res ){ return (Degree&1) ? 2*res-offset : 2*res-1-offset; } |
| template< int Degree > int BSplineElements< Degree >::_RotateLeft ( int offset , int res ){ return offset-2*res; } |
| template< int Degree > int BSplineElements< Degree >::_RotateRight ( int offset , int res ){ return offset+2*res; } |
|
|
| template< int Degree > |
| template< bool Left > |
| void BSplineElements< Degree >::_addPeriodic( int offset , bool negate ) |
| { |
| int res = int( std::vector< BSplineElementCoefficients< Degree > >::size() ); |
| bool set = false; |
| // Add in the corresponding B-spline elements (possibly negated) |
| for( int i=0 ; i<=Degree ; i++ ) |
| { |
| int idx = -_Off + offset + i; |
| if( idx>=0 && idx<res ) (*this)[idx][i] += negate ? -1 : 1 , set = true; |
| } |
| // If there is a change for additional overlap, give it a go |
| if( set ) _addPeriodic< Left >( Left ? _RotateLeft( offset , res ) : _RotateRight( offset , res ) , negate ); |
| } |
| template< int Degree > |
| void BSplineElements< Degree >::upSample( BSplineElements< Degree >& high ) const |
| { |
| int bCoefficients[ BSplineEvaluationData< Degree >::UpSampleSize ]; |
| Polynomial< Degree+1 >::BinomialCoefficients( bCoefficients ); |
|
|
| high.resize( std::vector< BSplineElementCoefficients< Degree > >::size()*2 ); |
| high.assign( high.size() , BSplineElementCoefficients< Degree >() ); |
| // [NOTE] We have flipped the order of the B-spline elements |
| for( int i=0 ; i<int(std::vector< BSplineElementCoefficients< Degree > >::size()) ; i++ ) for( int j=0 ; j<=Degree ; j++ ) |
| { |
| // At index I , B-spline element J corresponds to a B-spline centered at: |
| // I - SupportStart - J |
| int idx = i - BSplineEvaluationData< Degree >::SupportStart - j; |
| for( int k=BSplineEvaluationData< Degree >::UpSampleStart ; k<=BSplineEvaluationData< Degree >::UpSampleEnd ; k++ ) |
| { |
| // Index idx at the coarser resolution gets up-sampled into indices: |
| // 2*idx + [UpSampleStart,UpSampleEnd] |
| // at the finer resolution |
| int _idx = 2*idx + k; |
| // Compute the index of the B-spline element relative to 2*i and 2*i+1 |
| int _j1 = -_idx + 2*i - BSplineEvaluationData< Degree >::SupportStart , _j2 = -_idx + 2*i + 1 - BSplineEvaluationData< Degree >::SupportStart; |
| if( _j1>=0 && _j1<=Degree ) high[2*i+0][_j1] += (*this)[i][j] * bCoefficients[k-BSplineEvaluationData< Degree >::UpSampleStart]; |
| if( _j2>=0 && _j2<=Degree ) high[2*i+1][_j2] += (*this)[i][j] * bCoefficients[k-BSplineEvaluationData< Degree >::UpSampleStart]; |
| } |
| } |
| high.denominator = denominator<<Degree; |
| } |
|
|
| template< int Degree > |
| void BSplineElements< Degree >::differentiate( BSplineElements< Degree-1 >& d ) const |
| { |
| d.resize( std::vector< BSplineElementCoefficients< Degree > >::size() ); |
| d.assign( d.size() , BSplineElementCoefficients< Degree-1 >() ); |
| for( int i=0 ; i<int(std::vector< BSplineElementCoefficients< Degree > >::size()) ; i++ ) for( int j=0 ; j<=Degree ; j++ ) |
| { |
| if( j-1>=0 ) d[i][j-1] -= (*this)[i][j]; |
| if( j<Degree ) d[i][j ] += (*this)[i][j]; |
| } |
| d.denominator = denominator; |
| } |
|
|
| // If we were really good, we would implement this integral table to store |
| // rational values to improve precision... |
| template< int Degree1 , int Degree2 > |
| void SetBSplineElementIntegrals( double integrals[Degree1+1][Degree2+1] ) |
| { |
| for( int i=0 ; i<=Degree1 ; i++ ) |
| { |
| Polynomial< Degree1 > p1 = Polynomial< Degree1 >::BSplineComponent( Degree1-i ); |
| for( int j=0 ; j<=Degree2 ; j++ ) |
| { |
| Polynomial< Degree2 > p2 = Polynomial< Degree2 >::BSplineComponent( Degree2-j ); |
| integrals[i][j] = ( p1 * p2 ).integral( 0 , 1 ); |
| } |
| } |
| } |
|
|