1f220fa62Smrg/*
2f220fa62Smrg** License Applicability. Except to the extent portions of this file are
3f220fa62Smrg** made subject to an alternative license as permitted in the SGI Free
4f220fa62Smrg** Software License B, Version 1.1 (the "License"), the contents of this
5f220fa62Smrg** file are subject only to the provisions of the License. You may not use
6f220fa62Smrg** this file except in compliance with the License. You may obtain a copy
7f220fa62Smrg** of the License at Silicon Graphics, Inc., attn: Legal Services, 1600
8f220fa62Smrg** Amphitheatre Parkway, Mountain View, CA 94043-1351, or at:
9f220fa62Smrg**
10f220fa62Smrg** http://oss.sgi.com/projects/FreeB
11f220fa62Smrg**
12f220fa62Smrg** Note that, as provided in the License, the Software is distributed on an
13f220fa62Smrg** "AS IS" basis, with ALL EXPRESS AND IMPLIED WARRANTIES AND CONDITIONS
14f220fa62Smrg** DISCLAIMED, INCLUDING, WITHOUT LIMITATION, ANY IMPLIED WARRANTIES AND
15f220fa62Smrg** CONDITIONS OF MERCHANTABILITY, SATISFACTORY QUALITY, FITNESS FOR A
16f220fa62Smrg** PARTICULAR PURPOSE, AND NON-INFRINGEMENT.
17f220fa62Smrg**
18f220fa62Smrg** Original Code. The Original Code is: OpenGL Sample Implementation,
19f220fa62Smrg** Version 1.2.1, released January 26, 2000, developed by Silicon Graphics,
20f220fa62Smrg** Inc. The Original Code is Copyright (c) 1991-2000 Silicon Graphics, Inc.
21f220fa62Smrg** Copyright in any portions created by third parties is as indicated
22f220fa62Smrg** elsewhere herein. All Rights Reserved.
23f220fa62Smrg**
24f220fa62Smrg** Additional Notice Provisions: The application programming interfaces
25f220fa62Smrg** established by SGI in conjunction with the Original Code are The
26f220fa62Smrg** OpenGL(R) Graphics System: A Specification (Version 1.2.1), released
27f220fa62Smrg** April 1, 1999; The OpenGL(R) Graphics System Utility Library (Version
28f220fa62Smrg** 1.3), released November 4, 1998; and OpenGL(R) Graphics with the X
29f220fa62Smrg** Window System(R) (Version 1.3), released October 19, 1998. This software
30f220fa62Smrg** was created using the OpenGL(R) version 1.2.1 Sample Implementation
31f220fa62Smrg** published by SGI, but has not been independently verified as being
32f220fa62Smrg** compliant with the OpenGL(R) version 1.2.1 Specification.
33f220fa62Smrg*/
34f220fa62Smrg
35f220fa62Smrg/*
36f220fa62Smrg * curve.c++
37f220fa62Smrg *
38f220fa62Smrg */
39f220fa62Smrg
40f220fa62Smrg#include "glimports.h"
41f220fa62Smrg#include "myassert.h"
42f220fa62Smrg#include "mystdio.h"
43f220fa62Smrg#include "mymath.h"
44f220fa62Smrg#include "curve.h"
45f220fa62Smrg#include "mapdesc.h"
46f220fa62Smrg#include "types.h"
47f220fa62Smrg#include "quilt.h"
48f220fa62Smrg#include "nurbsconsts.h"
49f220fa62Smrg
50f220fa62Smrg/*--------------------------------------------------------------------------
51f220fa62Smrg * Curve::Curve - copy curve from quilt and transform control points
52f220fa62Smrg *--------------------------------------------------------------------------
53f220fa62Smrg */
54f220fa62Smrg
55f220fa62SmrgCurve::Curve( Quilt_ptr geo, REAL pta, REAL ptb, Curve *c )
56f220fa62Smrg{
57f220fa62Smrg    mapdesc = geo->mapdesc;
58f220fa62Smrg    next = c;
59f220fa62Smrg    needsSampling = mapdesc->isRangeSampling() ? 1 : 0;
60f220fa62Smrg    cullval = mapdesc->isCulling() ? CULL_ACCEPT : CULL_TRIVIAL_ACCEPT;
61f220fa62Smrg    order = geo->qspec[0].order;
62f220fa62Smrg    stride = MAXCOORDS;
63f220fa62Smrg    for( int i = 0; i < MAXORDER * MAXCOORDS; i++ ) {
64f220fa62Smrg        cpts[i] = 0;
65f220fa62Smrg        spts[i] = 0;
66f220fa62Smrg    }
67f220fa62Smrg    stepsize = 0;
68f220fa62Smrg    minstepsize = 0;
69f220fa62Smrg
70f220fa62Smrg    REAL *ps  = geo->cpts;
71f220fa62Smrg    Quiltspec_ptr qs = geo->qspec;
72f220fa62Smrg    ps += qs->offset;
73f220fa62Smrg    ps += qs->index * qs->order * qs->stride;
74f220fa62Smrg
75f220fa62Smrg    if( needsSampling )
76f220fa62Smrg	mapdesc->xformSampling( ps, qs->order, qs->stride, spts, stride );
77f220fa62Smrg
78f220fa62Smrg    if( cullval == CULL_ACCEPT )
79f220fa62Smrg	mapdesc->xformCulling(  ps, qs->order, qs->stride, cpts, stride );
80f220fa62Smrg
81f220fa62Smrg    /* set untrimmed curve range */
82f220fa62Smrg    range[0] = qs->breakpoints[qs->index];
83f220fa62Smrg    range[1] = qs->breakpoints[qs->index+1];
84f220fa62Smrg    range[2] = range[1] - range[0];
85f220fa62Smrg
86f220fa62Smrg    if( range[0] != pta ) {
87f220fa62Smrg	Curve lower( *this, pta, 0 );
88f220fa62Smrg	lower.next = next;
89f220fa62Smrg	*this = lower;
90f220fa62Smrg    }
91f220fa62Smrg    if( range[1] != ptb ) {
92f220fa62Smrg	Curve lower( *this, ptb, 0 );
93f220fa62Smrg    }
94f220fa62Smrg}
95f220fa62Smrg
96f220fa62Smrg/*--------------------------------------------------------------------------
97f220fa62Smrg * Curve::Curve - subdivide a curve along an isoparametric line
98f220fa62Smrg *--------------------------------------------------------------------------
99f220fa62Smrg */
100f220fa62Smrg
101f220fa62SmrgCurve::Curve( Curve& upper, REAL value, Curve *c )
102f220fa62Smrg{
103f220fa62Smrg    Curve &lower = *this;
104f220fa62Smrg
105f220fa62Smrg    lower.next = c;
106f220fa62Smrg    lower.mapdesc = upper.mapdesc;
107f220fa62Smrg    lower.needsSampling = upper.needsSampling;
108f220fa62Smrg    lower.order = upper.order;
109f220fa62Smrg    lower.stride = upper.stride;
110f220fa62Smrg    lower.cullval = upper.cullval;
111f220fa62Smrg
112f220fa62Smrg    REAL d = (value - upper.range[0]) / upper.range[2];
113f220fa62Smrg
114f220fa62Smrg    if( needsSampling )
115f220fa62Smrg        mapdesc->subdivide( upper.spts, lower.spts, d, upper.stride, upper.order );
116f220fa62Smrg
117f220fa62Smrg    if( cullval == CULL_ACCEPT )
118f220fa62Smrg        mapdesc->subdivide( upper.cpts, lower.cpts, d, upper.stride, upper.order );
119f220fa62Smrg
120f220fa62Smrg    lower.range[0] = upper.range[0];
121f220fa62Smrg    lower.range[1] = value;
122f220fa62Smrg    lower.range[2] = value - upper.range[0];
123f220fa62Smrg    upper.range[0] = value;
124f220fa62Smrg    upper.range[2] = upper.range[1] - value;
125f220fa62Smrg}
126f220fa62Smrg
127f220fa62Smrg
128f220fa62Smrg/*--------------------------------------------------------------------------
129f220fa62Smrg * Curve::clamp - clamp the sampling rate to a given maximum
130f220fa62Smrg *--------------------------------------------------------------------------
131f220fa62Smrg */
132f220fa62Smrg
133f220fa62Smrgvoid
134f220fa62SmrgCurve::clamp( void )
135f220fa62Smrg{
136f220fa62Smrg    if( stepsize < minstepsize )
137f220fa62Smrg        stepsize = mapdesc->clampfactor * minstepsize;
138f220fa62Smrg}
139f220fa62Smrg
140f220fa62Smrgvoid
141f220fa62SmrgCurve::setstepsize( REAL max )
142f220fa62Smrg{
143f220fa62Smrg    stepsize = ( max >= 1.0 ) ? (range[2] / max) : range[2];
144f220fa62Smrg    minstepsize = stepsize;
145f220fa62Smrg}
146f220fa62Smrg
147f220fa62Smrgvoid
148f220fa62SmrgCurve::getstepsize( void )
149f220fa62Smrg{
150f220fa62Smrg    minstepsize= 0;
151f220fa62Smrg
152f220fa62Smrg    if( mapdesc->isConstantSampling() ) {
153f220fa62Smrg	// fixed number of samples per patch in each direction
154f220fa62Smrg	// maxrate is number of s samples per patch
155f220fa62Smrg        setstepsize( mapdesc->maxrate );
156f220fa62Smrg    } else if( mapdesc->isDomainSampling() ) {
157f220fa62Smrg	// maxrate is number of s samples per unit s length of domain
158f220fa62Smrg        setstepsize( mapdesc->maxrate * range[2] );
159f220fa62Smrg    } else {
160f220fa62Smrg	// upper bound on path length between sample points
161f220fa62Smrg
162f220fa62Smrg	assert( order <= MAXORDER );
163f220fa62Smrg
164f220fa62Smrg	/* points have been transformed, therefore they are homogeneous */
165f220fa62Smrg        REAL tmp[MAXORDER][MAXCOORDS];
166f220fa62Smrg	const int tstride = sizeof(tmp[0]) / sizeof(REAL);
167f220fa62Smrg	int val = mapdesc->project( spts, stride, &tmp[0][0], tstride,  order );
168f220fa62Smrg
169f220fa62Smrg        if( val == 0 ) {
170f220fa62Smrg	    // control points cross infinity, therefore derivatives are undefined
171f220fa62Smrg            setstepsize( mapdesc->maxrate );
172f220fa62Smrg        } else {
173f220fa62Smrg            REAL t = mapdesc->getProperty( N_PIXEL_TOLERANCE );
174f220fa62Smrg	    if( mapdesc->isParametricDistanceSampling() ) {
175f220fa62Smrg		REAL d = mapdesc->calcPartialVelocity( &tmp[0][0], tstride, order, 2, range[2] );
176f220fa62Smrg		stepsize = (d > 0.0) ? sqrtf( 8.0 * t / d ) : range[2];
177f220fa62Smrg		minstepsize = ( mapdesc->maxrate > 0.0 ) ? (range[2] / mapdesc->maxrate) : 0.0;
178f220fa62Smrg	    } else if( mapdesc->isPathLengthSampling() ) {
179f220fa62Smrg		// t is upper bound on path (arc) length
180f220fa62Smrg		REAL d = mapdesc->calcPartialVelocity( &tmp[0][0], tstride, order, 1, range[2] );
181f220fa62Smrg		stepsize = ( d > 0.0 ) ? (t / d) : range[2];
182f220fa62Smrg		minstepsize = ( mapdesc->maxrate > 0.0 ) ? (range[2] / mapdesc->maxrate) : 0.0;
183f220fa62Smrg	    } else {
184f220fa62Smrg		// control points cross infinity, therefore partials are undefined
185f220fa62Smrg		setstepsize( mapdesc->maxrate );
186f220fa62Smrg	    }
187f220fa62Smrg	}
188f220fa62Smrg    }
189f220fa62Smrg}
190f220fa62Smrg
191f220fa62Smrgint
192f220fa62SmrgCurve::needsSamplingSubdivision( void )
193f220fa62Smrg{
194f220fa62Smrg    return ( stepsize < minstepsize )  ? 1 : 0;
195f220fa62Smrg}
196f220fa62Smrg
197f220fa62Smrgint
198f220fa62SmrgCurve::cullCheck( void )
199f220fa62Smrg{
200f220fa62Smrg    if( cullval == CULL_ACCEPT )
201f220fa62Smrg	cullval = mapdesc->cullCheck( cpts, order, stride );
202f220fa62Smrg    return cullval;
203f220fa62Smrg}
204f220fa62Smrg
205