]> git.xonotic.org Git - xonotic/netradiant.git/blob - libs/mathlib.h
Q3map2:
[xonotic/netradiant.git] / libs / mathlib.h
1 /*
2    Copyright (C) 1999-2006 Id Software, Inc. and contributors.
3    For a list of contributors, see the accompanying CONTRIBUTORS file.
4
5    This file is part of GtkRadiant.
6
7    GtkRadiant is free software; you can redistribute it and/or modify
8    it under the terms of the GNU General Public License as published by
9    the Free Software Foundation; either version 2 of the License, or
10    (at your option) any later version.
11
12    GtkRadiant is distributed in the hope that it will be useful,
13    but WITHOUT ANY WARRANTY; without even the implied warranty of
14    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
15    GNU General Public License for more details.
16
17    You should have received a copy of the GNU General Public License
18    along with GtkRadiant; if not, write to the Free Software
19    Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
20  */
21
22 #ifndef __MATHLIB__
23 #define __MATHLIB__
24
25 // mathlib.h
26 #include <math.h>
27 #include <float.h>
28
29 #ifdef __cplusplus
30
31 // start declarations of functions defined in C library.
32 extern "C"
33 {
34
35 #endif
36
37 #include "bytebool.h"
38
39 typedef float vec_t;
40 typedef vec_t vec3_t[3];
41 typedef vec_t vec5_t[5];
42 typedef vec_t vec4_t[4];
43
44 // Smallest positive value for vec_t such that 1.0 + VEC_SMALLEST_EPSILON_AROUND_ONE != 1.0.
45 // In the case of 32 bit floats (which is almost certainly the case), it's 0.00000011921.
46 // Don't forget that your epsilons should depend on the possible range of values,
47 // because for example adding VEC_SMALLEST_EPSILON_AROUND_ONE to 1024.0 will have no effect.
48 #define VEC_SMALLEST_EPSILON_AROUND_ONE FLT_EPSILON
49
50 #define SIDE_FRONT      0
51 #define SIDE_ON         2
52 #define SIDE_BACK       1
53 #define SIDE_CROSS      -2
54
55 // plane types are used to speed some tests
56 // 0-2 are axial planes
57 #define PLANE_X         0
58 #define PLANE_Y         1
59 #define PLANE_Z         2
60 #define PLANE_NON_AXIAL 3
61
62 #define Q_PI    3.14159265358979323846f
63
64 extern const vec3_t vec3_origin;
65
66 extern const vec3_t g_vec3_axis_x;
67 extern const vec3_t g_vec3_axis_y;
68 extern const vec3_t g_vec3_axis_z;
69
70 #define EQUAL_EPSILON   0.001
71
72 #define DotProduct( x,y ) ( ( x )[0] * ( y )[0] + ( x )[1] * ( y )[1] + ( x )[2] * ( y )[2] )
73 #define VectorSubtract( a,b,c ) ( ( c )[0] = ( a )[0] - ( b )[0],( c )[1] = ( a )[1] - ( b )[1],( c )[2] = ( a )[2] - ( b )[2] )
74 #define VectorAdd( a,b,c ) ( ( c )[0] = ( a )[0] + ( b )[0],( c )[1] = ( a )[1] + ( b )[1],( c )[2] = ( a )[2] + ( b )[2] )
75 #define VectorIncrement( a,b ) ( ( b )[0] += ( a )[0],( b )[1] += ( a )[1],( b )[2] += ( a )[2] )
76 #define VectorCopy( a,b ) ( ( b )[0] = ( a )[0],( b )[1] = ( a )[1],( b )[2] = ( a )[2] )
77 #define VectorSet( v, a, b, c ) ( ( v )[0] = ( a ),( v )[1] = ( b ),( v )[2] = ( c ) )
78 #define VectorScale( a,b,c ) ( ( c )[0] = ( b ) * ( a )[0],( c )[1] = ( b ) * ( a )[1],( c )[2] = ( b ) * ( a )[2] )
79 #define VectorMid( a,b,c ) ( ( c )[0] = ( ( a )[0] + ( b )[0] ) * 0.5f,( c )[1] = ( ( a )[1] + ( b )[1] ) * 0.5f,( c )[2] = ( ( a )[2] + ( b )[2] ) * 0.5f )
80 #define VectorNegate( a,b ) ( ( b )[0] = -( a )[0],( b )[1] = -( a )[1],( b )[2] = -( a )[2] )
81 #define CrossProduct( a,b,c ) ( ( c )[0] = ( a )[1] * ( b )[2] - ( a )[2] * ( b )[1],( c )[1] = ( a )[2] * ( b )[0] - ( a )[0] * ( b )[2],( c )[2] = ( a )[0] * ( b )[1] - ( a )[1] * ( b )[0] )
82 #define VectorClear( x ) ( ( x )[0] = ( x )[1] = ( x )[2] = 0 )
83
84 #define FLOAT_SNAP( f,snap ) ( (float)( floor( ( f ) / ( snap ) + 0.5 ) * ( snap ) ) )
85 #define FLOAT_TO_INTEGER( f ) ( (float)( floor( ( f ) + 0.5 ) ) )
86
87 #define RGBTOGRAY( x ) ( (float)( ( x )[0] ) * 0.2989f + (float)( ( x )[1] ) * 0.5870f + (float)( ( x )[2] ) * 0.1140f )
88
89 #define Q_rint( in ) ( (vec_t)floor( in + 0.5 ) )
90
91 qboolean VectorCompare( const vec3_t v1, const vec3_t v2 );
92
93 qboolean VectorIsOnAxis( vec3_t v );
94 qboolean VectorIsOnAxialPlane( vec3_t v );
95
96 vec_t VectorLength( const vec3_t v );
97
98 void VectorMA( const vec3_t va, vec_t scale, const vec3_t vb, vec3_t vc );
99
100 void _CrossProduct( vec3_t v1, vec3_t v2, vec3_t cross );
101 // I need this define in order to test some of the regression tests from time to time.
102 // This define affect the precision of VectorNormalize() function only.
103 #define MATHLIB_VECTOR_NORMALIZE_PRECISION_FIX 1
104 vec_t VectorAccurateNormalize( const vec3_t in, vec3_t out );
105 vec_t VectorFastNormalize_( const vec3_t in, vec3_t out );
106 #if MATHLIB_VECTOR_NORMALIZE_PRECISION_FIX
107 #define VectorNormalize VectorAccurateNormalize
108 #else
109 #define VectorNormalize VectorFastNormalize_
110 #endif
111
112 #if 0 //use fastnormalize in a few -light spots
113         #define VectorFastNormalize VectorFastNormalize_
114 #else
115         #define VectorFastNormalize VectorNormalize
116 #endif
117
118 vec_t ColorNormalize( const vec3_t in, vec3_t out );
119 void VectorInverse( vec3_t v );
120 void VectorPolar( vec3_t v, float radius, float theta, float phi );
121
122 // default snapping, to 1
123 void VectorSnap( vec3_t v );
124
125 // integer snapping
126 void VectorISnap( vec3_t point, int snap );
127
128 // Gef:   added snap to float for sub-integer grid sizes
129 // TTimo: we still use the int version of VectorSnap when possible
130 //        to avoid potential rounding issues
131 // TTimo: renaming to VectorFSnap for C implementation
132 void VectorFSnap( vec3_t point, float snap );
133
134 // NOTE: added these from Ritual's Q3Radiant
135 void ClearBounds( vec3_t mins, vec3_t maxs );
136 void AddPointToBounds( vec3_t v, vec3_t mins, vec3_t maxs );
137
138
139 #define PITCH               0       // up / down
140 #define YAW                 1       // left / right
141 #define ROLL                2       // fall over
142
143 void AngleVectors( vec3_t angles, vec3_t forward, vec3_t right, vec3_t up );
144 void VectorToAngles( vec3_t vec, vec3_t angles );
145
146 #define ZERO_EPSILON 1.0E-6
147 #define RAD2DEGMULT 57.29577951308232f
148 #define DEG2RADMULT 0.01745329251994329f
149 #define RAD2DEG( a ) ( ( a ) * RAD2DEGMULT )
150 #define DEG2RAD( a ) ( ( a ) * DEG2RADMULT )
151
152 void VectorRotate( vec3_t vIn, vec3_t vRotation, vec3_t out );
153 void VectorRotateOrigin( vec3_t vIn, vec3_t vRotation, vec3_t vOrigin, vec3_t out );
154
155 // some function merged from tools mathlib code
156
157 qboolean PlaneFromPoints( vec4_t plane, const vec3_t a, const vec3_t b, const vec3_t c );
158 void NormalToLatLong( const vec3_t normal, byte bytes[2] );
159 int PlaneTypeForNormal( vec3_t normal );
160 void RotatePointAroundVector( vec3_t dst, const vec3_t dir, const vec3_t point, float degrees );
161
162
163 /*!
164    \todo
165    FIXME test calls such as intersect tests should be named test_
166  */
167
168 typedef vec_t m3x3_t[9];
169 /*!NOTE
170    m4x4 looks like this..
171
172                 x  y  z
173    x axis        ( 0  1  2)
174    y axis        ( 4  5  6)
175    z axis        ( 8  9 10)
176    translation   (12 13 14)
177    scale         ( 0  5 10)
178  */
179 typedef vec_t m4x4_t[16];
180
181 #define M4X4_INDEX( m,row,col ) ( m[( col << 2 ) + row] )
182
183 typedef enum { eXYZ, eYZX, eZXY, eXZY, eYXZ, eZYX } eulerOrder_t;
184
185 #define CLIP_PASS 0x00 // 000000
186 #define CLIP_LT_X 0x01 // 000001
187 #define CLIP_GT_X 0x02 // 000010
188 #define CLIP_LT_Y 0x04 // 000100
189 #define CLIP_GT_Y 0x08 // 001000
190 #define CLIP_LT_Z 0x10 // 010000
191 #define CLIP_GT_Z 0x20 // 100000
192 #define CLIP_FAIL 0x3F // 111111
193 typedef unsigned char clipmask_t;
194
195 extern const m4x4_t g_m4x4_identity;
196
197 #define M4X4_COPY( dst,src ) ( \
198                 ( dst )[0] = ( src )[0], \
199                 ( dst )[1] = ( src )[1], \
200                 ( dst )[2] = ( src )[2], \
201                 ( dst )[3] = ( src )[3], \
202                 ( dst )[4] = ( src )[4], \
203                 ( dst )[5] = ( src )[5], \
204                 ( dst )[6] = ( src )[6], \
205                 ( dst )[7] = ( src )[7], \
206                 ( dst )[8] = ( src )[8], \
207                 ( dst )[9] = ( src )[9], \
208                 ( dst )[10] = ( src )[10], \
209                 ( dst )[11] = ( src )[11], \
210                 ( dst )[12] = ( src )[12], \
211                 ( dst )[13] = ( src )[13], \
212                 ( dst )[14] = ( src )[14], \
213                 ( dst )[15] = ( src )[15] )
214
215 typedef enum
216 {
217         eRightHanded = 0,
218         eLeftHanded = 1,
219 }
220 m4x4Handedness_t;
221
222 m4x4Handedness_t m4x4_handedness( const m4x4_t matrix );
223
224 /*! assign other m4x4 to this m4x4 */
225 void m4x4_assign( m4x4_t matrix, const m4x4_t other );
226
227 // constructors
228 /*! create m4x4 as identity matrix */
229 void m4x4_identity( m4x4_t matrix );
230 /*! create m4x4 as a translation matrix, for a translation vec3 */
231 void m4x4_translation_for_vec3( m4x4_t matrix, const vec3_t translation );
232 /*! create m4x4 as a rotation matrix, for an euler angles (degrees) vec3 */
233 void m4x4_rotation_for_vec3( m4x4_t matrix, const vec3_t euler, eulerOrder_t order );
234 /*! create m4x4 as a scaling matrix, for a scale vec3 */
235 void m4x4_scale_for_vec3( m4x4_t matrix, const vec3_t scale );
236 /*! create m4x4 as a rotation matrix, for a quaternion vec4 */
237 void m4x4_rotation_for_quat( m4x4_t matrix, const vec4_t rotation );
238 /*! create m4x4 as a rotation matrix, for an axis vec3 and an angle (radians) */
239 void m4x4_rotation_for_axisangle( m4x4_t matrix, const vec3_t axis, double angle );
240 /*! generate a perspective matrix by specifying the view frustum */
241 void m4x4_frustum( m4x4_t matrix, vec_t left, vec_t right, vec_t bottom, vec_t top, vec_t nearval, vec_t farval );
242
243 // a valid m4x4 to access is always first argument
244 /*! extract translation vec3 from matrix */
245 void m4x4_get_translation_vec3( const m4x4_t matrix, vec3_t translation );
246 /*! extract euler rotation angles from a rotation-only matrix */
247 void m4x4_get_rotation_vec3( const m4x4_t matrix, vec3_t euler, eulerOrder_t order );
248 /*! extract scale vec3 from matrix */
249 void m4x4_get_scale_vec3( const m4x4_t matrix, vec3_t scale );
250 /*! extract translation/euler/scale from an orthogonal matrix. NOTE: requires right-handed axis-base */
251 void m4x4_get_transform_vec3( const m4x4_t matrix, vec3_t translation, vec3_t euler, eulerOrder_t order, vec3_t scale );
252
253 // a valid m4x4 to be modified is always first argument
254 /*! translate m4x4 by a translation vec3 */
255 void m4x4_translate_by_vec3( m4x4_t matrix, const vec3_t translation );
256 /*! rotate m4x4 by a euler (degrees) vec3 */
257 void m4x4_rotate_by_vec3( m4x4_t matrix, const vec3_t euler, eulerOrder_t order );
258 /*! scale m4x4 by a scaling vec3 */
259 void m4x4_scale_by_vec3( m4x4_t matrix, const vec3_t scale );
260 /*! rotate m4x4 by a quaternion vec4 */
261 void m4x4_rotate_by_quat( m4x4_t matrix, const vec4_t rotation );
262 /*! rotate m4x4 by an axis vec3 and an angle (radians) */
263 void m4x4_rotate_by_axisangle( m4x4_t matrix, const vec3_t axis, double angle );
264 /*! transform m4x4 by translation/eulerZYX/scaling vec3 (transform = scale * eulerZ * eulerY * eulerX * translation) */
265 void m4x4_transform_by_vec3( m4x4_t matrix, const vec3_t translation, const vec3_t euler, eulerOrder_t order, const vec3_t scale );
266 /*! rotate m4x4 around a pivot point by eulerZYX vec3 */
267 void m4x4_pivoted_rotate_by_vec3( m4x4_t matrix, const vec3_t euler, eulerOrder_t order, const vec3_t pivotpoint );
268 /*! scale m4x4 around a pivot point by scaling vec3 */
269 void m4x4_pivoted_scale_by_vec3( m4x4_t matrix, const vec3_t scale, const vec3_t pivotpoint );
270 /*! transform m4x4 around a pivot point by translation/eulerZYX/scaling vec3 */
271 void m4x4_pivoted_transform_by_vec3( m4x4_t matrix, const vec3_t translation, const vec3_t euler, eulerOrder_t order, const vec3_t scale, const vec3_t pivotpoint );
272 /*! transform m4x4 around a pivot point by translation/rotation/scaling vec3 */
273 void m4x4_pivoted_transform_by_rotation( m4x4_t matrix, const vec3_t translation, const m4x4_t rotation, const vec3_t scale, const vec3_t pivotpoint );
274 /*! rotate m4x4 around a pivot point by quaternion vec4 */
275 void m4x4_pivoted_rotate_by_quat( m4x4_t matrix, const vec4_t quat, const vec3_t pivotpoint );
276 /*! rotate m4x4 around a pivot point by axis vec3 and angle (radians) */
277 void m4x4_pivoted_rotate_by_axisangle( m4x4_t matrix, const vec3_t axis, double angle, const vec3_t pivotpoint );
278
279 /*! postmultiply m4x4 by another m4x4 */
280 void m4x4_multiply_by_m4x4( m4x4_t matrix, const m4x4_t matrix_src );
281 /*! premultiply m4x4 by another m4x4 */
282 void m4x4_premultiply_by_m4x4( m4x4_t matrix, const m4x4_t matrix_src );
283 /*! postmultiply orthogonal m4x4 by another orthogonal m4x4 */
284 void m4x4_orthogonal_multiply_by_m4x4( m4x4_t matrix, const m4x4_t matrix_src );
285 /*! premultiply orthogonal m4x4 by another orthogonal m4x4 */
286 void m4x4_orthogonal_premultiply_by_m4x4( m4x4_t matrix, const m4x4_t matrix_src );
287
288 /*! multiply a point (x,y,z,1) by matrix */
289 void m4x4_transform_point( const m4x4_t matrix, vec3_t point );
290 /*! multiply a normal (x,y,z,0) by matrix */
291 void m4x4_transform_normal( const m4x4_t matrix, vec3_t normal );
292 /*! multiply a vec4 (x,y,z,w) by matrix */
293 void m4x4_transform_vec4( const m4x4_t matrix, vec4_t vector );
294
295 /*! multiply a point (x,y,z,1) by matrix */
296 void m4x4_transform_point( const m4x4_t matrix, vec3_t point );
297 /*! multiply a normal (x,y,z,0) by matrix */
298 void m4x4_transform_normal( const m4x4_t matrix, vec3_t normal );
299
300 /*! transpose a m4x4 */
301 void m4x4_transpose( m4x4_t matrix );
302 /*! invert an orthogonal 4x3 subset of a 4x4 matrix */
303 int m4x4_orthogonal_invert( m4x4_t matrix );
304 /*! m4_det */
305 float m4_det( m4x4_t mr );
306 /*! invert any m4x4 using Kramer's rule.. return 1 if matrix is singular, else return 0 */
307 int m4x4_invert( m4x4_t matrix );
308
309 /*! clip a point (x,y,z,1) by canonical matrix */
310 clipmask_t m4x4_clip_point( const m4x4_t matrix, const vec3_t point, vec4_t clipped );
311 /*! device-space polygon for clipped triangle */
312 unsigned int m4x4_clip_triangle( const m4x4_t matrix, const vec3_t p0, const vec3_t p1, const vec3_t p2, vec4_t clipped[9] );
313 /*! device-space line for clipped line  */
314 unsigned int m4x4_clip_line( const m4x4_t matrix, const vec3_t p0, const vec3_t p1, vec4_t clipped[2] );
315
316
317 //! quaternion identity
318 void quat_identity( vec4_t quat );
319 //! quaternion from two unit vectors
320 void quat_for_unit_vectors( vec4_t quat, const vec3_t from, const vec3_t to );
321 //! quaternion from axis and angle (radians)
322 void quat_for_axisangle( vec4_t quat, const vec3_t axis, double angle );
323 //! concatenates two rotations.. equivalent to m4x4_multiply_by_m4x4 .. postmultiply.. the right-hand side is the first rotation performed
324 void quat_multiply_by_quat( vec4_t quat, const vec4_t other );
325 //! negate a quaternion
326 void quat_conjugate( vec4_t quat );
327 //! normalise a quaternion
328 void quat_normalise( vec4_t quat );
329
330
331
332 /*!
333    \todo object/ray intersection functions should maybe return a point rather than a distance?
334  */
335
336 /*!
337    aabb_t -  "axis-aligned" bounding box...
338    origin: centre of bounding box...
339    extents: +/- extents of box from origin...
340  */
341 typedef struct aabb_s
342 {
343         vec3_t origin;
344         vec3_t extents;
345 } aabb_t;
346
347 extern const aabb_t g_aabb_null;
348
349 /*!
350    bbox_t - oriented bounding box...
351    aabb: axis-aligned bounding box...
352    axes: orientation axes...
353  */
354 typedef struct bbox_s
355 {
356         aabb_t aabb;
357         vec3_t axes[3];
358         vec_t radius;
359 } bbox_t;
360
361 /*!
362    ray_t - origin point and direction unit-vector
363  */
364 typedef struct ray_s
365 {
366         vec3_t origin;
367         vec3_t direction;
368 } ray_t;
369
370 /*!
371    line_t - centre point and displacement of end point from centre
372  */
373 typedef struct line_s
374 {
375         vec3_t origin;
376         vec3_t extents;
377 } line_t;
378
379
380 /*! Generate line from start/end points. */
381 void line_construct_for_vec3( line_t* line, const vec3_t start, const vec3_t end );
382 /*! Return 2 if line is behind plane, else return 1 if line intersects plane, else return 0. */
383 int line_test_plane( const line_t* line, const vec4_t plane );
384
385 /*! Generate AABB from min/max. */
386 void aabb_construct_for_vec3( aabb_t* aabb, const vec3_t min, const vec3_t max );
387 /*! Initialise AABB to negative size. */
388 void aabb_clear( aabb_t* aabb );
389
390 /*! Extend AABB to include point. */
391 void aabb_extend_by_point( aabb_t* aabb, const vec3_t point );
392 /*! Extend AABB to include aabb_src. */
393 void aabb_extend_by_aabb( aabb_t* aabb, const aabb_t* aabb_src );
394 /*! Extend AABB by +/- extension vector. */
395 void aabb_extend_by_vec3( aabb_t* aabb, vec3_t extension );
396
397 /*! Return 2 if point is inside, else 1 if point is on surface, else 0. */
398 int aabb_test_point( const aabb_t* aabb, const vec3_t point );
399 /*! Return 2 if aabb_src intersects, else 1 if aabb_src touches exactly, else 0. */
400 int aabb_test_aabb( const aabb_t* aabb, const aabb_t* aabb_src );
401 /*! Return 2 if aabb is behind plane, else 1 if aabb intersects plane, else 0. */
402 int aabb_test_plane( const aabb_t* aabb, const float* plane );
403 /*! Return 1 if aabb intersects ray, else 0... dist = closest intersection. */
404 int aabb_intersect_ray( const aabb_t* aabb, const ray_t* ray, vec3_t intersection );
405 /*! Return 1 if aabb intersects ray, else 0. Faster, but does not provide point of intersection */
406 int aabb_test_ray( const aabb_t* aabb, const ray_t* ray );
407
408 /*! Return 2 if oriented aabb is behind plane, else 1 if aabb intersects plane, else 0. */
409 int aabb_oriented_intersect_plane( const aabb_t* aabb, const m4x4_t transform, const vec_t* plane );
410
411 /*! Calculate the corners of the aabb. */
412 void aabb_corners( const aabb_t * aabb, vec3_t corners[8] );
413
414 /*! (deprecated) Generate AABB from oriented bounding box. */
415 void aabb_for_bbox( aabb_t* aabb, const bbox_t* bbox );
416 /*! (deprecated) Generate AABB from 2-dimensions of min/max, specified by axis. */
417 void aabb_for_area( aabb_t* aabb, vec3_t area_tl, vec3_t area_br, int axis );
418 /*! Generate AABB to contain src*  transform. NOTE: transform must be orthogonal */
419 void aabb_for_transformed_aabb( aabb_t* dst, const aabb_t* src, const m4x4_t transform );
420
421 /*! Update bounding-sphere radius. */
422 void bbox_update_radius( bbox_t* bbox );
423 /*! Generate oriented bounding box from AABB and transformation matrix. */
424 /*!\todo Remove need to specify eulerZYX/scale. */
425 void bbox_for_oriented_aabb( bbox_t* bbox, const aabb_t* aabb,
426                                                          const m4x4_t matrix, const vec3_t eulerZYX, const vec3_t scale );
427 /*! Return 2 if bbox is behind plane, else return 1 if bbox intersects plane, else return 0. */
428 int bbox_intersect_plane( const bbox_t* bbox, const vec_t* plane );
429
430
431 /*! Generate a ray from an origin point and a direction unit-vector */
432 void ray_construct_for_vec3( ray_t* ray, const vec3_t origin, const vec3_t direction );
433
434 /*! Transform a ray */
435 void ray_transform( ray_t* ray, const m4x4_t matrix );
436
437 /*! distance from ray origin in ray direction to point. FLT_MAX if no intersection. */
438 vec_t ray_intersect_point( const ray_t* ray, const vec3_t point, vec_t epsilon, vec_t divergence );
439 /*! distance from ray origin in ray direction to triangle. FLT_MAX if no intersection. */
440 vec_t ray_intersect_triangle( const ray_t* ray, qboolean bCullBack, const vec3_t vert0, const vec3_t vert1, const vec3_t vert2 );
441 /*! distance from ray origin in ray direction to plane. */
442 vec_t ray_intersect_plane( const ray_t* ray, const vec3_t normal, vec_t dist );
443
444
445 int plane_intersect_planes( const vec4_t plane1, const vec4_t plane2, const vec4_t plane3, vec3_t intersection );
446
447
448
449 ////////////////////////////////////////////////////////////////////////////////
450 // Below is double-precision math stuff.  This was initially needed by the new
451 // "base winding" code in q3map2 brush processing in order to fix the famous
452 // "disappearing triangles" issue.  These definitions can be used wherever extra
453 // precision is needed.
454 ////////////////////////////////////////////////////////////////////////////////
455
456 typedef double vec_accu_t;
457 typedef vec_accu_t vec3_accu_t[3];
458
459 // Smallest positive value for vec_accu_t such that 1.0 + VEC_ACCU_SMALLEST_EPSILON_AROUND_ONE != 1.0.
460 // In the case of 64 bit doubles (which is almost certainly the case), it's 0.00000000000000022204.
461 // Don't forget that your epsilons should depend on the possible range of values,
462 // because for example adding VEC_ACCU_SMALLEST_EPSILON_AROUND_ONE to 1024.0 will have no effect.
463 #define VEC_ACCU_SMALLEST_EPSILON_AROUND_ONE DBL_EPSILON
464
465 vec_accu_t VectorLengthAccu( const vec3_accu_t v );
466
467 // I have a feeling it may be safer to break these #define functions out into actual functions
468 // in order to avoid accidental loss of precision.  For example, say you call
469 // VectorScaleAccu(vec3_t, vec_t, vec3_accu_t).  The scale would take place in 32 bit land
470 // and the result would be cast to 64 bit, which would cause total loss of precision when
471 // scaling by a large factor.
472 //#define DotProductAccu(x, y) ((x)[0] * (y)[0] + (x)[1] * (y)[1] + (x)[2] * (y)[2])
473 //#define VectorSubtractAccu(a, b, c) ((c)[0] = (a)[0] - (b)[0], (c)[1] = (a)[1] - (b)[1], (c)[2] = (a)[2] - (b)[2])
474 //#define VectorAddAccu(a, b, c) ((c)[0] = (a)[0] + (b)[0], (c)[1] = (a)[1] + (b)[1], (c)[2] = (a)[2] + (b)[2])
475 //#define VectorCopyAccu(a, b) ((b)[0] = (a)[0], (b)[1] = (a)[1], (b)[2] = (a)[2])
476 //#define VectorScaleAccu(a, b, c) ((c)[0] = (b) * (a)[0], (c)[1] = (b) * (a)[1], (c)[2] = (b) * (a)[2])
477 //#define CrossProductAccu(a, b, c) ((c)[0] = (a)[1] * (b)[2] - (a)[2] * (b)[1], (c)[1] = (a)[2] * (b)[0] - (a)[0] * (b)[2], (c)[2] = (a)[0] * (b)[1] - (a)[1] * (b)[0])
478 //#define Q_rintAccu(in) ((vec_accu_t) floor(in + 0.5))
479
480 vec_accu_t DotProductAccu( const vec3_accu_t a, const vec3_accu_t b );
481 void VectorSubtractAccu( const vec3_accu_t a, const vec3_accu_t b, vec3_accu_t out );
482 void VectorAddAccu( const vec3_accu_t a, const vec3_accu_t b, vec3_accu_t out );
483 void VectorCopyAccu( const vec3_accu_t in, vec3_accu_t out );
484 void VectorScaleAccu( const vec3_accu_t in, vec_accu_t scaleFactor, vec3_accu_t out );
485 void CrossProductAccu( const vec3_accu_t a, const vec3_accu_t b, vec3_accu_t out );
486 vec_accu_t Q_rintAccu( vec_accu_t val );
487
488 void VectorCopyAccuToRegular( const vec3_accu_t in, vec3_t out );
489 void VectorCopyRegularToAccu( const vec3_t in, vec3_accu_t out );
490 vec_accu_t VectorNormalizeAccu( const vec3_accu_t in, vec3_accu_t out );
491
492 #ifdef __cplusplus
493 }
494 #endif
495
496 #endif /* __MATHLIB__ */