663 lines
20 KiB
C++
663 lines
20 KiB
C++
/**************************************************************************
|
|
* *
|
|
* Copyright (C) 1992, Silicon Graphics, Inc. *
|
|
* *
|
|
* These coded instructions, statements, and computer programs contain *
|
|
* unpublished proprietary information of Silicon Graphics, Inc., and *
|
|
* are protected by Federal copyright law. They may not be disclosed *
|
|
* to third parties or copied or duplicated in any form, in whole or *
|
|
* in part, without the prior written consent of Silicon Graphics, Inc. *
|
|
* *
|
|
**************************************************************************/
|
|
|
|
/*
|
|
* tobezier.c++ - $Revision: 1.6 $
|
|
* Derrick Burns - 1991
|
|
*/
|
|
|
|
#include "glimport.h"
|
|
#include "myassert.h"
|
|
#include "mystdio.h"
|
|
#include "mystring.h"
|
|
#include "quilt.h"
|
|
#include "knotvect.h"
|
|
|
|
/* local type definitions */
|
|
struct Breakpt { /* breakpoints */
|
|
Knot value; /* value */
|
|
int multi; /* multiplicity */
|
|
int def; /* deficit */
|
|
};
|
|
|
|
struct Knotspec { /* knotvector format */
|
|
long order; /* order of spline */
|
|
Knot_ptr inkbegin; /* input knot sequence */
|
|
Knot_ptr inkend; /* location after last knot */
|
|
Knot_ptr outkbegin; /* in-process knot subsequence */
|
|
Knot_ptr outkend; /* location after last knot */
|
|
Knot_ptr kleft; /* */
|
|
Knot_ptr kright; /* */
|
|
Knot_ptr kfirst; /* */
|
|
Knot_ptr klast; /* */
|
|
Knot_ptr sbegin; /* conversion factor values */
|
|
Breakpt * bbegin; /* in-process breakpoints */
|
|
Breakpt * bend; /* last breakpoint */
|
|
int ncoords; /* coordinates per control point */
|
|
int prestride; /* stride between input points */
|
|
int poststride; /* stride between output points */
|
|
int preoffset; /* scaled point offset */
|
|
int postoffset; /* scaled point offset */
|
|
int prewidth; /* width of dimension */
|
|
int postwidth; /* width of dimension */
|
|
int istransformed; /* was dimension transformed */
|
|
Knotspec * next; /* next knotspec */
|
|
Knotspec * kspectotrans; /* knotspec in transformation direction */
|
|
|
|
Knotspec( void );
|
|
~Knotspec( void );
|
|
void factors( void );
|
|
void insert( REAL * );
|
|
void preselect();
|
|
void select( void );
|
|
void copy( INREAL *, REAL * );
|
|
void breakpoints( void );
|
|
void knots( void );
|
|
void transform( REAL * );
|
|
void showpts( REAL * );
|
|
|
|
void pt_io_copy( REAL *, INREAL * );
|
|
void pt_oo_copy( REAL *, REAL * );
|
|
void pt_oo_sum( REAL*, REAL*, REAL*, Knot, Knot );
|
|
};
|
|
|
|
struct Splinespec { /* a non-uniform tensor element */
|
|
Splinespec( int );
|
|
~Splinespec(void);
|
|
Knotspec *kspec; /* format of each param. dir. */
|
|
int dim; /* domain dimension */
|
|
REAL * outcpts; /* Bezier control points */
|
|
|
|
void kspecinit( Knotvector & );
|
|
void kspecinit( Knotvector &, Knotvector & );
|
|
void select( void );
|
|
void layout( long );
|
|
void setupquilt( Quilt_ptr );
|
|
void copy( INREAL * );
|
|
void transform( void );
|
|
};
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
* Quilt::toBezier - convert from NURBS to rational Bezier
|
|
*-----------------------------------------------------------------------------
|
|
*/
|
|
|
|
void
|
|
Quilt::toBezier(
|
|
Knotvector& knotvector, /* a knot vector */
|
|
INREAL *ctlpts, /* input contol points */
|
|
long ncoords ) /* number of coordinates per control point */
|
|
{
|
|
Splinespec spline( 1 );
|
|
spline.kspecinit( knotvector );
|
|
spline.select();
|
|
spline.layout( ncoords );
|
|
spline.setupquilt( this );
|
|
spline.copy( ctlpts );
|
|
spline.transform();
|
|
}
|
|
|
|
void
|
|
Quilt::toBezier(
|
|
Knotvector& sknotvector, /* a knot vector */
|
|
Knotvector& tknotvector, /* a knot vector */
|
|
INREAL *ctlpts, /* input contol points */
|
|
long ncoords ) /* number of coordinates per control point */
|
|
{
|
|
Splinespec spline( 2 );
|
|
spline.kspecinit( sknotvector, tknotvector );
|
|
spline.select();
|
|
spline.layout( ncoords );
|
|
spline.setupquilt( this );
|
|
spline.copy( ctlpts );
|
|
spline.transform();
|
|
}
|
|
Splinespec::Splinespec( int dimen )
|
|
{
|
|
dim = dimen;
|
|
}
|
|
|
|
Splinespec::~Splinespec( void )
|
|
{
|
|
/* Note: do NOT delete 'outcpts' here since its address (not contents)
|
|
* is copied in 'cpts' in this file in function Splinespec::setupquilt().
|
|
* This block of memory will eventually be deleted in file quilt.c++ in
|
|
* function Quilt::deleteMe() through 'cpts' so do NOT delete it here!
|
|
*/
|
|
Knotspec *ktrav= kspec; //start at beginning of list
|
|
while (ktrav != 0) { //any items to delete?
|
|
Knotspec *deleteThis= ktrav; //remember to delete this
|
|
ktrav= ktrav->next; //go to next item if any
|
|
delete deleteThis; //delete it
|
|
}
|
|
} /* ~Splinespec() */
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
* Splinespec::kspecinit - initialize Splinespec structure
|
|
*
|
|
* Client: Quilt::toBezier
|
|
*-----------------------------------------------------------------------------
|
|
*/
|
|
|
|
void
|
|
Splinespec::kspecinit( Knotvector& knotvector )
|
|
{
|
|
kspec = new Knotspec;
|
|
kspec->inkbegin = knotvector.knotlist;
|
|
kspec->inkend = knotvector.knotlist + knotvector.knotcount;
|
|
kspec->prestride = (int) knotvector.stride;
|
|
kspec->order = knotvector.order;
|
|
kspec->next = NULL;
|
|
}
|
|
|
|
void
|
|
Splinespec::kspecinit( Knotvector& sknotvector, Knotvector& tknotvector )
|
|
{
|
|
kspec = new Knotspec;
|
|
Knotspec *tkspec = new Knotspec;
|
|
|
|
kspec->inkbegin = sknotvector.knotlist;
|
|
kspec->inkend = sknotvector.knotlist + sknotvector.knotcount;
|
|
kspec->prestride = (int) sknotvector.stride;
|
|
kspec->order = sknotvector.order;
|
|
kspec->next = tkspec;
|
|
|
|
tkspec->inkbegin = tknotvector.knotlist;
|
|
tkspec->inkend = tknotvector.knotlist + tknotvector.knotcount;
|
|
tkspec->prestride = (int) tknotvector.stride;
|
|
tkspec->order = tknotvector.order;
|
|
tkspec->next = NULL;
|
|
}
|
|
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
* Splinespec::select - select the subsegments to copy
|
|
*
|
|
* Client: gl_quilt_to_bezier
|
|
*-----------------------------------------------------------------------------
|
|
*/
|
|
|
|
void
|
|
Splinespec::select( )
|
|
{
|
|
for( Knotspec *knotspec = kspec; knotspec; knotspec = knotspec->next ) {
|
|
knotspec->preselect();
|
|
knotspec->select();
|
|
}
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
* Splinespec::layout -
|
|
*
|
|
* Client: gl_quilt_to_bezier
|
|
*-----------------------------------------------------------------------------
|
|
*/
|
|
|
|
void
|
|
Splinespec::layout( long ncoords )
|
|
{
|
|
|
|
long stride = ncoords;
|
|
for( Knotspec *knotspec = kspec; knotspec; knotspec=knotspec->next ) {
|
|
knotspec->poststride = (int) stride;
|
|
stride *= (long)((knotspec->bend-knotspec->bbegin)*knotspec->order + knotspec->postoffset);
|
|
knotspec->preoffset *= knotspec->prestride;
|
|
knotspec->prewidth *= knotspec->poststride;
|
|
knotspec->postwidth *= knotspec->poststride;
|
|
knotspec->postoffset *= knotspec->poststride;
|
|
knotspec->ncoords = (int) ncoords;
|
|
}
|
|
outcpts = new REAL[stride];
|
|
assert( outcpts != 0 );
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
* Splinespec::copy - copy the control points of current subobject
|
|
*
|
|
* Client: gl_quilt_to_bezier
|
|
*-----------------------------------------------------------------------------
|
|
*/
|
|
|
|
void
|
|
Splinespec::copy( INREAL *incpts )
|
|
{
|
|
kspec->copy( incpts, outcpts );
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
* Splinespec::setupquilt - assign all quilt variables from knotspec
|
|
*
|
|
* Client: gl_quilt_to_bezier
|
|
*-----------------------------------------------------------------------------
|
|
*/
|
|
|
|
void
|
|
Splinespec::setupquilt( Quilt_ptr quilt )
|
|
{
|
|
Quiltspec_ptr qspec = quilt->qspec;
|
|
quilt->eqspec = qspec + dim;
|
|
for( Knotspec *knotspec = kspec; knotspec; knotspec=knotspec->next, qspec++ ) {
|
|
qspec->stride = knotspec->poststride;
|
|
qspec->width = knotspec->bend - knotspec->bbegin;
|
|
qspec->order = (int) knotspec->order;
|
|
qspec->offset = knotspec->postoffset;
|
|
qspec->index = 0;
|
|
qspec->bdry[0] = (knotspec->kleft == knotspec->kfirst) ? 1 : 0;
|
|
qspec->bdry[1] = (knotspec->kright == knotspec->klast) ? 1 : 0;
|
|
qspec->breakpoints = new Knot[qspec->width+1];
|
|
Knot_ptr k = qspec->breakpoints;
|
|
for( Breakpt *bk = knotspec->bbegin; bk <= knotspec->bend; bk++ )
|
|
*(k++) = bk->value;
|
|
}
|
|
quilt->cpts = outcpts;
|
|
quilt->next = 0;
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
* Splinespec::transform - convert a spline to Bezier format
|
|
*
|
|
* Client: gl_quilt_to_bezier
|
|
*-----------------------------------------------------------------------------
|
|
*/
|
|
|
|
void
|
|
Splinespec::transform( void )
|
|
{
|
|
for( Knotspec *knotspec = kspec; knotspec; knotspec=knotspec->next )
|
|
knotspec->istransformed = 0;
|
|
|
|
for( knotspec = kspec; knotspec; knotspec=knotspec->next ) {
|
|
for( Knotspec *kspec2 = kspec; kspec2; kspec2=kspec2->next )
|
|
kspec2->kspectotrans = knotspec;
|
|
kspec->transform( outcpts );
|
|
knotspec->istransformed = 1;
|
|
}
|
|
}
|
|
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
* Knotspec::Knotspec - constuct a knot spec
|
|
*-----------------------------------------------------------------------------
|
|
*/
|
|
|
|
Knotspec::Knotspec( void )
|
|
{
|
|
bbegin = 0;
|
|
sbegin = 0;
|
|
outkbegin = 0;
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
* Knotspec::copy - copy the control points along minor direction
|
|
*
|
|
* Client: Splinespec::copy
|
|
*-----------------------------------------------------------------------------
|
|
*/
|
|
|
|
void
|
|
Knotspec::copy( INREAL *inpt, REAL *outpt )
|
|
{
|
|
inpt = (INREAL *) (((char *) inpt) + preoffset);
|
|
|
|
if( next ) {
|
|
for( REAL *lpt=outpt+prewidth; outpt != lpt; outpt += poststride ) {
|
|
next->copy( inpt, outpt );
|
|
inpt = (INREAL *) (((char *) inpt) + prestride);
|
|
}
|
|
} else {
|
|
for( REAL *lpt=outpt+prewidth; outpt != lpt; outpt += poststride ) {
|
|
pt_io_copy( outpt, inpt );
|
|
inpt = (INREAL *) (((char *) inpt) + prestride);
|
|
}
|
|
}
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
* Knotspec::showpts - print out points before transformation
|
|
*
|
|
* Client: Knotspec::select
|
|
*-----------------------------------------------------------------------------
|
|
*/
|
|
void
|
|
Knotspec::showpts( REAL *outpt )
|
|
{
|
|
if( next ) {
|
|
for( REAL *lpt=outpt+prewidth; outpt != lpt; outpt += poststride )
|
|
next->showpts( outpt );
|
|
} else {
|
|
for( REAL *lpt=outpt+prewidth; outpt != lpt; outpt += poststride )
|
|
dprintf( "show %g %g %g\n", outpt[0], outpt[1], outpt[2] );
|
|
}
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
* Knotspec::factors - precompute scale factors
|
|
* - overwrites knot vector, actual new knot vector is NOT produced
|
|
*
|
|
* Client: Knotspec::select
|
|
*-----------------------------------------------------------------------------
|
|
*/
|
|
|
|
void
|
|
Knotspec::factors( void )
|
|
{
|
|
Knot *mid = (outkend - 1) - order + bend->multi;
|
|
Knot_ptr fptr = sbegin;
|
|
|
|
for( Breakpt *bpt = bend; bpt >= bbegin; bpt-- ) {
|
|
mid -= bpt->multi; // last knot less than knot to insert
|
|
int def = bpt->def - 1; // number of knots to insert
|
|
if( def <= 0 ) continue;
|
|
Knot kv = bpt->value; // knot to insert
|
|
|
|
Knot *kf = (mid-def) + (order-1);
|
|
for( Knot *kl = kf + def; kl != kf; kl-- ) {
|
|
Knot *kh, *kt;
|
|
for( kt=kl, kh=mid; kt != kf; kh--, kt-- )
|
|
*(fptr++) = (kv - *kh) / (*kt - *kh);
|
|
*kl = kv;
|
|
}
|
|
}
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
* Knotspec::insert - convert subobject in direction of kspec into Bezier
|
|
*
|
|
* Client: Knotspec::transform
|
|
*-----------------------------------------------------------------------------
|
|
*/
|
|
|
|
void
|
|
Knotspec::insert( REAL *p )
|
|
{
|
|
Knot_ptr fptr = sbegin;
|
|
REAL *srcpt = p + prewidth - poststride;
|
|
REAL *dstpt = p + postwidth + postoffset - poststride;
|
|
Breakpt *bpt = bend;
|
|
|
|
for( REAL *pend = srcpt - poststride*bpt->def; srcpt != pend; pend +=poststride ) {
|
|
REAL *p1 = srcpt;
|
|
for( REAL *p2 = srcpt-poststride; p2 != pend; p1 = p2, p2 -= poststride ) {
|
|
pt_oo_sum( p1, p1, p2, *fptr, 1.0-*fptr );
|
|
fptr++;
|
|
}
|
|
}
|
|
|
|
for( --bpt; bpt >= bbegin; bpt-- ) {
|
|
|
|
for( int multi = bpt->multi; multi > 0; multi-- ) {
|
|
pt_oo_copy( dstpt, srcpt );
|
|
dstpt -= poststride;
|
|
srcpt -= poststride;
|
|
}
|
|
|
|
for( REAL *pend = srcpt - poststride*bpt->def; srcpt != pend; pend +=poststride, dstpt-=poststride ) {
|
|
pt_oo_copy( dstpt, srcpt );
|
|
REAL *p1 = srcpt;
|
|
|
|
for( REAL *p2 = srcpt-poststride; p2 != pend; p1=p2, p2 -= poststride ) {
|
|
pt_oo_sum( p1, p1, p2, *fptr, 1.0-*fptr );
|
|
fptr++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
* Knotspec::preselect - initialize kspec for processing
|
|
*
|
|
* Client: Splinespec::select
|
|
*-----------------------------------------------------------------------------
|
|
*/
|
|
|
|
void
|
|
Knotspec::preselect( void )
|
|
{
|
|
Knot kval;
|
|
|
|
/* position klast after last knot of "last" breakpoint */
|
|
for( klast = inkend - order, kval = *klast; klast != inkend; klast++ )
|
|
if( ! identical( *klast, kval ) ) break;
|
|
|
|
/* position kfirst after last knot of "first" breakpoint */
|
|
for( kfirst = inkbegin+order-1, kval= *kfirst; kfirst != inkend; kfirst++ )
|
|
if( ! identical( *kfirst, kval ) ) break;
|
|
|
|
/* compute multiplicity of first breakpoint */
|
|
for( Knot_ptr k = kfirst - 1; k >= inkbegin; k-- )
|
|
if( ! identical( kval, *k ) ) break;
|
|
k++;
|
|
|
|
/* allocate space for breakpoints -
|
|
use worst case estimate on number of breakpoints */
|
|
|
|
bbegin = new Breakpt[(klast - kfirst)+1];
|
|
/* record multiplicity and value of first breakpoint */
|
|
bbegin->multi = kfirst - k;
|
|
bbegin->value = kval;
|
|
bend = bbegin;
|
|
|
|
kleft = kright = kfirst;
|
|
}
|
|
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
* Knotspec::select - Knotspec::select segments and precompute scale factors
|
|
*
|
|
* Client: Splinespec::select
|
|
*-----------------------------------------------------------------------------
|
|
*/
|
|
|
|
void
|
|
Knotspec::select( void )
|
|
{
|
|
breakpoints();
|
|
knots();
|
|
factors();
|
|
|
|
preoffset = kleft - (inkbegin + order);
|
|
postwidth = (int)((bend - bbegin) * order);
|
|
prewidth = (int)((outkend - outkbegin) - order);
|
|
postoffset = (bbegin->def > 1) ? (bbegin->def-1) : 0;
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
* Knotspec::breakpoints - compute breakpoints for knotspec
|
|
*
|
|
* Client: Knotspec::select
|
|
*-----------------------------------------------------------------------------
|
|
*/
|
|
|
|
void
|
|
Knotspec::breakpoints( void )
|
|
{
|
|
Breakpt *ubpt = bbegin;
|
|
Breakpt *ubend = bend;
|
|
long nfactors = 0;
|
|
|
|
ubpt->value = ubend->value;
|
|
ubpt->multi = ubend->multi;
|
|
|
|
kleft = kright;
|
|
|
|
for( ; kright != klast; kright++ ) {
|
|
if ( identical(*kright,ubpt->value) ) {
|
|
(ubpt->multi)++;
|
|
} else {
|
|
ubpt->def = (int) (order - ubpt->multi);
|
|
nfactors += (ubpt->def * (ubpt->def - 1)) / 2;
|
|
(++ubpt)->value = *kright;
|
|
ubpt->multi = 1;
|
|
}
|
|
}
|
|
ubpt->def = (int) (order - ubpt->multi);
|
|
nfactors += (ubpt->def * (ubpt->def - 1)) / 2;
|
|
|
|
bend = ubpt;
|
|
|
|
if( nfactors ) {
|
|
sbegin = new Knot[nfactors];
|
|
} else {
|
|
sbegin = NULL;
|
|
}
|
|
}
|
|
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
* Knotspec::knots - copy relevant subsequence of knots into temporary area
|
|
*
|
|
* Client: Knotspec::select
|
|
*-----------------------------------------------------------------------------
|
|
*/
|
|
|
|
void
|
|
Knotspec::knots( void )
|
|
{
|
|
Knot_ptr inkpt = kleft - order;
|
|
Knot_ptr inkend = kright + bend->def;
|
|
|
|
/* allocate space for knots and factors */
|
|
outkbegin = new Knot[inkend-inkpt];
|
|
for( Knot_ptr outkpt = outkbegin; inkpt != inkend; inkpt++, outkpt++ )
|
|
*outkpt = *inkpt;
|
|
|
|
outkend = outkpt;
|
|
}
|
|
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
* Knotspec::transform - convert a spline along a given direction
|
|
*
|
|
* Client: Splienspec::transform
|
|
*-----------------------------------------------------------------------------
|
|
*/
|
|
|
|
void
|
|
Knotspec::transform( REAL *p )
|
|
{
|
|
if( next ) {
|
|
if( this == kspectotrans ) {
|
|
next->transform( p );
|
|
} else {
|
|
if( istransformed ) {
|
|
p += postoffset;
|
|
for( REAL *pend = p + postwidth; p != pend; p += poststride )
|
|
next->transform( p );
|
|
} else {
|
|
REAL *pend = p + prewidth;
|
|
for( ; p != pend; p += poststride )
|
|
next->transform( p );
|
|
}
|
|
}
|
|
} else {
|
|
if( this == kspectotrans ) {
|
|
insert( p );
|
|
} else {
|
|
if( istransformed ) {
|
|
p += postoffset;
|
|
for( REAL *pend = p + postwidth; p != pend; p += poststride )
|
|
kspectotrans->insert( p );
|
|
} else {
|
|
REAL *pend = p + prewidth;
|
|
for( ; p != pend; p += poststride )
|
|
kspectotrans->insert( p );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
* Knotspec::~Knotspec - free space alocated for knotspec
|
|
*-----------------------------------------------------------------------------
|
|
*/
|
|
|
|
Knotspec::~Knotspec( void )
|
|
{
|
|
if( bbegin ) delete[] bbegin;
|
|
if( sbegin ) delete[] sbegin;
|
|
if( outkbegin ) delete[] outkbegin;
|
|
}
|
|
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
* pt_io_copy - make internal copy of input cntrl pt. of x coords
|
|
*-----------------------------------------------------------------------------
|
|
*/
|
|
|
|
void
|
|
Knotspec::pt_io_copy( REAL *topt, INREAL *frompt )
|
|
{
|
|
switch( ncoords ) {
|
|
case 4:
|
|
topt[3] = (REAL) frompt[3];
|
|
case 3:
|
|
topt[2] = (REAL) frompt[2];
|
|
case 2:
|
|
topt[1] = (REAL) frompt[1];
|
|
case 1:
|
|
topt[0] = (REAL) frompt[0];
|
|
break;
|
|
default: {
|
|
for( int i = 0; i < ncoords; i++ )
|
|
*topt++ = (REAL) *frompt++;
|
|
}
|
|
}
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
* pt_oo_copy - make internal copy of internal cntrl pt. of x coords
|
|
*-----------------------------------------------------------------------------
|
|
*/
|
|
|
|
void
|
|
Knotspec::pt_oo_copy( REAL *topt, REAL *frompt )
|
|
{
|
|
switch( ncoords ) {
|
|
case 4:
|
|
topt[3] = frompt[3];
|
|
case 3:
|
|
topt[2] = frompt[2];
|
|
case 2:
|
|
topt[1] = frompt[1];
|
|
case 1:
|
|
topt[0] = frompt[0];
|
|
break;
|
|
default:
|
|
memcpy( topt, frompt, ncoords * sizeof( REAL ) );
|
|
}
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------------
|
|
* pt_oo_sum - compute affine combination of internal cntrl pts
|
|
*-----------------------------------------------------------------------------
|
|
*/
|
|
|
|
void
|
|
Knotspec::pt_oo_sum( REAL *x, REAL *y, REAL *z, Knot a, Knot b )
|
|
{
|
|
switch( ncoords ) {
|
|
case 4:
|
|
x[3] = a * y[3] + b * z[3];
|
|
case 3:
|
|
x[2] = a * y[2] + b * z[2];
|
|
case 2:
|
|
x[1] = a * y[1] + b * z[1];
|
|
case 1:
|
|
x[0] = a * y[0] + b * z[0];
|
|
break;
|
|
default: {
|
|
for( int i = 0; i < ncoords; i++ )
|
|
*x++ = a * *y++ + b * *z++;
|
|
}
|
|
}
|
|
}
|