~ubuntu-branches/debian/squeeze/gmsh/squeeze

« back to all changes in this revision

Viewing changes to Numeric/robustPredicates.cpp

  • Committer: Bazaar Package Importer
  • Author(s): Christophe Prud'homme, Christophe Prud'homme
  • Date: 2009-09-27 17:36:40 UTC
  • mfrom: (1.4.2 upstream)
  • mto: This revision was merged to the branch mainline in revision 10.
  • Revision ID: james.westby@ubuntu.com-20090927173640-meoeywl4f5hq5qas
Tags: 2.4.2.dfsg-1
[Christophe Prud'homme]
* New upstream release
  + solver code refactoring
  + better IDE integration.

Show diffs side-by-side

added added

removed removed

Lines of Context:
 
1
/*****************************************************************************/
 
2
/*                                                                           */
 
3
/*  Routines for Arbitrary Precision Floating-point Arithmetic               */
 
4
/*  and Fast Robust Geometric Predicates                                     */
 
5
/*  (predicates.c)                                                           */
 
6
/*                                                                           */
 
7
/*  May 18, 1996                                                             */
 
8
/*                                                                           */
 
9
/*  Placed in the public domain by                                           */
 
10
/*  Jonathan Richard Shewchuk                                                */
 
11
/*  School of Computer Science                                               */
 
12
/*  Carnegie Mellon University                                               */
 
13
/*  5000 Forbes Avenue                                                       */
 
14
/*  Pittsburgh, Pennsylvania  15213-3891                                     */
 
15
/*  jrs@cs.cmu.edu                                                           */
 
16
/*                                                                           */
 
17
/*  This file contains C implementation of algorithms for exact addition     */
 
18
/*    and multiplication of floating-point numbers, and predicates for       */
 
19
/*    robustly performing the orientation and incircle tests used in         */
 
20
/*    computational geometry.  The algorithms and underlying theory are      */
 
21
/*    described in Jonathan Richard Shewchuk.  "Adaptive Precision Floating- */
 
22
/*    Point Arithmetic and Fast Robust Geometric Predicates."  Technical     */
 
23
/*    Report CMU-CS-96-140, School of Computer Science, Carnegie Mellon      */
 
24
/*    University, Pittsburgh, Pennsylvania, May 1996.  (Submitted to         */
 
25
/*    Discrete & Computational Geometry.)                                    */
 
26
/*                                                                           */
 
27
/*  This file, the paper listed above, and other information are available   */
 
28
/*    from the Web page http://www.cs.cmu.edu/~quake/robust.html .           */
 
29
/*                                                                           */
 
30
/*****************************************************************************/
 
31
 
 
32
/*****************************************************************************/
 
33
/*                                                                           */
 
34
/*  Using this code:                                                         */
 
35
/*                                                                           */
 
36
/*  First, read the short or long version of the paper (from the Web page    */
 
37
/*    above).                                                                */
 
38
/*                                                                           */
 
39
/*  Be sure to call exactinit() once, before calling any of the arithmetic   */
 
40
/*    functions or geometric predicates.  Also be sure to turn on the        */
 
41
/*    optimizer when compiling this file.                                    */
 
42
/*                                                                           */
 
43
/*                                                                           */
 
44
/*  Several geometric predicates are defined.  Their parameters are all      */
 
45
/*    points.  Each point is an array of two or three floating-point         */
 
46
/*    numbers.  The geometric predicates, described in the papers, are       */
 
47
/*                                                                           */
 
48
/*    orient2d(pa, pb, pc)                                                   */
 
49
/*    orient2dfast(pa, pb, pc)                                               */
 
50
/*    orient3d(pa, pb, pc, pd)                                               */
 
51
/*    orient3dfast(pa, pb, pc, pd)                                           */
 
52
/*    incircle(pa, pb, pc, pd)                                               */
 
53
/*    incirclefast(pa, pb, pc, pd)                                           */
 
54
/*    insphere(pa, pb, pc, pd, pe)                                           */
 
55
/*    inspherefast(pa, pb, pc, pd, pe)                                       */
 
56
/*                                                                           */
 
57
/*  Those with suffix "fast" are approximate, non-robust versions.  Those    */
 
58
/*    without the suffix are adaptive precision, robust versions.  There     */
 
59
/*    are also versions with the suffices "exact" and "slow", which are      */
 
60
/*    non-adaptive, exact arithmetic versions, which I use only for timings  */
 
61
/*    in my arithmetic papers.                                               */
 
62
/*                                                                           */
 
63
/*                                                                           */
 
64
/*  An expansion is represented by an array of floating-point numbers,       */
 
65
/*    sorted from smallest to largest magnitude (possibly with interspersed  */
 
66
/*    zeros).  The length of each expansion is stored as a separate integer, */
 
67
/*    and each arithmetic function returns an integer which is the length    */
 
68
/*    of the expansion it created.                                           */
 
69
/*                                                                           */
 
70
/*  Several arithmetic functions are defined.  Their parameters are          */
 
71
/*                                                                           */
 
72
/*    e, f           Input expansions                                        */
 
73
/*    elen, flen     Lengths of input expansions (must be >= 1)              */
 
74
/*    h              Output expansion                                        */
 
75
/*    b              Input scalar                                            */
 
76
/*                                                                           */
 
77
/*  The arithmetic functions are                                             */
 
78
/*                                                                           */
 
79
/*    grow_expansion(elen, e, b, h)                                          */
 
80
/*    grow_expansion_zeroelim(elen, e, b, h)                                 */
 
81
/*    expansion_sum(elen, e, flen, f, h)                                     */
 
82
/*    expansion_sum_zeroelim1(elen, e, flen, f, h)                           */
 
83
/*    expansion_sum_zeroelim2(elen, e, flen, f, h)                           */
 
84
/*    fast_expansion_sum(elen, e, flen, f, h)                                */
 
85
/*    fast_expansion_sum_zeroelim(elen, e, flen, f, h)                       */
 
86
/*    linear_expansion_sum(elen, e, flen, f, h)                              */
 
87
/*    linear_expansion_sum_zeroelim(elen, e, flen, f, h)                     */
 
88
/*    scale_expansion(elen, e, b, h)                                         */
 
89
/*    scale_expansion_zeroelim(elen, e, b, h)                                */
 
90
/*    compress(elen, e, h)                                                   */
 
91
/*                                                                           */
 
92
/*  All of these are described in the long version of the paper; some are    */
 
93
/*    described in the short version.  All return an integer that is the     */
 
94
/*    length of h.  Those with suffix _zeroelim perform zero elimination,    */
 
95
/*    and are recommended over their counterparts.  The procedure            */
 
96
/*    fast_expansion_sum_zeroelim() (or linear_expansion_sum_zeroelim() on   */
 
97
/*    processors that do not use the round-to-even tiebreaking rule) is      */
 
98
/*    recommended over expansion_sum_zeroelim().  Each procedure has a       */
 
99
/*    little note next to it (in the code below) that tells you whether or   */
 
100
/*    not the output expansion may be the same array as one of the input     */
 
101
/*    expansions.                                                            */
 
102
/*                                                                           */
 
103
/*                                                                           */
 
104
/*  If you look around below, you'll also find macros for a bunch of         */
 
105
/*    simple unrolled arithmetic operations, and procedures for printing     */
 
106
/*    expansions (commented out because they don't work with all C           */
 
107
/*    compilers) and for generating random floating-point numbers whose      */
 
108
/*    significand bits are all random.  Most of the macros have undocumented */
 
109
/*    requirements that certain of their parameters should not be the same   */
 
110
/*    variable; for safety, better to make sure all the parameters are       */
 
111
/*    distinct variables.  Feel free to send email to jrs@cs.cmu.edu if you  */
 
112
/*    have questions.                                                        */
 
113
/*                                                                           */
 
114
/*****************************************************************************/
 
115
 
 
116
#include <stdio.h>
 
117
#include <stdlib.h>
 
118
#include <math.h>
 
119
#ifdef CPU86
 
120
#include <float.h>
 
121
#endif /* CPU86 */
 
122
#ifdef LINUX
 
123
#include <fpu_control.h>
 
124
#endif /* LINUX */
 
125
 
 
126
namespace robustPredicates
 
127
{
 
128
 
 
129
/* On some machines, the exact arithmetic routines might be defeated by the  */
 
130
/*   use of internal extended precision floating-point registers.  Sometimes */
 
131
/*   this problem can be fixed by defining certain values to be volatile,    */
 
132
/*   thus forcing them to be stored to memory and rounded off.  This isn't   */
 
133
/*   a great solution, though, as it slows the arithmetic down.              */
 
134
/*                                                                           */
 
135
/* To try this out, write "#define INEXACT volatile" below.  Normally,       */
 
136
/*   however, INEXACT should be defined to be nothing.  ("#define INEXACT".) */
 
137
 
 
138
#define INEXACT                          /* Nothing */
 
139
/* #define INEXACT volatile */
 
140
 
 
141
#define REAL double                      /* float or double */
 
142
#define REALPRINT doubleprint
 
143
#define REALRAND doublerand
 
144
#define NARROWRAND narrowdoublerand
 
145
#define UNIFORMRAND uniformdoublerand
 
146
 
 
147
/* Which of the following two methods of finding the absolute values is      */
 
148
/*   fastest is compiler-dependent.  A few compilers can inline and optimize */
 
149
/*   the fabs() call; but most will incur the overhead of a function call,   */
 
150
/*   which is disastrously slow.  A faster way on IEEE machines might be to  */
 
151
/*   mask the appropriate bit, but that's difficult to do in C.              */
 
152
 
 
153
#define Absolute(a)  ((a) >= 0.0 ? (a) : -(a))
 
154
/* #define Absolute(a)  fabs(a) */
 
155
 
 
156
/* Many of the operations are broken up into two pieces, a main part that    */
 
157
/*   performs an approximate operation, and a "tail" that computes the       */
 
158
/*   roundoff error of that operation.                                       */
 
159
/*                                                                           */
 
160
/* The operations Fast_Two_Sum(), Fast_Two_Diff(), Two_Sum(), Two_Diff(),    */
 
161
/*   Split(), and Two_Product() are all implemented as described in the      */
 
162
/*   reference.  Each of these macros requires certain variables to be       */
 
163
/*   defined in the calling routine.  The variables `bvirt', `c', `abig',    */
 
164
/*   `_i', `_j', `_k', `_l', `_m', and `_n' are declared `INEXACT' because   */
 
165
/*   they store the result of an operation that may incur roundoff error.    */
 
166
/*   The input parameter `x' (or the highest numbered `x_' parameter) must   */
 
167
/*   also be declared `INEXACT'.                                             */
 
168
 
 
169
#define Fast_Two_Sum_Tail(a, b, x, y) \
 
170
  bvirt = x - a; \
 
171
  y = b - bvirt
 
172
 
 
173
#define Fast_Two_Sum(a, b, x, y) \
 
174
  x = (REAL) (a + b); \
 
175
  Fast_Two_Sum_Tail(a, b, x, y)
 
176
 
 
177
#define Fast_Two_Diff_Tail(a, b, x, y) \
 
178
  bvirt = a - x; \
 
179
  y = bvirt - b
 
180
 
 
181
#define Fast_Two_Diff(a, b, x, y) \
 
182
  x = (REAL) (a - b); \
 
183
  Fast_Two_Diff_Tail(a, b, x, y)
 
184
 
 
185
#define Two_Sum_Tail(a, b, x, y) \
 
186
  bvirt = (REAL) (x - a); \
 
187
  avirt = x - bvirt; \
 
188
  bround = b - bvirt; \
 
189
  around = a - avirt; \
 
190
  y = around + bround
 
191
 
 
192
#define Two_Sum(a, b, x, y) \
 
193
  x = (REAL) (a + b); \
 
194
  Two_Sum_Tail(a, b, x, y)
 
195
 
 
196
#define Two_Diff_Tail(a, b, x, y) \
 
197
  bvirt = (REAL) (a - x); \
 
198
  avirt = x + bvirt; \
 
199
  bround = bvirt - b; \
 
200
  around = a - avirt; \
 
201
  y = around + bround
 
202
 
 
203
#define Two_Diff(a, b, x, y) \
 
204
  x = (REAL) (a - b); \
 
205
  Two_Diff_Tail(a, b, x, y)
 
206
 
 
207
#define Split(a, ahi, alo) \
 
208
  c = (REAL) (splitter * a); \
 
209
  abig = (REAL) (c - a); \
 
210
  ahi = c - abig; \
 
211
  alo = a - ahi
 
212
 
 
213
#define Two_Product_Tail(a, b, x, y) \
 
214
  Split(a, ahi, alo); \
 
215
  Split(b, bhi, blo); \
 
216
  err1 = x - (ahi * bhi); \
 
217
  err2 = err1 - (alo * bhi); \
 
218
  err3 = err2 - (ahi * blo); \
 
219
  y = (alo * blo) - err3
 
220
 
 
221
#define Two_Product(a, b, x, y) \
 
222
  x = (REAL) (a * b); \
 
223
  Two_Product_Tail(a, b, x, y)
 
224
 
 
225
/* Two_Product_Presplit() is Two_Product() where one of the inputs has       */
 
226
/*   already been split.  Avoids redundant splitting.                        */
 
227
 
 
228
#define Two_Product_Presplit(a, b, bhi, blo, x, y) \
 
229
  x = (REAL) (a * b); \
 
230
  Split(a, ahi, alo); \
 
231
  err1 = x - (ahi * bhi); \
 
232
  err2 = err1 - (alo * bhi); \
 
233
  err3 = err2 - (ahi * blo); \
 
234
  y = (alo * blo) - err3
 
235
 
 
236
/* Two_Product_2Presplit() is Two_Product() where both of the inputs have    */
 
237
/*   already been split.  Avoids redundant splitting.                        */
 
238
 
 
239
#define Two_Product_2Presplit(a, ahi, alo, b, bhi, blo, x, y) \
 
240
  x = (REAL) (a * b); \
 
241
  err1 = x - (ahi * bhi); \
 
242
  err2 = err1 - (alo * bhi); \
 
243
  err3 = err2 - (ahi * blo); \
 
244
  y = (alo * blo) - err3
 
245
 
 
246
/* Square() can be done more quickly than Two_Product().                     */
 
247
 
 
248
#define Square_Tail(a, x, y) \
 
249
  Split(a, ahi, alo); \
 
250
  err1 = x - (ahi * ahi); \
 
251
  err3 = err1 - ((ahi + ahi) * alo); \
 
252
  y = (alo * alo) - err3
 
253
 
 
254
#define Square(a, x, y) \
 
255
  x = (REAL) (a * a); \
 
256
  Square_Tail(a, x, y)
 
257
 
 
258
/* Macros for summing expansions of various fixed lengths.  These are all    */
 
259
/*   unrolled versions of Expansion_Sum().                                   */
 
260
 
 
261
#define Two_One_Sum(a1, a0, b, x2, x1, x0) \
 
262
  Two_Sum(a0, b , _i, x0); \
 
263
  Two_Sum(a1, _i, x2, x1)
 
264
 
 
265
#define Two_One_Diff(a1, a0, b, x2, x1, x0) \
 
266
  Two_Diff(a0, b , _i, x0); \
 
267
  Two_Sum( a1, _i, x2, x1)
 
268
 
 
269
#define Two_Two_Sum(a1, a0, b1, b0, x3, x2, x1, x0) \
 
270
  Two_One_Sum(a1, a0, b0, _j, _0, x0); \
 
271
  Two_One_Sum(_j, _0, b1, x3, x2, x1)
 
272
 
 
273
#define Two_Two_Diff(a1, a0, b1, b0, x3, x2, x1, x0) \
 
274
  Two_One_Diff(a1, a0, b0, _j, _0, x0); \
 
275
  Two_One_Diff(_j, _0, b1, x3, x2, x1)
 
276
 
 
277
#define Four_One_Sum(a3, a2, a1, a0, b, x4, x3, x2, x1, x0) \
 
278
  Two_One_Sum(a1, a0, b , _j, x1, x0); \
 
279
  Two_One_Sum(a3, a2, _j, x4, x3, x2)
 
280
 
 
281
#define Four_Two_Sum(a3, a2, a1, a0, b1, b0, x5, x4, x3, x2, x1, x0) \
 
282
  Four_One_Sum(a3, a2, a1, a0, b0, _k, _2, _1, _0, x0); \
 
283
  Four_One_Sum(_k, _2, _1, _0, b1, x5, x4, x3, x2, x1)
 
284
 
 
285
#define Four_Four_Sum(a3, a2, a1, a0, b4, b3, b1, b0, x7, x6, x5, x4, x3, x2, \
 
286
                      x1, x0) \
 
287
  Four_Two_Sum(a3, a2, a1, a0, b1, b0, _l, _2, _1, _0, x1, x0); \
 
288
  Four_Two_Sum(_l, _2, _1, _0, b4, b3, x7, x6, x5, x4, x3, x2)
 
289
 
 
290
#define Eight_One_Sum(a7, a6, a5, a4, a3, a2, a1, a0, b, x8, x7, x6, x5, x4, \
 
291
                      x3, x2, x1, x0) \
 
292
  Four_One_Sum(a3, a2, a1, a0, b , _j, x3, x2, x1, x0); \
 
293
  Four_One_Sum(a7, a6, a5, a4, _j, x8, x7, x6, x5, x4)
 
294
 
 
295
#define Eight_Two_Sum(a7, a6, a5, a4, a3, a2, a1, a0, b1, b0, x9, x8, x7, \
 
296
                      x6, x5, x4, x3, x2, x1, x0) \
 
297
  Eight_One_Sum(a7, a6, a5, a4, a3, a2, a1, a0, b0, _k, _6, _5, _4, _3, _2, \
 
298
                _1, _0, x0); \
 
299
  Eight_One_Sum(_k, _6, _5, _4, _3, _2, _1, _0, b1, x9, x8, x7, x6, x5, x4, \
 
300
                x3, x2, x1)
 
301
 
 
302
#define Eight_Four_Sum(a7, a6, a5, a4, a3, a2, a1, a0, b4, b3, b1, b0, x11, \
 
303
                       x10, x9, x8, x7, x6, x5, x4, x3, x2, x1, x0) \
 
304
  Eight_Two_Sum(a7, a6, a5, a4, a3, a2, a1, a0, b1, b0, _l, _6, _5, _4, _3, \
 
305
                _2, _1, _0, x1, x0); \
 
306
  Eight_Two_Sum(_l, _6, _5, _4, _3, _2, _1, _0, b4, b3, x11, x10, x9, x8, \
 
307
                x7, x6, x5, x4, x3, x2)
 
308
 
 
309
/* Macros for multiplying expansions of various fixed lengths.               */
 
310
 
 
311
#define Two_One_Product(a1, a0, b, x3, x2, x1, x0) \
 
312
  Split(b, bhi, blo); \
 
313
  Two_Product_Presplit(a0, b, bhi, blo, _i, x0); \
 
314
  Two_Product_Presplit(a1, b, bhi, blo, _j, _0); \
 
315
  Two_Sum(_i, _0, _k, x1); \
 
316
  Fast_Two_Sum(_j, _k, x3, x2)
 
317
 
 
318
#define Four_One_Product(a3, a2, a1, a0, b, x7, x6, x5, x4, x3, x2, x1, x0) \
 
319
  Split(b, bhi, blo); \
 
320
  Two_Product_Presplit(a0, b, bhi, blo, _i, x0); \
 
321
  Two_Product_Presplit(a1, b, bhi, blo, _j, _0); \
 
322
  Two_Sum(_i, _0, _k, x1); \
 
323
  Fast_Two_Sum(_j, _k, _i, x2); \
 
324
  Two_Product_Presplit(a2, b, bhi, blo, _j, _0); \
 
325
  Two_Sum(_i, _0, _k, x3); \
 
326
  Fast_Two_Sum(_j, _k, _i, x4); \
 
327
  Two_Product_Presplit(a3, b, bhi, blo, _j, _0); \
 
328
  Two_Sum(_i, _0, _k, x5); \
 
329
  Fast_Two_Sum(_j, _k, x7, x6)
 
330
 
 
331
#define Two_Two_Product(a1, a0, b1, b0, x7, x6, x5, x4, x3, x2, x1, x0) \
 
332
  Split(a0, a0hi, a0lo); \
 
333
  Split(b0, bhi, blo); \
 
334
  Two_Product_2Presplit(a0, a0hi, a0lo, b0, bhi, blo, _i, x0); \
 
335
  Split(a1, a1hi, a1lo); \
 
336
  Two_Product_2Presplit(a1, a1hi, a1lo, b0, bhi, blo, _j, _0); \
 
337
  Two_Sum(_i, _0, _k, _1); \
 
338
  Fast_Two_Sum(_j, _k, _l, _2); \
 
339
  Split(b1, bhi, blo); \
 
340
  Two_Product_2Presplit(a0, a0hi, a0lo, b1, bhi, blo, _i, _0); \
 
341
  Two_Sum(_1, _0, _k, x1); \
 
342
  Two_Sum(_2, _k, _j, _1); \
 
343
  Two_Sum(_l, _j, _m, _2); \
 
344
  Two_Product_2Presplit(a1, a1hi, a1lo, b1, bhi, blo, _j, _0); \
 
345
  Two_Sum(_i, _0, _n, _0); \
 
346
  Two_Sum(_1, _0, _i, x2); \
 
347
  Two_Sum(_2, _i, _k, _1); \
 
348
  Two_Sum(_m, _k, _l, _2); \
 
349
  Two_Sum(_j, _n, _k, _0); \
 
350
  Two_Sum(_1, _0, _j, x3); \
 
351
  Two_Sum(_2, _j, _i, _1); \
 
352
  Two_Sum(_l, _i, _m, _2); \
 
353
  Two_Sum(_1, _k, _i, x4); \
 
354
  Two_Sum(_2, _i, _k, x5); \
 
355
  Two_Sum(_m, _k, x7, x6)
 
356
 
 
357
/* An expansion of length two can be squared more quickly than finding the   */
 
358
/*   product of two different expansions of length two, and the result is    */
 
359
/*   guaranteed to have no more than six (rather than eight) components.     */
 
360
 
 
361
#define Two_Square(a1, a0, x5, x4, x3, x2, x1, x0) \
 
362
  Square(a0, _j, x0); \
 
363
  _0 = a0 + a0; \
 
364
  Two_Product(a1, _0, _k, _1); \
 
365
  Two_One_Sum(_k, _1, _j, _l, _2, x1); \
 
366
  Square(a1, _j, _1); \
 
367
  Two_Two_Sum(_j, _1, _l, _2, x5, x4, x3, x2)
 
368
 
 
369
/* splitter = 2^ceiling(p / 2) + 1.  Used to split floats in half.           */
 
370
static REAL splitter;
 
371
static REAL epsilon;         /* = 2^(-p).  Used to estimate roundoff errors. */
 
372
/* A set of coefficients used to calculate maximum roundoff errors.          */
 
373
static REAL resulterrbound;
 
374
static REAL ccwerrboundA, ccwerrboundB, ccwerrboundC;
 
375
static REAL o3derrboundA, o3derrboundB, o3derrboundC;
 
376
static REAL iccerrboundA, iccerrboundB, iccerrboundC;
 
377
static REAL isperrboundA, isperrboundB, isperrboundC;
 
378
 
 
379
/*****************************************************************************/
 
380
/*                                                                           */
 
381
/*  doubleprint()   Print the bit representation of a double.                */
 
382
/*                                                                           */
 
383
/*  Useful for debugging exact arithmetic routines.                          */
 
384
/*                                                                           */
 
385
/*****************************************************************************/
 
386
 
 
387
/*
 
388
void doubleprint(number)
 
389
double number;
 
390
{
 
391
  unsigned long long no;
 
392
  unsigned long long sign, expo;
 
393
  int exponent;
 
394
  int i, bottomi;
 
395
 
 
396
  no = *(unsigned long long *) &number;
 
397
  sign = no & 0x8000000000000000ll;
 
398
  expo = (no >> 52) & 0x7ffll;
 
399
  exponent = (int) expo;
 
400
  exponent = exponent - 1023;
 
401
  if (sign) {
 
402
    printf("-");
 
403
  } else {
 
404
    printf(" ");
 
405
  }
 
406
  if (exponent == -1023) {
 
407
    printf(
 
408
      "0.0000000000000000000000000000000000000000000000000000_     (   )");
 
409
  } else {
 
410
    printf("1.");
 
411
    bottomi = -1;
 
412
    for (i = 0; i < 52; i++) {
 
413
      if (no & 0x0008000000000000ll) {
 
414
        printf("1");
 
415
        bottomi = i;
 
416
      } else {
 
417
        printf("0");
 
418
      }
 
419
      no <<= 1;
 
420
    }
 
421
    printf("_%d  (%d)", exponent, exponent - 1 - bottomi);
 
422
  }
 
423
}
 
424
*/
 
425
 
 
426
/*****************************************************************************/
 
427
/*                                                                           */
 
428
/*  floatprint()   Print the bit representation of a float.                  */
 
429
/*                                                                           */
 
430
/*  Useful for debugging exact arithmetic routines.                          */
 
431
/*                                                                           */
 
432
/*****************************************************************************/
 
433
 
 
434
/*
 
435
void floatprint(number)
 
436
float number;
 
437
{
 
438
  unsigned no;
 
439
  unsigned sign, expo;
 
440
  int exponent;
 
441
  int i, bottomi;
 
442
 
 
443
  no = *(unsigned *) &number;
 
444
  sign = no & 0x80000000;
 
445
  expo = (no >> 23) & 0xff;
 
446
  exponent = (int) expo;
 
447
  exponent = exponent - 127;
 
448
  if (sign) {
 
449
    printf("-");
 
450
  } else {
 
451
    printf(" ");
 
452
  }
 
453
  if (exponent == -127) {
 
454
    printf("0.00000000000000000000000_     (   )");
 
455
  } else {
 
456
    printf("1.");
 
457
    bottomi = -1;
 
458
    for (i = 0; i < 23; i++) {
 
459
      if (no & 0x00400000) {
 
460
        printf("1");
 
461
        bottomi = i;
 
462
      } else {
 
463
        printf("0");
 
464
      }
 
465
      no <<= 1;
 
466
    }
 
467
    printf("_%3d  (%3d)", exponent, exponent - 1 - bottomi);
 
468
  }
 
469
}
 
470
*/
 
471
 
 
472
/*****************************************************************************/
 
473
/*                                                                           */
 
474
/*  expansion_print()   Print the bit representation of an expansion.        */
 
475
/*                                                                           */
 
476
/*  Useful for debugging exact arithmetic routines.                          */
 
477
/*                                                                           */
 
478
/*****************************************************************************/
 
479
 
 
480
/*
 
481
void expansion_print(elen, e)
 
482
int elen;
 
483
REAL *e;
 
484
{
 
485
  int i;
 
486
 
 
487
  for (i = elen - 1; i >= 0; i--) {
 
488
    REALPRINT(e[i]);
 
489
    if (i > 0) {
 
490
      printf(" +\n");
 
491
    } else {
 
492
      printf("\n");
 
493
    }
 
494
  }
 
495
}
 
496
*/
 
497
 
 
498
/*****************************************************************************/
 
499
/*                                                                           */
 
500
/*  doublerand()   Generate a double with random 53-bit significand and a    */
 
501
/*                 random exponent in [0, 511].                              */
 
502
/*                                                                           */
 
503
/*****************************************************************************/
 
504
 
 
505
/*
 
506
double doublerand()
 
507
{
 
508
  double result;
 
509
  double expo;
 
510
  long a, b, c;
 
511
  long i;
 
512
 
 
513
  a = random();
 
514
  b = random();
 
515
  c = random();
 
516
  result = (double) (a - 1073741824) * 8388608.0 + (double) (b >> 8);
 
517
  for (i = 512, expo = 2; i <= 131072; i *= 2, expo = expo * expo) {
 
518
    if (c & i) {
 
519
      result *= expo;
 
520
    }
 
521
  }
 
522
  return result;
 
523
}
 
524
*/
 
525
 
 
526
/*****************************************************************************/
 
527
/*                                                                           */
 
528
/*  narrowdoublerand()   Generate a double with random 53-bit significand    */
 
529
/*                       and a random exponent in [0, 7].                    */
 
530
/*                                                                           */
 
531
/*****************************************************************************/
 
532
 
 
533
/*
 
534
double narrowdoublerand()
 
535
{
 
536
  double result;
 
537
  double expo;
 
538
  long a, b, c;
 
539
  long i;
 
540
 
 
541
  a = random();
 
542
  b = random();
 
543
  c = random();
 
544
  result = (double) (a - 1073741824) * 8388608.0 + (double) (b >> 8);
 
545
  for (i = 512, expo = 2; i <= 2048; i *= 2, expo = expo * expo) {
 
546
    if (c & i) {
 
547
      result *= expo;
 
548
    }
 
549
  }
 
550
  return result;
 
551
}
 
552
*/
 
553
 
 
554
/*****************************************************************************/
 
555
/*                                                                           */
 
556
/*  uniformdoublerand()   Generate a double with random 53-bit significand.  */
 
557
/*                                                                           */
 
558
/*****************************************************************************/
 
559
 
 
560
/*
 
561
double uniformdoublerand()
 
562
{
 
563
  double result;
 
564
  long a, b;
 
565
 
 
566
  a = random();
 
567
  b = random();
 
568
  result = (double) (a - 1073741824) * 8388608.0 + (double) (b >> 8);
 
569
  return result;
 
570
}
 
571
*/
 
572
 
 
573
/*****************************************************************************/
 
574
/*                                                                           */
 
575
/*  floatrand()   Generate a float with random 24-bit significand and a      */
 
576
/*                random exponent in [0, 63].                                */
 
577
/*                                                                           */
 
578
/*****************************************************************************/
 
579
 
 
580
/*
 
581
float floatrand()
 
582
{
 
583
  float result;
 
584
  float expo;
 
585
  long a, c;
 
586
  long i;
 
587
 
 
588
  a = random();
 
589
  c = random();
 
590
  result = (float) ((a - 1073741824) >> 6);
 
591
  for (i = 512, expo = 2; i <= 16384; i *= 2, expo = expo * expo) {
 
592
    if (c & i) {
 
593
      result *= expo;
 
594
    }
 
595
  }
 
596
  return result;
 
597
}
 
598
*/
 
599
 
 
600
/*****************************************************************************/
 
601
/*                                                                           */
 
602
/*  narrowfloatrand()   Generate a float with random 24-bit significand and  */
 
603
/*                      a random exponent in [0, 7].                         */
 
604
/*                                                                           */
 
605
/*****************************************************************************/
 
606
 
 
607
/*
 
608
float narrowfloatrand()
 
609
{
 
610
  float result;
 
611
  float expo;
 
612
  long a, c;
 
613
  long i;
 
614
 
 
615
  a = random();
 
616
  c = random();
 
617
  result = (float) ((a - 1073741824) >> 6);
 
618
  for (i = 512, expo = 2; i <= 2048; i *= 2, expo = expo * expo) {
 
619
    if (c & i) {
 
620
      result *= expo;
 
621
    }
 
622
  }
 
623
  return result;
 
624
}
 
625
*/
 
626
 
 
627
/*****************************************************************************/
 
628
/*                                                                           */
 
629
/*  uniformfloatrand()   Generate a float with random 24-bit significand.    */
 
630
/*                                                                           */
 
631
/*****************************************************************************/
 
632
 
 
633
/*
 
634
float uniformfloatrand()
 
635
{
 
636
  float result;
 
637
  long a;
 
638
 
 
639
  a = random();
 
640
  result = (float) ((a - 1073741824) >> 6);
 
641
  return result;
 
642
}
 
643
*/
 
644
 
 
645
/*****************************************************************************/
 
646
/*                                                                           */
 
647
/*  exactinit()   Initialize the variables used for exact arithmetic.        */
 
648
/*                                                                           */
 
649
/*  `epsilon' is the largest power of two such that 1.0 + epsilon = 1.0 in   */
 
650
/*  floating-point arithmetic.  `epsilon' bounds the relative roundoff       */
 
651
/*  error.  It is used for floating-point error analysis.                    */
 
652
/*                                                                           */
 
653
/*  `splitter' is used to split floating-point numbers into two half-        */
 
654
/*  length significands for exact multiplication.                            */
 
655
/*                                                                           */
 
656
/*  I imagine that a highly optimizing compiler might be too smart for its   */
 
657
/*  own good, and somehow cause this routine to fail, if it pretends that    */
 
658
/*  floating-point arithmetic is too much like real arithmetic.              */
 
659
/*                                                                           */
 
660
/*  Don't change this routine unless you fully understand it.                */
 
661
/*                                                                           */
 
662
/*****************************************************************************/
 
663
 
 
664
REAL exactinit()
 
665
{
 
666
  REAL half;
 
667
  REAL check, lastcheck;
 
668
  int every_other;
 
669
#ifdef LINUX
 
670
  int cword;
 
671
#endif /* LINUX */
 
672
 
 
673
#ifdef CPU86
 
674
#ifdef SINGLE
 
675
  _control87(_PC_24, _MCW_PC); /* Set FPU control word for single precision. */
 
676
#else /* not SINGLE */
 
677
  _control87(_PC_53, _MCW_PC); /* Set FPU control word for double precision. */
 
678
#endif /* not SINGLE */
 
679
#endif /* CPU86 */
 
680
#ifdef LINUX
 
681
#ifdef SINGLE
 
682
  /*  cword = 4223; */
 
683
  cword = 4210;                 /* set FPU control word for single precision */
 
684
#else /* not SINGLE */
 
685
  /*  cword = 4735; */
 
686
  cword = 4722;                 /* set FPU control word for double precision */
 
687
#endif /* not SINGLE */
 
688
  _FPU_SETCW(cword);
 
689
#endif /* LINUX */
 
690
 
 
691
  every_other = 1;
 
692
  half = 0.5;
 
693
  epsilon = 1.0;
 
694
  splitter = 1.0;
 
695
  check = 1.0;
 
696
  /* Repeatedly divide `epsilon' by two until it is too small to add to    */
 
697
  /*   one without causing roundoff.  (Also check if the sum is equal to   */
 
698
  /*   the previous sum, for machines that round up instead of using exact */
 
699
  /*   rounding.  Not that this library will work on such machines anyway. */
 
700
  do {
 
701
    lastcheck = check;
 
702
    epsilon *= half;
 
703
    if (every_other) {
 
704
      splitter *= 2.0;
 
705
    }
 
706
    every_other = !every_other;
 
707
    check = 1.0 + epsilon;
 
708
  } while ((check != 1.0) && (check != lastcheck));
 
709
  splitter += 1.0;
 
710
 
 
711
  /* Error bounds for orientation and incircle tests. */
 
712
  resulterrbound = (3.0 + 8.0 * epsilon) * epsilon;
 
713
  ccwerrboundA = (3.0 + 16.0 * epsilon) * epsilon;
 
714
  ccwerrboundB = (2.0 + 12.0 * epsilon) * epsilon;
 
715
  ccwerrboundC = (9.0 + 64.0 * epsilon) * epsilon * epsilon;
 
716
  o3derrboundA = (7.0 + 56.0 * epsilon) * epsilon;
 
717
  o3derrboundB = (3.0 + 28.0 * epsilon) * epsilon;
 
718
  o3derrboundC = (26.0 + 288.0 * epsilon) * epsilon * epsilon;
 
719
  iccerrboundA = (10.0 + 96.0 * epsilon) * epsilon;
 
720
  iccerrboundB = (4.0 + 48.0 * epsilon) * epsilon;
 
721
  iccerrboundC = (44.0 + 576.0 * epsilon) * epsilon * epsilon;
 
722
  isperrboundA = (16.0 + 224.0 * epsilon) * epsilon;
 
723
  isperrboundB = (5.0 + 72.0 * epsilon) * epsilon;
 
724
  isperrboundC = (71.0 + 1408.0 * epsilon) * epsilon * epsilon;
 
725
 
 
726
  return epsilon; /* Added by H. Si 30 Juli, 2004. */
 
727
}
 
728
 
 
729
/*****************************************************************************/
 
730
/*                                                                           */
 
731
/*  grow_expansion()   Add a scalar to an expansion.                         */
 
732
/*                                                                           */
 
733
/*  Sets h = e + b.  See the long version of my paper for details.           */
 
734
/*                                                                           */
 
735
/*  Maintains the nonoverlapping property.  If round-to-even is used (as     */
 
736
/*  with IEEE 754), maintains the strongly nonoverlapping and nonadjacent    */
 
737
/*  properties as well.  (That is, if e has one of these properties, so      */
 
738
/*  will h.)                                                                 */
 
739
/*                                                                           */
 
740
/*****************************************************************************/
 
741
 
 
742
int grow_expansion(int elen, REAL *e, REAL b, REAL *h)
 
743
/* e and h can be the same. */
 
744
{
 
745
  REAL Q;
 
746
  INEXACT REAL Qnew;
 
747
  int eindex;
 
748
  REAL enow;
 
749
  INEXACT REAL bvirt;
 
750
  REAL avirt, bround, around;
 
751
 
 
752
  Q = b;
 
753
  for (eindex = 0; eindex < elen; eindex++) {
 
754
    enow = e[eindex];
 
755
    Two_Sum(Q, enow, Qnew, h[eindex]);
 
756
    Q = Qnew;
 
757
  }
 
758
  h[eindex] = Q;
 
759
  return eindex + 1;
 
760
}
 
761
 
 
762
/*****************************************************************************/
 
763
/*                                                                           */
 
764
/*  grow_expansion_zeroelim()   Add a scalar to an expansion, eliminating    */
 
765
/*                              zero components from the output expansion.   */
 
766
/*                                                                           */
 
767
/*  Sets h = e + b.  See the long version of my paper for details.           */
 
768
/*                                                                           */
 
769
/*  Maintains the nonoverlapping property.  If round-to-even is used (as     */
 
770
/*  with IEEE 754), maintains the strongly nonoverlapping and nonadjacent    */
 
771
/*  properties as well.  (That is, if e has one of these properties, so      */
 
772
/*  will h.)                                                                 */
 
773
/*                                                                           */
 
774
/*****************************************************************************/
 
775
 
 
776
int grow_expansion_zeroelim(int elen, REAL *e, REAL b, REAL *h)
 
777
/* e and h can be the same. */
 
778
{
 
779
  REAL Q, hh;
 
780
  INEXACT REAL Qnew;
 
781
  int eindex, hindex;
 
782
  REAL enow;
 
783
  INEXACT REAL bvirt;
 
784
  REAL avirt, bround, around;
 
785
 
 
786
  hindex = 0;
 
787
  Q = b;
 
788
  for (eindex = 0; eindex < elen; eindex++) {
 
789
    enow = e[eindex];
 
790
    Two_Sum(Q, enow, Qnew, hh);
 
791
    Q = Qnew;
 
792
    if (hh != 0.0) {
 
793
      h[hindex++] = hh;
 
794
    }
 
795
  }
 
796
  if ((Q != 0.0) || (hindex == 0)) {
 
797
    h[hindex++] = Q;
 
798
  }
 
799
  return hindex;
 
800
}
 
801
 
 
802
/*****************************************************************************/
 
803
/*                                                                           */
 
804
/*  expansion_sum()   Sum two expansions.                                    */
 
805
/*                                                                           */
 
806
/*  Sets h = e + f.  See the long version of my paper for details.           */
 
807
/*                                                                           */
 
808
/*  Maintains the nonoverlapping property.  If round-to-even is used (as     */
 
809
/*  with IEEE 754), maintains the nonadjacent property as well.  (That is,   */
 
810
/*  if e has one of these properties, so will h.)  Does NOT maintain the     */
 
811
/*  strongly nonoverlapping property.                                        */
 
812
/*                                                                           */
 
813
/*****************************************************************************/
 
814
 
 
815
int expansion_sum(int elen, REAL *e, int flen, REAL *f, REAL *h)
 
816
/* e and h can be the same, but f and h cannot. */
 
817
{
 
818
  REAL Q;
 
819
  INEXACT REAL Qnew;
 
820
  int findex, hindex, hlast;
 
821
  REAL hnow;
 
822
  INEXACT REAL bvirt;
 
823
  REAL avirt, bround, around;
 
824
 
 
825
  Q = f[0];
 
826
  for (hindex = 0; hindex < elen; hindex++) {
 
827
    hnow = e[hindex];
 
828
    Two_Sum(Q, hnow, Qnew, h[hindex]);
 
829
    Q = Qnew;
 
830
  }
 
831
  h[hindex] = Q;
 
832
  hlast = hindex;
 
833
  for (findex = 1; findex < flen; findex++) {
 
834
    Q = f[findex];
 
835
    for (hindex = findex; hindex <= hlast; hindex++) {
 
836
      hnow = h[hindex];
 
837
      Two_Sum(Q, hnow, Qnew, h[hindex]);
 
838
      Q = Qnew;
 
839
    }
 
840
    h[++hlast] = Q;
 
841
  }
 
842
  return hlast + 1;
 
843
}
 
844
 
 
845
/*****************************************************************************/
 
846
/*                                                                           */
 
847
/*  expansion_sum_zeroelim1()   Sum two expansions, eliminating zero         */
 
848
/*                              components from the output expansion.        */
 
849
/*                                                                           */
 
850
/*  Sets h = e + f.  See the long version of my paper for details.           */
 
851
/*                                                                           */
 
852
/*  Maintains the nonoverlapping property.  If round-to-even is used (as     */
 
853
/*  with IEEE 754), maintains the nonadjacent property as well.  (That is,   */
 
854
/*  if e has one of these properties, so will h.)  Does NOT maintain the     */
 
855
/*  strongly nonoverlapping property.                                        */
 
856
/*                                                                           */
 
857
/*****************************************************************************/
 
858
 
 
859
int expansion_sum_zeroelim1(int elen, REAL *e, int flen, REAL *f, REAL *h)
 
860
/* e and h can be the same, but f and h cannot. */
 
861
{
 
862
  REAL Q;
 
863
  INEXACT REAL Qnew;
 
864
  int index, findex, hindex, hlast;
 
865
  REAL hnow;
 
866
  INEXACT REAL bvirt;
 
867
  REAL avirt, bround, around;
 
868
 
 
869
  Q = f[0];
 
870
  for (hindex = 0; hindex < elen; hindex++) {
 
871
    hnow = e[hindex];
 
872
    Two_Sum(Q, hnow, Qnew, h[hindex]);
 
873
    Q = Qnew;
 
874
  }
 
875
  h[hindex] = Q;
 
876
  hlast = hindex;
 
877
  for (findex = 1; findex < flen; findex++) {
 
878
    Q = f[findex];
 
879
    for (hindex = findex; hindex <= hlast; hindex++) {
 
880
      hnow = h[hindex];
 
881
      Two_Sum(Q, hnow, Qnew, h[hindex]);
 
882
      Q = Qnew;
 
883
    }
 
884
    h[++hlast] = Q;
 
885
  }
 
886
  hindex = -1;
 
887
  for (index = 0; index <= hlast; index++) {
 
888
    hnow = h[index];
 
889
    if (hnow != 0.0) {
 
890
      h[++hindex] = hnow;
 
891
    }
 
892
  }
 
893
  if (hindex == -1) {
 
894
    return 1;
 
895
  } else {
 
896
    return hindex + 1;
 
897
  }
 
898
}
 
899
 
 
900
/*****************************************************************************/
 
901
/*                                                                           */
 
902
/*  expansion_sum_zeroelim2()   Sum two expansions, eliminating zero         */
 
903
/*                              components from the output expansion.        */
 
904
/*                                                                           */
 
905
/*  Sets h = e + f.  See the long version of my paper for details.           */
 
906
/*                                                                           */
 
907
/*  Maintains the nonoverlapping property.  If round-to-even is used (as     */
 
908
/*  with IEEE 754), maintains the nonadjacent property as well.  (That is,   */
 
909
/*  if e has one of these properties, so will h.)  Does NOT maintain the     */
 
910
/*  strongly nonoverlapping property.                                        */
 
911
/*                                                                           */
 
912
/*****************************************************************************/
 
913
 
 
914
int expansion_sum_zeroelim2(int elen, REAL *e, int flen, REAL *f, REAL *h)
 
915
/* e and h can be the same, but f and h cannot. */
 
916
{
 
917
  REAL Q, hh;
 
918
  INEXACT REAL Qnew;
 
919
  int eindex, findex, hindex, hlast;
 
920
  REAL enow;
 
921
  INEXACT REAL bvirt;
 
922
  REAL avirt, bround, around;
 
923
 
 
924
  hindex = 0;
 
925
  Q = f[0];
 
926
  for (eindex = 0; eindex < elen; eindex++) {
 
927
    enow = e[eindex];
 
928
    Two_Sum(Q, enow, Qnew, hh);
 
929
    Q = Qnew;
 
930
    if (hh != 0.0) {
 
931
      h[hindex++] = hh;
 
932
    }
 
933
  }
 
934
  h[hindex] = Q;
 
935
  hlast = hindex;
 
936
  for (findex = 1; findex < flen; findex++) {
 
937
    hindex = 0;
 
938
    Q = f[findex];
 
939
    for (eindex = 0; eindex <= hlast; eindex++) {
 
940
      enow = h[eindex];
 
941
      Two_Sum(Q, enow, Qnew, hh);
 
942
      Q = Qnew;
 
943
      if (hh != 0) {
 
944
        h[hindex++] = hh;
 
945
      }
 
946
    }
 
947
    h[hindex] = Q;
 
948
    hlast = hindex;
 
949
  }
 
950
  return hlast + 1;
 
951
}
 
952
 
 
953
/*****************************************************************************/
 
954
/*                                                                           */
 
955
/*  fast_expansion_sum()   Sum two expansions.                               */
 
956
/*                                                                           */
 
957
/*  Sets h = e + f.  See the long version of my paper for details.           */
 
958
/*                                                                           */
 
959
/*  If round-to-even is used (as with IEEE 754), maintains the strongly      */
 
960
/*  nonoverlapping property.  (That is, if e is strongly nonoverlapping, h   */
 
961
/*  will be also.)  Does NOT maintain the nonoverlapping or nonadjacent      */
 
962
/*  properties.                                                              */
 
963
/*                                                                           */
 
964
/*****************************************************************************/
 
965
 
 
966
int fast_expansion_sum(int elen, REAL *e, int flen, REAL *f, REAL *h)
 
967
/* h cannot be e or f. */
 
968
{
 
969
  REAL Q;
 
970
  INEXACT REAL Qnew;
 
971
  INEXACT REAL bvirt;
 
972
  REAL avirt, bround, around;
 
973
  int eindex, findex, hindex;
 
974
  REAL enow, fnow;
 
975
 
 
976
  enow = e[0];
 
977
  fnow = f[0];
 
978
  eindex = findex = 0;
 
979
  if ((fnow > enow) == (fnow > -enow)) {
 
980
    Q = enow;
 
981
    enow = e[++eindex];
 
982
  } else {
 
983
    Q = fnow;
 
984
    fnow = f[++findex];
 
985
  }
 
986
  hindex = 0;
 
987
  if ((eindex < elen) && (findex < flen)) {
 
988
    if ((fnow > enow) == (fnow > -enow)) {
 
989
      Fast_Two_Sum(enow, Q, Qnew, h[0]);
 
990
      enow = e[++eindex];
 
991
    } else {
 
992
      Fast_Two_Sum(fnow, Q, Qnew, h[0]);
 
993
      fnow = f[++findex];
 
994
    }
 
995
    Q = Qnew;
 
996
    hindex = 1;
 
997
    while ((eindex < elen) && (findex < flen)) {
 
998
      if ((fnow > enow) == (fnow > -enow)) {
 
999
        Two_Sum(Q, enow, Qnew, h[hindex]);
 
1000
        enow = e[++eindex];
 
1001
      } else {
 
1002
        Two_Sum(Q, fnow, Qnew, h[hindex]);
 
1003
        fnow = f[++findex];
 
1004
      }
 
1005
      Q = Qnew;
 
1006
      hindex++;
 
1007
    }
 
1008
  }
 
1009
  while (eindex < elen) {
 
1010
    Two_Sum(Q, enow, Qnew, h[hindex]);
 
1011
    enow = e[++eindex];
 
1012
    Q = Qnew;
 
1013
    hindex++;
 
1014
  }
 
1015
  while (findex < flen) {
 
1016
    Two_Sum(Q, fnow, Qnew, h[hindex]);
 
1017
    fnow = f[++findex];
 
1018
    Q = Qnew;
 
1019
    hindex++;
 
1020
  }
 
1021
  h[hindex] = Q;
 
1022
  return hindex + 1;
 
1023
}
 
1024
 
 
1025
/*****************************************************************************/
 
1026
/*                                                                           */
 
1027
/*  fast_expansion_sum_zeroelim()   Sum two expansions, eliminating zero     */
 
1028
/*                                  components from the output expansion.    */
 
1029
/*                                                                           */
 
1030
/*  Sets h = e + f.  See the long version of my paper for details.           */
 
1031
/*                                                                           */
 
1032
/*  If round-to-even is used (as with IEEE 754), maintains the strongly      */
 
1033
/*  nonoverlapping property.  (That is, if e is strongly nonoverlapping, h   */
 
1034
/*  will be also.)  Does NOT maintain the nonoverlapping or nonadjacent      */
 
1035
/*  properties.                                                              */
 
1036
/*                                                                           */
 
1037
/*****************************************************************************/
 
1038
 
 
1039
int fast_expansion_sum_zeroelim(int elen, REAL *e, int flen, REAL *f, REAL *h)
 
1040
/* h cannot be e or f. */
 
1041
{
 
1042
  REAL Q;
 
1043
  INEXACT REAL Qnew;
 
1044
  INEXACT REAL hh;
 
1045
  INEXACT REAL bvirt;
 
1046
  REAL avirt, bround, around;
 
1047
  int eindex, findex, hindex;
 
1048
  REAL enow, fnow;
 
1049
 
 
1050
  enow = e[0];
 
1051
  fnow = f[0];
 
1052
  eindex = findex = 0;
 
1053
  if ((fnow > enow) == (fnow > -enow)) {
 
1054
    Q = enow;
 
1055
    enow = e[++eindex];
 
1056
  } else {
 
1057
    Q = fnow;
 
1058
    fnow = f[++findex];
 
1059
  }
 
1060
  hindex = 0;
 
1061
  if ((eindex < elen) && (findex < flen)) {
 
1062
    if ((fnow > enow) == (fnow > -enow)) {
 
1063
      Fast_Two_Sum(enow, Q, Qnew, hh);
 
1064
      enow = e[++eindex];
 
1065
    } else {
 
1066
      Fast_Two_Sum(fnow, Q, Qnew, hh);
 
1067
      fnow = f[++findex];
 
1068
    }
 
1069
    Q = Qnew;
 
1070
    if (hh != 0.0) {
 
1071
      h[hindex++] = hh;
 
1072
    }
 
1073
    while ((eindex < elen) && (findex < flen)) {
 
1074
      if ((fnow > enow) == (fnow > -enow)) {
 
1075
        Two_Sum(Q, enow, Qnew, hh);
 
1076
        enow = e[++eindex];
 
1077
      } else {
 
1078
        Two_Sum(Q, fnow, Qnew, hh);
 
1079
        fnow = f[++findex];
 
1080
      }
 
1081
      Q = Qnew;
 
1082
      if (hh != 0.0) {
 
1083
        h[hindex++] = hh;
 
1084
      }
 
1085
    }
 
1086
  }
 
1087
  while (eindex < elen) {
 
1088
    Two_Sum(Q, enow, Qnew, hh);
 
1089
    enow = e[++eindex];
 
1090
    Q = Qnew;
 
1091
    if (hh != 0.0) {
 
1092
      h[hindex++] = hh;
 
1093
    }
 
1094
  }
 
1095
  while (findex < flen) {
 
1096
    Two_Sum(Q, fnow, Qnew, hh);
 
1097
    fnow = f[++findex];
 
1098
    Q = Qnew;
 
1099
    if (hh != 0.0) {
 
1100
      h[hindex++] = hh;
 
1101
    }
 
1102
  }
 
1103
  if ((Q != 0.0) || (hindex == 0)) {
 
1104
    h[hindex++] = Q;
 
1105
  }
 
1106
  return hindex;
 
1107
}
 
1108
 
 
1109
/*****************************************************************************/
 
1110
/*                                                                           */
 
1111
/*  linear_expansion_sum()   Sum two expansions.                             */
 
1112
/*                                                                           */
 
1113
/*  Sets h = e + f.  See either version of my paper for details.             */
 
1114
/*                                                                           */
 
1115
/*  Maintains the nonoverlapping property.  (That is, if e is                */
 
1116
/*  nonoverlapping, h will be also.)                                         */
 
1117
/*                                                                           */
 
1118
/*****************************************************************************/
 
1119
 
 
1120
int linear_expansion_sum(int elen, REAL *e, int flen, REAL *f, REAL *h)
 
1121
/* h cannot be e or f. */
 
1122
{
 
1123
  REAL Q, q;
 
1124
  INEXACT REAL Qnew;
 
1125
  INEXACT REAL R;
 
1126
  INEXACT REAL bvirt;
 
1127
  REAL avirt, bround, around;
 
1128
  int eindex, findex, hindex;
 
1129
  REAL enow, fnow;
 
1130
  REAL g0;
 
1131
 
 
1132
  enow = e[0];
 
1133
  fnow = f[0];
 
1134
  eindex = findex = 0;
 
1135
  if ((fnow > enow) == (fnow > -enow)) {
 
1136
    g0 = enow;
 
1137
    enow = e[++eindex];
 
1138
  } else {
 
1139
    g0 = fnow;
 
1140
    fnow = f[++findex];
 
1141
  }
 
1142
  if ((eindex < elen) && ((findex >= flen)
 
1143
                          || ((fnow > enow) == (fnow > -enow)))) {
 
1144
    Fast_Two_Sum(enow, g0, Qnew, q);
 
1145
    enow = e[++eindex];
 
1146
  } else {
 
1147
    Fast_Two_Sum(fnow, g0, Qnew, q);
 
1148
    fnow = f[++findex];
 
1149
  }
 
1150
  Q = Qnew;
 
1151
  for (hindex = 0; hindex < elen + flen - 2; hindex++) {
 
1152
    if ((eindex < elen) && ((findex >= flen)
 
1153
                            || ((fnow > enow) == (fnow > -enow)))) {
 
1154
      Fast_Two_Sum(enow, q, R, h[hindex]);
 
1155
      enow = e[++eindex];
 
1156
    } else {
 
1157
      Fast_Two_Sum(fnow, q, R, h[hindex]);
 
1158
      fnow = f[++findex];
 
1159
    }
 
1160
    Two_Sum(Q, R, Qnew, q);
 
1161
    Q = Qnew;
 
1162
  }
 
1163
  h[hindex] = q;
 
1164
  h[hindex + 1] = Q;
 
1165
  return hindex + 2;
 
1166
}
 
1167
 
 
1168
/*****************************************************************************/
 
1169
/*                                                                           */
 
1170
/*  linear_expansion_sum_zeroelim()   Sum two expansions, eliminating zero   */
 
1171
/*                                    components from the output expansion.  */
 
1172
/*                                                                           */
 
1173
/*  Sets h = e + f.  See either version of my paper for details.             */
 
1174
/*                                                                           */
 
1175
/*  Maintains the nonoverlapping property.  (That is, if e is                */
 
1176
/*  nonoverlapping, h will be also.)                                         */
 
1177
/*                                                                           */
 
1178
/*****************************************************************************/
 
1179
 
 
1180
int linear_expansion_sum_zeroelim(int elen, REAL *e, int flen, REAL *f,
 
1181
                                  REAL *h)
 
1182
/* h cannot be e or f. */
 
1183
{
 
1184
  REAL Q, q, hh;
 
1185
  INEXACT REAL Qnew;
 
1186
  INEXACT REAL R;
 
1187
  INEXACT REAL bvirt;
 
1188
  REAL avirt, bround, around;
 
1189
  int eindex, findex, hindex;
 
1190
  int count;
 
1191
  REAL enow, fnow;
 
1192
  REAL g0;
 
1193
 
 
1194
  enow = e[0];
 
1195
  fnow = f[0];
 
1196
  eindex = findex = 0;
 
1197
  hindex = 0;
 
1198
  if ((fnow > enow) == (fnow > -enow)) {
 
1199
    g0 = enow;
 
1200
    enow = e[++eindex];
 
1201
  } else {
 
1202
    g0 = fnow;
 
1203
    fnow = f[++findex];
 
1204
  }
 
1205
  if ((eindex < elen) && ((findex >= flen)
 
1206
                          || ((fnow > enow) == (fnow > -enow)))) {
 
1207
    Fast_Two_Sum(enow, g0, Qnew, q);
 
1208
    enow = e[++eindex];
 
1209
  } else {
 
1210
    Fast_Two_Sum(fnow, g0, Qnew, q);
 
1211
    fnow = f[++findex];
 
1212
  }
 
1213
  Q = Qnew;
 
1214
  for (count = 2; count < elen + flen; count++) {
 
1215
    if ((eindex < elen) && ((findex >= flen)
 
1216
                            || ((fnow > enow) == (fnow > -enow)))) {
 
1217
      Fast_Two_Sum(enow, q, R, hh);
 
1218
      enow = e[++eindex];
 
1219
    } else {
 
1220
      Fast_Two_Sum(fnow, q, R, hh);
 
1221
      fnow = f[++findex];
 
1222
    }
 
1223
    Two_Sum(Q, R, Qnew, q);
 
1224
    Q = Qnew;
 
1225
    if (hh != 0) {
 
1226
      h[hindex++] = hh;
 
1227
    }
 
1228
  }
 
1229
  if (q != 0) {
 
1230
    h[hindex++] = q;
 
1231
  }
 
1232
  if ((Q != 0.0) || (hindex == 0)) {
 
1233
    h[hindex++] = Q;
 
1234
  }
 
1235
  return hindex;
 
1236
}
 
1237
 
 
1238
/*****************************************************************************/
 
1239
/*                                                                           */
 
1240
/*  scale_expansion()   Multiply an expansion by a scalar.                   */
 
1241
/*                                                                           */
 
1242
/*  Sets h = be.  See either version of my paper for details.                */
 
1243
/*                                                                           */
 
1244
/*  Maintains the nonoverlapping property.  If round-to-even is used (as     */
 
1245
/*  with IEEE 754), maintains the strongly nonoverlapping and nonadjacent    */
 
1246
/*  properties as well.  (That is, if e has one of these properties, so      */
 
1247
/*  will h.)                                                                 */
 
1248
/*                                                                           */
 
1249
/*****************************************************************************/
 
1250
 
 
1251
int scale_expansion(int elen, REAL *e, REAL b, REAL *h)
 
1252
/* e and h cannot be the same. */
 
1253
{
 
1254
  INEXACT REAL Q;
 
1255
  INEXACT REAL sum;
 
1256
  INEXACT REAL product1;
 
1257
  REAL product0;
 
1258
  int eindex, hindex;
 
1259
  REAL enow;
 
1260
  INEXACT REAL bvirt;
 
1261
  REAL avirt, bround, around;
 
1262
  INEXACT REAL c;
 
1263
  INEXACT REAL abig;
 
1264
  REAL ahi, alo, bhi, blo;
 
1265
  REAL err1, err2, err3;
 
1266
 
 
1267
  Split(b, bhi, blo);
 
1268
  Two_Product_Presplit(e[0], b, bhi, blo, Q, h[0]);
 
1269
  hindex = 1;
 
1270
  for (eindex = 1; eindex < elen; eindex++) {
 
1271
    enow = e[eindex];
 
1272
    Two_Product_Presplit(enow, b, bhi, blo, product1, product0);
 
1273
    Two_Sum(Q, product0, sum, h[hindex]);
 
1274
    hindex++;
 
1275
    Two_Sum(product1, sum, Q, h[hindex]);
 
1276
    hindex++;
 
1277
  }
 
1278
  h[hindex] = Q;
 
1279
  return elen + elen;
 
1280
}
 
1281
 
 
1282
/*****************************************************************************/
 
1283
/*                                                                           */
 
1284
/*  scale_expansion_zeroelim()   Multiply an expansion by a scalar,          */
 
1285
/*                               eliminating zero components from the        */
 
1286
/*                               output expansion.                           */
 
1287
/*                                                                           */
 
1288
/*  Sets h = be.  See either version of my paper for details.                */
 
1289
/*                                                                           */
 
1290
/*  Maintains the nonoverlapping property.  If round-to-even is used (as     */
 
1291
/*  with IEEE 754), maintains the strongly nonoverlapping and nonadjacent    */
 
1292
/*  properties as well.  (That is, if e has one of these properties, so      */
 
1293
/*  will h.)                                                                 */
 
1294
/*                                                                           */
 
1295
/*****************************************************************************/
 
1296
 
 
1297
int scale_expansion_zeroelim(int elen, REAL *e, REAL b, REAL *h)
 
1298
/* e and h cannot be the same. */
 
1299
{
 
1300
  INEXACT REAL Q, sum;
 
1301
  REAL hh;
 
1302
  INEXACT REAL product1;
 
1303
  REAL product0;
 
1304
  int eindex, hindex;
 
1305
  REAL enow;
 
1306
  INEXACT REAL bvirt;
 
1307
  REAL avirt, bround, around;
 
1308
  INEXACT REAL c;
 
1309
  INEXACT REAL abig;
 
1310
  REAL ahi, alo, bhi, blo;
 
1311
  REAL err1, err2, err3;
 
1312
 
 
1313
  Split(b, bhi, blo);
 
1314
  Two_Product_Presplit(e[0], b, bhi, blo, Q, hh);
 
1315
  hindex = 0;
 
1316
  if (hh != 0) {
 
1317
    h[hindex++] = hh;
 
1318
  }
 
1319
  for (eindex = 1; eindex < elen; eindex++) {
 
1320
    enow = e[eindex];
 
1321
    Two_Product_Presplit(enow, b, bhi, blo, product1, product0);
 
1322
    Two_Sum(Q, product0, sum, hh);
 
1323
    if (hh != 0) {
 
1324
      h[hindex++] = hh;
 
1325
    }
 
1326
    Fast_Two_Sum(product1, sum, Q, hh);
 
1327
    if (hh != 0) {
 
1328
      h[hindex++] = hh;
 
1329
    }
 
1330
  }
 
1331
  if ((Q != 0.0) || (hindex == 0)) {
 
1332
    h[hindex++] = Q;
 
1333
  }
 
1334
  return hindex;
 
1335
}
 
1336
 
 
1337
/*****************************************************************************/
 
1338
/*                                                                           */
 
1339
/*  compress()   Compress an expansion.                                      */
 
1340
/*                                                                           */
 
1341
/*  See the long version of my paper for details.                            */
 
1342
/*                                                                           */
 
1343
/*  Maintains the nonoverlapping property.  If round-to-even is used (as     */
 
1344
/*  with IEEE 754), then any nonoverlapping expansion is converted to a      */
 
1345
/*  nonadjacent expansion.                                                   */
 
1346
/*                                                                           */
 
1347
/*****************************************************************************/
 
1348
 
 
1349
int compress(int elen, REAL *e, REAL *h)
 
1350
/* e and h may be the same. */
 
1351
{
 
1352
  REAL Q, q;
 
1353
  INEXACT REAL Qnew;
 
1354
  int eindex, hindex;
 
1355
  INEXACT REAL bvirt;
 
1356
  REAL enow, hnow;
 
1357
  int top, bottom;
 
1358
 
 
1359
  bottom = elen - 1;
 
1360
  Q = e[bottom];
 
1361
  for (eindex = elen - 2; eindex >= 0; eindex--) {
 
1362
    enow = e[eindex];
 
1363
    Fast_Two_Sum(Q, enow, Qnew, q);
 
1364
    if (q != 0) {
 
1365
      h[bottom--] = Qnew;
 
1366
      Q = q;
 
1367
    } else {
 
1368
      Q = Qnew;
 
1369
    }
 
1370
  }
 
1371
  top = 0;
 
1372
  for (hindex = bottom + 1; hindex < elen; hindex++) {
 
1373
    hnow = h[hindex];
 
1374
    Fast_Two_Sum(hnow, Q, Qnew, q);
 
1375
    if (q != 0) {
 
1376
      h[top++] = q;
 
1377
    }
 
1378
    Q = Qnew;
 
1379
  }
 
1380
  h[top] = Q;
 
1381
  return top + 1;
 
1382
}
 
1383
 
 
1384
/*****************************************************************************/
 
1385
/*                                                                           */
 
1386
/*  estimate()   Produce a one-word estimate of an expansion's value.        */
 
1387
/*                                                                           */
 
1388
/*  See either version of my paper for details.                              */
 
1389
/*                                                                           */
 
1390
/*****************************************************************************/
 
1391
 
 
1392
REAL estimate(int elen, REAL *e)
 
1393
{
 
1394
  REAL Q;
 
1395
  int eindex;
 
1396
 
 
1397
  Q = e[0];
 
1398
  for (eindex = 1; eindex < elen; eindex++) {
 
1399
    Q += e[eindex];
 
1400
  }
 
1401
  return Q;
 
1402
}
 
1403
 
 
1404
/*****************************************************************************/
 
1405
/*                                                                           */
 
1406
/*  orient2dfast()   Approximate 2D orientation test.  Nonrobust.            */
 
1407
/*  orient2dexact()   Exact 2D orientation test.  Robust.                    */
 
1408
/*  orient2dslow()   Another exact 2D orientation test.  Robust.             */
 
1409
/*  orient2d()   Adaptive exact 2D orientation test.  Robust.                */
 
1410
/*                                                                           */
 
1411
/*               Return a positive value if the points pa, pb, and pc occur  */
 
1412
/*               in counterclockwise order; a negative value if they occur   */
 
1413
/*               in clockwise order; and zero if they are collinear.  The    */
 
1414
/*               result is also a rough approximation of twice the signed    */
 
1415
/*               area of the triangle defined by the three points.           */
 
1416
/*                                                                           */
 
1417
/*  Only the first and last routine should be used; the middle two are for   */
 
1418
/*  timings.                                                                 */
 
1419
/*                                                                           */
 
1420
/*  The last three use exact arithmetic to ensure a correct answer.  The     */
 
1421
/*  result returned is the determinant of a matrix.  In orient2d() only,     */
 
1422
/*  this determinant is computed adaptively, in the sense that exact         */
 
1423
/*  arithmetic is used only to the degree it is needed to ensure that the    */
 
1424
/*  returned value has the correct sign.  Hence, orient2d() is usually quite */
 
1425
/*  fast, but will run more slowly when the input points are collinear or    */
 
1426
/*  nearly so.                                                               */
 
1427
/*                                                                           */
 
1428
/*****************************************************************************/
 
1429
 
 
1430
REAL orient2dfast(REAL *pa, REAL *pb, REAL *pc)
 
1431
{
 
1432
  REAL acx, bcx, acy, bcy;
 
1433
 
 
1434
  acx = pa[0] - pc[0];
 
1435
  bcx = pb[0] - pc[0];
 
1436
  acy = pa[1] - pc[1];
 
1437
  bcy = pb[1] - pc[1];
 
1438
  return acx * bcy - acy * bcx;
 
1439
}
 
1440
 
 
1441
REAL orient2dexact(REAL *pa, REAL *pb, REAL *pc)
 
1442
{
 
1443
  INEXACT REAL axby1, axcy1, bxcy1, bxay1, cxay1, cxby1;
 
1444
  REAL axby0, axcy0, bxcy0, bxay0, cxay0, cxby0;
 
1445
  REAL aterms[4], bterms[4], cterms[4];
 
1446
  INEXACT REAL aterms3, bterms3, cterms3;
 
1447
  REAL v[8], w[12];
 
1448
  int vlength, wlength;
 
1449
 
 
1450
  INEXACT REAL bvirt;
 
1451
  REAL avirt, bround, around;
 
1452
  INEXACT REAL c;
 
1453
  INEXACT REAL abig;
 
1454
  REAL ahi, alo, bhi, blo;
 
1455
  REAL err1, err2, err3;
 
1456
  INEXACT REAL _i, _j;
 
1457
  REAL _0;
 
1458
 
 
1459
  Two_Product(pa[0], pb[1], axby1, axby0);
 
1460
  Two_Product(pa[0], pc[1], axcy1, axcy0);
 
1461
  Two_Two_Diff(axby1, axby0, axcy1, axcy0,
 
1462
               aterms3, aterms[2], aterms[1], aterms[0]);
 
1463
  aterms[3] = aterms3;
 
1464
 
 
1465
  Two_Product(pb[0], pc[1], bxcy1, bxcy0);
 
1466
  Two_Product(pb[0], pa[1], bxay1, bxay0);
 
1467
  Two_Two_Diff(bxcy1, bxcy0, bxay1, bxay0,
 
1468
               bterms3, bterms[2], bterms[1], bterms[0]);
 
1469
  bterms[3] = bterms3;
 
1470
 
 
1471
  Two_Product(pc[0], pa[1], cxay1, cxay0);
 
1472
  Two_Product(pc[0], pb[1], cxby1, cxby0);
 
1473
  Two_Two_Diff(cxay1, cxay0, cxby1, cxby0,
 
1474
               cterms3, cterms[2], cterms[1], cterms[0]);
 
1475
  cterms[3] = cterms3;
 
1476
 
 
1477
  vlength = fast_expansion_sum_zeroelim(4, aterms, 4, bterms, v);
 
1478
  wlength = fast_expansion_sum_zeroelim(vlength, v, 4, cterms, w);
 
1479
 
 
1480
  return w[wlength - 1];
 
1481
}
 
1482
 
 
1483
REAL orient2dslow(REAL *pa, REAL *pb, REAL *pc)
 
1484
{
 
1485
  INEXACT REAL acx, acy, bcx, bcy;
 
1486
  REAL acxtail, acytail;
 
1487
  REAL bcxtail, bcytail;
 
1488
  REAL negate, negatetail;
 
1489
  REAL axby[8], bxay[8];
 
1490
  INEXACT REAL axby7, bxay7;
 
1491
  REAL deter[16];
 
1492
  int deterlen;
 
1493
 
 
1494
  INEXACT REAL bvirt;
 
1495
  REAL avirt, bround, around;
 
1496
  INEXACT REAL c;
 
1497
  INEXACT REAL abig;
 
1498
  REAL a0hi, a0lo, a1hi, a1lo, bhi, blo;
 
1499
  REAL err1, err2, err3;
 
1500
  INEXACT REAL _i, _j, _k, _l, _m, _n;
 
1501
  REAL _0, _1, _2;
 
1502
 
 
1503
  Two_Diff(pa[0], pc[0], acx, acxtail);
 
1504
  Two_Diff(pa[1], pc[1], acy, acytail);
 
1505
  Two_Diff(pb[0], pc[0], bcx, bcxtail);
 
1506
  Two_Diff(pb[1], pc[1], bcy, bcytail);
 
1507
 
 
1508
  Two_Two_Product(acx, acxtail, bcy, bcytail,
 
1509
                  axby7, axby[6], axby[5], axby[4],
 
1510
                  axby[3], axby[2], axby[1], axby[0]);
 
1511
  axby[7] = axby7;
 
1512
  negate = -acy;
 
1513
  negatetail = -acytail;
 
1514
  Two_Two_Product(bcx, bcxtail, negate, negatetail,
 
1515
                  bxay7, bxay[6], bxay[5], bxay[4],
 
1516
                  bxay[3], bxay[2], bxay[1], bxay[0]);
 
1517
  bxay[7] = bxay7;
 
1518
 
 
1519
  deterlen = fast_expansion_sum_zeroelim(8, axby, 8, bxay, deter);
 
1520
 
 
1521
  return deter[deterlen - 1];
 
1522
}
 
1523
 
 
1524
REAL orient2dadapt(REAL *pa, REAL *pb, REAL *pc, REAL detsum)
 
1525
{
 
1526
  INEXACT REAL acx, acy, bcx, bcy;
 
1527
  REAL acxtail, acytail, bcxtail, bcytail;
 
1528
  INEXACT REAL detleft, detright;
 
1529
  REAL detlefttail, detrighttail;
 
1530
  REAL det, errbound;
 
1531
  REAL B[4], C1[8], C2[12], D[16];
 
1532
  INEXACT REAL B3;
 
1533
  int C1length, C2length, Dlength;
 
1534
  REAL u[4];
 
1535
  INEXACT REAL u3;
 
1536
  INEXACT REAL s1, t1;
 
1537
  REAL s0, t0;
 
1538
 
 
1539
  INEXACT REAL bvirt;
 
1540
  REAL avirt, bround, around;
 
1541
  INEXACT REAL c;
 
1542
  INEXACT REAL abig;
 
1543
  REAL ahi, alo, bhi, blo;
 
1544
  REAL err1, err2, err3;
 
1545
  INEXACT REAL _i, _j;
 
1546
  REAL _0;
 
1547
 
 
1548
  acx = (REAL) (pa[0] - pc[0]);
 
1549
  bcx = (REAL) (pb[0] - pc[0]);
 
1550
  acy = (REAL) (pa[1] - pc[1]);
 
1551
  bcy = (REAL) (pb[1] - pc[1]);
 
1552
 
 
1553
  Two_Product(acx, bcy, detleft, detlefttail);
 
1554
  Two_Product(acy, bcx, detright, detrighttail);
 
1555
 
 
1556
  Two_Two_Diff(detleft, detlefttail, detright, detrighttail,
 
1557
               B3, B[2], B[1], B[0]);
 
1558
  B[3] = B3;
 
1559
 
 
1560
  det = estimate(4, B);
 
1561
  errbound = ccwerrboundB * detsum;
 
1562
  if ((det >= errbound) || (-det >= errbound)) {
 
1563
    return det;
 
1564
  }
 
1565
 
 
1566
  Two_Diff_Tail(pa[0], pc[0], acx, acxtail);
 
1567
  Two_Diff_Tail(pb[0], pc[0], bcx, bcxtail);
 
1568
  Two_Diff_Tail(pa[1], pc[1], acy, acytail);
 
1569
  Two_Diff_Tail(pb[1], pc[1], bcy, bcytail);
 
1570
 
 
1571
  if ((acxtail == 0.0) && (acytail == 0.0)
 
1572
      && (bcxtail == 0.0) && (bcytail == 0.0)) {
 
1573
    return det;
 
1574
  }
 
1575
 
 
1576
  errbound = ccwerrboundC * detsum + resulterrbound * Absolute(det);
 
1577
  det += (acx * bcytail + bcy * acxtail)
 
1578
       - (acy * bcxtail + bcx * acytail);
 
1579
  if ((det >= errbound) || (-det >= errbound)) {
 
1580
    return det;
 
1581
  }
 
1582
 
 
1583
  Two_Product(acxtail, bcy, s1, s0);
 
1584
  Two_Product(acytail, bcx, t1, t0);
 
1585
  Two_Two_Diff(s1, s0, t1, t0, u3, u[2], u[1], u[0]);
 
1586
  u[3] = u3;
 
1587
  C1length = fast_expansion_sum_zeroelim(4, B, 4, u, C1);
 
1588
 
 
1589
  Two_Product(acx, bcytail, s1, s0);
 
1590
  Two_Product(acy, bcxtail, t1, t0);
 
1591
  Two_Two_Diff(s1, s0, t1, t0, u3, u[2], u[1], u[0]);
 
1592
  u[3] = u3;
 
1593
  C2length = fast_expansion_sum_zeroelim(C1length, C1, 4, u, C2);
 
1594
 
 
1595
  Two_Product(acxtail, bcytail, s1, s0);
 
1596
  Two_Product(acytail, bcxtail, t1, t0);
 
1597
  Two_Two_Diff(s1, s0, t1, t0, u3, u[2], u[1], u[0]);
 
1598
  u[3] = u3;
 
1599
  Dlength = fast_expansion_sum_zeroelim(C2length, C2, 4, u, D);
 
1600
 
 
1601
  return(D[Dlength - 1]);
 
1602
}
 
1603
 
 
1604
REAL orient2d(REAL *pa, REAL *pb, REAL *pc)
 
1605
{
 
1606
  REAL detleft, detright, det;
 
1607
  REAL detsum, errbound;
 
1608
 
 
1609
  detleft = (pa[0] - pc[0]) * (pb[1] - pc[1]);
 
1610
  detright = (pa[1] - pc[1]) * (pb[0] - pc[0]);
 
1611
  det = detleft - detright;
 
1612
 
 
1613
  if (detleft > 0.0) {
 
1614
    if (detright <= 0.0) {
 
1615
      return det;
 
1616
    } else {
 
1617
      detsum = detleft + detright;
 
1618
    }
 
1619
  } else if (detleft < 0.0) {
 
1620
    if (detright >= 0.0) {
 
1621
      return det;
 
1622
    } else {
 
1623
      detsum = -detleft - detright;
 
1624
    }
 
1625
  } else {
 
1626
    return det;
 
1627
  }
 
1628
 
 
1629
  errbound = ccwerrboundA * detsum;
 
1630
  if ((det >= errbound) || (-det >= errbound)) {
 
1631
    return det;
 
1632
  }
 
1633
 
 
1634
  return orient2dadapt(pa, pb, pc, detsum);
 
1635
}
 
1636
 
 
1637
/*****************************************************************************/
 
1638
/*                                                                           */
 
1639
/*  orient3dfast()   Approximate 3D orientation test.  Nonrobust.            */
 
1640
/*  orient3dexact()   Exact 3D orientation test.  Robust.                    */
 
1641
/*  orient3dslow()   Another exact 3D orientation test.  Robust.             */
 
1642
/*  orient3d()   Adaptive exact 3D orientation test.  Robust.                */
 
1643
/*                                                                           */
 
1644
/*               Return a positive value if the point pd lies below the      */
 
1645
/*               plane passing through pa, pb, and pc; "below" is defined so */
 
1646
/*               that pa, pb, and pc appear in counterclockwise order when   */
 
1647
/*               viewed from above the plane.  Returns a negative value if   */
 
1648
/*               pd lies above the plane.  Returns zero if the points are    */
 
1649
/*               coplanar.  The result is also a rough approximation of six  */
 
1650
/*               times the signed volume of the tetrahedron defined by the   */
 
1651
/*               four points.                                                */
 
1652
/*                                                                           */
 
1653
/*  Only the first and last routine should be used; the middle two are for   */
 
1654
/*  timings.                                                                 */
 
1655
/*                                                                           */
 
1656
/*  The last three use exact arithmetic to ensure a correct answer.  The     */
 
1657
/*  result returned is the determinant of a matrix.  In orient3d() only,     */
 
1658
/*  this determinant is computed adaptively, in the sense that exact         */
 
1659
/*  arithmetic is used only to the degree it is needed to ensure that the    */
 
1660
/*  returned value has the correct sign.  Hence, orient3d() is usually quite */
 
1661
/*  fast, but will run more slowly when the input points are coplanar or     */
 
1662
/*  nearly so.                                                               */
 
1663
/*                                                                           */
 
1664
/*****************************************************************************/
 
1665
 
 
1666
REAL orient3dfast(REAL *pa, REAL *pb, REAL *pc, REAL *pd)
 
1667
{
 
1668
  REAL adx, bdx, cdx;
 
1669
  REAL ady, bdy, cdy;
 
1670
  REAL adz, bdz, cdz;
 
1671
 
 
1672
  adx = pa[0] - pd[0];
 
1673
  bdx = pb[0] - pd[0];
 
1674
  cdx = pc[0] - pd[0];
 
1675
  ady = pa[1] - pd[1];
 
1676
  bdy = pb[1] - pd[1];
 
1677
  cdy = pc[1] - pd[1];
 
1678
  adz = pa[2] - pd[2];
 
1679
  bdz = pb[2] - pd[2];
 
1680
  cdz = pc[2] - pd[2];
 
1681
 
 
1682
  return adx * (bdy * cdz - bdz * cdy)
 
1683
       + bdx * (cdy * adz - cdz * ady)
 
1684
       + cdx * (ady * bdz - adz * bdy);
 
1685
}
 
1686
 
 
1687
REAL orient3dexact(REAL *pa, REAL *pb, REAL *pc, REAL *pd)
 
1688
{
 
1689
  INEXACT REAL axby1, bxcy1, cxdy1, dxay1, axcy1, bxdy1;
 
1690
  INEXACT REAL bxay1, cxby1, dxcy1, axdy1, cxay1, dxby1;
 
1691
  REAL axby0, bxcy0, cxdy0, dxay0, axcy0, bxdy0;
 
1692
  REAL bxay0, cxby0, dxcy0, axdy0, cxay0, dxby0;
 
1693
  REAL ab[4], bc[4], cd[4], da[4], ac[4], bd[4];
 
1694
  REAL temp8[8];
 
1695
  int templen;
 
1696
  REAL abc[12], bcd[12], cda[12], dab[12];
 
1697
  int abclen, bcdlen, cdalen, dablen;
 
1698
  REAL adet[24], bdet[24], cdet[24], ddet[24];
 
1699
  int alen, blen, clen, dlen;
 
1700
  REAL abdet[48], cddet[48];
 
1701
  int ablen, cdlen;
 
1702
  REAL deter[96];
 
1703
  int deterlen;
 
1704
  int i;
 
1705
 
 
1706
  INEXACT REAL bvirt;
 
1707
  REAL avirt, bround, around;
 
1708
  INEXACT REAL c;
 
1709
  INEXACT REAL abig;
 
1710
  REAL ahi, alo, bhi, blo;
 
1711
  REAL err1, err2, err3;
 
1712
  INEXACT REAL _i, _j;
 
1713
  REAL _0;
 
1714
 
 
1715
  Two_Product(pa[0], pb[1], axby1, axby0);
 
1716
  Two_Product(pb[0], pa[1], bxay1, bxay0);
 
1717
  Two_Two_Diff(axby1, axby0, bxay1, bxay0, ab[3], ab[2], ab[1], ab[0]);
 
1718
 
 
1719
  Two_Product(pb[0], pc[1], bxcy1, bxcy0);
 
1720
  Two_Product(pc[0], pb[1], cxby1, cxby0);
 
1721
  Two_Two_Diff(bxcy1, bxcy0, cxby1, cxby0, bc[3], bc[2], bc[1], bc[0]);
 
1722
 
 
1723
  Two_Product(pc[0], pd[1], cxdy1, cxdy0);
 
1724
  Two_Product(pd[0], pc[1], dxcy1, dxcy0);
 
1725
  Two_Two_Diff(cxdy1, cxdy0, dxcy1, dxcy0, cd[3], cd[2], cd[1], cd[0]);
 
1726
 
 
1727
  Two_Product(pd[0], pa[1], dxay1, dxay0);
 
1728
  Two_Product(pa[0], pd[1], axdy1, axdy0);
 
1729
  Two_Two_Diff(dxay1, dxay0, axdy1, axdy0, da[3], da[2], da[1], da[0]);
 
1730
 
 
1731
  Two_Product(pa[0], pc[1], axcy1, axcy0);
 
1732
  Two_Product(pc[0], pa[1], cxay1, cxay0);
 
1733
  Two_Two_Diff(axcy1, axcy0, cxay1, cxay0, ac[3], ac[2], ac[1], ac[0]);
 
1734
 
 
1735
  Two_Product(pb[0], pd[1], bxdy1, bxdy0);
 
1736
  Two_Product(pd[0], pb[1], dxby1, dxby0);
 
1737
  Two_Two_Diff(bxdy1, bxdy0, dxby1, dxby0, bd[3], bd[2], bd[1], bd[0]);
 
1738
 
 
1739
  templen = fast_expansion_sum_zeroelim(4, cd, 4, da, temp8);
 
1740
  cdalen = fast_expansion_sum_zeroelim(templen, temp8, 4, ac, cda);
 
1741
  templen = fast_expansion_sum_zeroelim(4, da, 4, ab, temp8);
 
1742
  dablen = fast_expansion_sum_zeroelim(templen, temp8, 4, bd, dab);
 
1743
  for (i = 0; i < 4; i++) {
 
1744
    bd[i] = -bd[i];
 
1745
    ac[i] = -ac[i];
 
1746
  }
 
1747
  templen = fast_expansion_sum_zeroelim(4, ab, 4, bc, temp8);
 
1748
  abclen = fast_expansion_sum_zeroelim(templen, temp8, 4, ac, abc);
 
1749
  templen = fast_expansion_sum_zeroelim(4, bc, 4, cd, temp8);
 
1750
  bcdlen = fast_expansion_sum_zeroelim(templen, temp8, 4, bd, bcd);
 
1751
 
 
1752
  alen = scale_expansion_zeroelim(bcdlen, bcd, pa[2], adet);
 
1753
  blen = scale_expansion_zeroelim(cdalen, cda, -pb[2], bdet);
 
1754
  clen = scale_expansion_zeroelim(dablen, dab, pc[2], cdet);
 
1755
  dlen = scale_expansion_zeroelim(abclen, abc, -pd[2], ddet);
 
1756
 
 
1757
  ablen = fast_expansion_sum_zeroelim(alen, adet, blen, bdet, abdet);
 
1758
  cdlen = fast_expansion_sum_zeroelim(clen, cdet, dlen, ddet, cddet);
 
1759
  deterlen = fast_expansion_sum_zeroelim(ablen, abdet, cdlen, cddet, deter);
 
1760
 
 
1761
  return deter[deterlen - 1];
 
1762
}
 
1763
 
 
1764
REAL orient3dslow(REAL *pa, REAL *pb, REAL *pc, REAL *pd)
 
1765
{
 
1766
  INEXACT REAL adx, ady, adz, bdx, bdy, bdz, cdx, cdy, cdz;
 
1767
  REAL adxtail, adytail, adztail;
 
1768
  REAL bdxtail, bdytail, bdztail;
 
1769
  REAL cdxtail, cdytail, cdztail;
 
1770
  REAL negate, negatetail;
 
1771
  INEXACT REAL axby7, bxcy7, axcy7, bxay7, cxby7, cxay7;
 
1772
  REAL axby[8], bxcy[8], axcy[8], bxay[8], cxby[8], cxay[8];
 
1773
  REAL temp16[16], temp32[32], temp32t[32];
 
1774
  int temp16len, temp32len, temp32tlen;
 
1775
  REAL adet[64], bdet[64], cdet[64];
 
1776
  int alen, blen, clen;
 
1777
  REAL abdet[128];
 
1778
  int ablen;
 
1779
  REAL deter[192];
 
1780
  int deterlen;
 
1781
 
 
1782
  INEXACT REAL bvirt;
 
1783
  REAL avirt, bround, around;
 
1784
  INEXACT REAL c;
 
1785
  INEXACT REAL abig;
 
1786
  REAL a0hi, a0lo, a1hi, a1lo, bhi, blo;
 
1787
  REAL err1, err2, err3;
 
1788
  INEXACT REAL _i, _j, _k, _l, _m, _n;
 
1789
  REAL _0, _1, _2;
 
1790
 
 
1791
  Two_Diff(pa[0], pd[0], adx, adxtail);
 
1792
  Two_Diff(pa[1], pd[1], ady, adytail);
 
1793
  Two_Diff(pa[2], pd[2], adz, adztail);
 
1794
  Two_Diff(pb[0], pd[0], bdx, bdxtail);
 
1795
  Two_Diff(pb[1], pd[1], bdy, bdytail);
 
1796
  Two_Diff(pb[2], pd[2], bdz, bdztail);
 
1797
  Two_Diff(pc[0], pd[0], cdx, cdxtail);
 
1798
  Two_Diff(pc[1], pd[1], cdy, cdytail);
 
1799
  Two_Diff(pc[2], pd[2], cdz, cdztail);
 
1800
 
 
1801
  Two_Two_Product(adx, adxtail, bdy, bdytail,
 
1802
                  axby7, axby[6], axby[5], axby[4],
 
1803
                  axby[3], axby[2], axby[1], axby[0]);
 
1804
  axby[7] = axby7;
 
1805
  negate = -ady;
 
1806
  negatetail = -adytail;
 
1807
  Two_Two_Product(bdx, bdxtail, negate, negatetail,
 
1808
                  bxay7, bxay[6], bxay[5], bxay[4],
 
1809
                  bxay[3], bxay[2], bxay[1], bxay[0]);
 
1810
  bxay[7] = bxay7;
 
1811
  Two_Two_Product(bdx, bdxtail, cdy, cdytail,
 
1812
                  bxcy7, bxcy[6], bxcy[5], bxcy[4],
 
1813
                  bxcy[3], bxcy[2], bxcy[1], bxcy[0]);
 
1814
  bxcy[7] = bxcy7;
 
1815
  negate = -bdy;
 
1816
  negatetail = -bdytail;
 
1817
  Two_Two_Product(cdx, cdxtail, negate, negatetail,
 
1818
                  cxby7, cxby[6], cxby[5], cxby[4],
 
1819
                  cxby[3], cxby[2], cxby[1], cxby[0]);
 
1820
  cxby[7] = cxby7;
 
1821
  Two_Two_Product(cdx, cdxtail, ady, adytail,
 
1822
                  cxay7, cxay[6], cxay[5], cxay[4],
 
1823
                  cxay[3], cxay[2], cxay[1], cxay[0]);
 
1824
  cxay[7] = cxay7;
 
1825
  negate = -cdy;
 
1826
  negatetail = -cdytail;
 
1827
  Two_Two_Product(adx, adxtail, negate, negatetail,
 
1828
                  axcy7, axcy[6], axcy[5], axcy[4],
 
1829
                  axcy[3], axcy[2], axcy[1], axcy[0]);
 
1830
  axcy[7] = axcy7;
 
1831
 
 
1832
  temp16len = fast_expansion_sum_zeroelim(8, bxcy, 8, cxby, temp16);
 
1833
  temp32len = scale_expansion_zeroelim(temp16len, temp16, adz, temp32);
 
1834
  temp32tlen = scale_expansion_zeroelim(temp16len, temp16, adztail, temp32t);
 
1835
  alen = fast_expansion_sum_zeroelim(temp32len, temp32, temp32tlen, temp32t,
 
1836
                                     adet);
 
1837
 
 
1838
  temp16len = fast_expansion_sum_zeroelim(8, cxay, 8, axcy, temp16);
 
1839
  temp32len = scale_expansion_zeroelim(temp16len, temp16, bdz, temp32);
 
1840
  temp32tlen = scale_expansion_zeroelim(temp16len, temp16, bdztail, temp32t);
 
1841
  blen = fast_expansion_sum_zeroelim(temp32len, temp32, temp32tlen, temp32t,
 
1842
                                     bdet);
 
1843
 
 
1844
  temp16len = fast_expansion_sum_zeroelim(8, axby, 8, bxay, temp16);
 
1845
  temp32len = scale_expansion_zeroelim(temp16len, temp16, cdz, temp32);
 
1846
  temp32tlen = scale_expansion_zeroelim(temp16len, temp16, cdztail, temp32t);
 
1847
  clen = fast_expansion_sum_zeroelim(temp32len, temp32, temp32tlen, temp32t,
 
1848
                                     cdet);
 
1849
 
 
1850
  ablen = fast_expansion_sum_zeroelim(alen, adet, blen, bdet, abdet);
 
1851
  deterlen = fast_expansion_sum_zeroelim(ablen, abdet, clen, cdet, deter);
 
1852
 
 
1853
  return deter[deterlen - 1];
 
1854
}
 
1855
 
 
1856
REAL orient3dadapt(REAL *pa, REAL *pb, REAL *pc, REAL *pd, REAL permanent)
 
1857
{
 
1858
  INEXACT REAL adx, bdx, cdx, ady, bdy, cdy, adz, bdz, cdz;
 
1859
  REAL det, errbound;
 
1860
 
 
1861
  INEXACT REAL bdxcdy1, cdxbdy1, cdxady1, adxcdy1, adxbdy1, bdxady1;
 
1862
  REAL bdxcdy0, cdxbdy0, cdxady0, adxcdy0, adxbdy0, bdxady0;
 
1863
  REAL bc[4], ca[4], ab[4];
 
1864
  INEXACT REAL bc3, ca3, ab3;
 
1865
  REAL adet[8], bdet[8], cdet[8];
 
1866
  int alen, blen, clen;
 
1867
  REAL abdet[16];
 
1868
  int ablen;
 
1869
  REAL *finnow, *finother, *finswap;
 
1870
  REAL fin1[192], fin2[192];
 
1871
  int finlength;
 
1872
 
 
1873
  REAL adxtail, bdxtail, cdxtail;
 
1874
  REAL adytail, bdytail, cdytail;
 
1875
  REAL adztail, bdztail, cdztail;
 
1876
  INEXACT REAL at_blarge, at_clarge;
 
1877
  INEXACT REAL bt_clarge, bt_alarge;
 
1878
  INEXACT REAL ct_alarge, ct_blarge;
 
1879
  REAL at_b[4], at_c[4], bt_c[4], bt_a[4], ct_a[4], ct_b[4];
 
1880
  int at_blen, at_clen, bt_clen, bt_alen, ct_alen, ct_blen;
 
1881
  INEXACT REAL bdxt_cdy1, cdxt_bdy1, cdxt_ady1;
 
1882
  INEXACT REAL adxt_cdy1, adxt_bdy1, bdxt_ady1;
 
1883
  REAL bdxt_cdy0, cdxt_bdy0, cdxt_ady0;
 
1884
  REAL adxt_cdy0, adxt_bdy0, bdxt_ady0;
 
1885
  INEXACT REAL bdyt_cdx1, cdyt_bdx1, cdyt_adx1;
 
1886
  INEXACT REAL adyt_cdx1, adyt_bdx1, bdyt_adx1;
 
1887
  REAL bdyt_cdx0, cdyt_bdx0, cdyt_adx0;
 
1888
  REAL adyt_cdx0, adyt_bdx0, bdyt_adx0;
 
1889
  REAL bct[8], cat[8], abt[8];
 
1890
  int bctlen, catlen, abtlen;
 
1891
  INEXACT REAL bdxt_cdyt1, cdxt_bdyt1, cdxt_adyt1;
 
1892
  INEXACT REAL adxt_cdyt1, adxt_bdyt1, bdxt_adyt1;
 
1893
  REAL bdxt_cdyt0, cdxt_bdyt0, cdxt_adyt0;
 
1894
  REAL adxt_cdyt0, adxt_bdyt0, bdxt_adyt0;
 
1895
  REAL u[4], v[12], w[16];
 
1896
  INEXACT REAL u3;
 
1897
  int vlength, wlength;
 
1898
  REAL negate;
 
1899
 
 
1900
  INEXACT REAL bvirt;
 
1901
  REAL avirt, bround, around;
 
1902
  INEXACT REAL c;
 
1903
  INEXACT REAL abig;
 
1904
  REAL ahi, alo, bhi, blo;
 
1905
  REAL err1, err2, err3;
 
1906
  INEXACT REAL _i, _j, _k;
 
1907
  REAL _0;
 
1908
 
 
1909
  adx = (REAL) (pa[0] - pd[0]);
 
1910
  bdx = (REAL) (pb[0] - pd[0]);
 
1911
  cdx = (REAL) (pc[0] - pd[0]);
 
1912
  ady = (REAL) (pa[1] - pd[1]);
 
1913
  bdy = (REAL) (pb[1] - pd[1]);
 
1914
  cdy = (REAL) (pc[1] - pd[1]);
 
1915
  adz = (REAL) (pa[2] - pd[2]);
 
1916
  bdz = (REAL) (pb[2] - pd[2]);
 
1917
  cdz = (REAL) (pc[2] - pd[2]);
 
1918
 
 
1919
  Two_Product(bdx, cdy, bdxcdy1, bdxcdy0);
 
1920
  Two_Product(cdx, bdy, cdxbdy1, cdxbdy0);
 
1921
  Two_Two_Diff(bdxcdy1, bdxcdy0, cdxbdy1, cdxbdy0, bc3, bc[2], bc[1], bc[0]);
 
1922
  bc[3] = bc3;
 
1923
  alen = scale_expansion_zeroelim(4, bc, adz, adet);
 
1924
 
 
1925
  Two_Product(cdx, ady, cdxady1, cdxady0);
 
1926
  Two_Product(adx, cdy, adxcdy1, adxcdy0);
 
1927
  Two_Two_Diff(cdxady1, cdxady0, adxcdy1, adxcdy0, ca3, ca[2], ca[1], ca[0]);
 
1928
  ca[3] = ca3;
 
1929
  blen = scale_expansion_zeroelim(4, ca, bdz, bdet);
 
1930
 
 
1931
  Two_Product(adx, bdy, adxbdy1, adxbdy0);
 
1932
  Two_Product(bdx, ady, bdxady1, bdxady0);
 
1933
  Two_Two_Diff(adxbdy1, adxbdy0, bdxady1, bdxady0, ab3, ab[2], ab[1], ab[0]);
 
1934
  ab[3] = ab3;
 
1935
  clen = scale_expansion_zeroelim(4, ab, cdz, cdet);
 
1936
 
 
1937
  ablen = fast_expansion_sum_zeroelim(alen, adet, blen, bdet, abdet);
 
1938
  finlength = fast_expansion_sum_zeroelim(ablen, abdet, clen, cdet, fin1);
 
1939
 
 
1940
  det = estimate(finlength, fin1);
 
1941
  errbound = o3derrboundB * permanent;
 
1942
  if ((det >= errbound) || (-det >= errbound)) {
 
1943
    return det;
 
1944
  }
 
1945
 
 
1946
  Two_Diff_Tail(pa[0], pd[0], adx, adxtail);
 
1947
  Two_Diff_Tail(pb[0], pd[0], bdx, bdxtail);
 
1948
  Two_Diff_Tail(pc[0], pd[0], cdx, cdxtail);
 
1949
  Two_Diff_Tail(pa[1], pd[1], ady, adytail);
 
1950
  Two_Diff_Tail(pb[1], pd[1], bdy, bdytail);
 
1951
  Two_Diff_Tail(pc[1], pd[1], cdy, cdytail);
 
1952
  Two_Diff_Tail(pa[2], pd[2], adz, adztail);
 
1953
  Two_Diff_Tail(pb[2], pd[2], bdz, bdztail);
 
1954
  Two_Diff_Tail(pc[2], pd[2], cdz, cdztail);
 
1955
 
 
1956
  if ((adxtail == 0.0) && (bdxtail == 0.0) && (cdxtail == 0.0)
 
1957
      && (adytail == 0.0) && (bdytail == 0.0) && (cdytail == 0.0)
 
1958
      && (adztail == 0.0) && (bdztail == 0.0) && (cdztail == 0.0)) {
 
1959
    return det;
 
1960
  }
 
1961
 
 
1962
  errbound = o3derrboundC * permanent + resulterrbound * Absolute(det);
 
1963
  det += (adz * ((bdx * cdytail + cdy * bdxtail)
 
1964
                 - (bdy * cdxtail + cdx * bdytail))
 
1965
          + adztail * (bdx * cdy - bdy * cdx))
 
1966
       + (bdz * ((cdx * adytail + ady * cdxtail)
 
1967
                 - (cdy * adxtail + adx * cdytail))
 
1968
          + bdztail * (cdx * ady - cdy * adx))
 
1969
       + (cdz * ((adx * bdytail + bdy * adxtail)
 
1970
                 - (ady * bdxtail + bdx * adytail))
 
1971
          + cdztail * (adx * bdy - ady * bdx));
 
1972
  if ((det >= errbound) || (-det >= errbound)) {
 
1973
    return det;
 
1974
  }
 
1975
 
 
1976
  finnow = fin1;
 
1977
  finother = fin2;
 
1978
 
 
1979
  if (adxtail == 0.0) {
 
1980
    if (adytail == 0.0) {
 
1981
      at_b[0] = 0.0;
 
1982
      at_blen = 1;
 
1983
      at_c[0] = 0.0;
 
1984
      at_clen = 1;
 
1985
    } else {
 
1986
      negate = -adytail;
 
1987
      Two_Product(negate, bdx, at_blarge, at_b[0]);
 
1988
      at_b[1] = at_blarge;
 
1989
      at_blen = 2;
 
1990
      Two_Product(adytail, cdx, at_clarge, at_c[0]);
 
1991
      at_c[1] = at_clarge;
 
1992
      at_clen = 2;
 
1993
    }
 
1994
  } else {
 
1995
    if (adytail == 0.0) {
 
1996
      Two_Product(adxtail, bdy, at_blarge, at_b[0]);
 
1997
      at_b[1] = at_blarge;
 
1998
      at_blen = 2;
 
1999
      negate = -adxtail;
 
2000
      Two_Product(negate, cdy, at_clarge, at_c[0]);
 
2001
      at_c[1] = at_clarge;
 
2002
      at_clen = 2;
 
2003
    } else {
 
2004
      Two_Product(adxtail, bdy, adxt_bdy1, adxt_bdy0);
 
2005
      Two_Product(adytail, bdx, adyt_bdx1, adyt_bdx0);
 
2006
      Two_Two_Diff(adxt_bdy1, adxt_bdy0, adyt_bdx1, adyt_bdx0,
 
2007
                   at_blarge, at_b[2], at_b[1], at_b[0]);
 
2008
      at_b[3] = at_blarge;
 
2009
      at_blen = 4;
 
2010
      Two_Product(adytail, cdx, adyt_cdx1, adyt_cdx0);
 
2011
      Two_Product(adxtail, cdy, adxt_cdy1, adxt_cdy0);
 
2012
      Two_Two_Diff(adyt_cdx1, adyt_cdx0, adxt_cdy1, adxt_cdy0,
 
2013
                   at_clarge, at_c[2], at_c[1], at_c[0]);
 
2014
      at_c[3] = at_clarge;
 
2015
      at_clen = 4;
 
2016
    }
 
2017
  }
 
2018
  if (bdxtail == 0.0) {
 
2019
    if (bdytail == 0.0) {
 
2020
      bt_c[0] = 0.0;
 
2021
      bt_clen = 1;
 
2022
      bt_a[0] = 0.0;
 
2023
      bt_alen = 1;
 
2024
    } else {
 
2025
      negate = -bdytail;
 
2026
      Two_Product(negate, cdx, bt_clarge, bt_c[0]);
 
2027
      bt_c[1] = bt_clarge;
 
2028
      bt_clen = 2;
 
2029
      Two_Product(bdytail, adx, bt_alarge, bt_a[0]);
 
2030
      bt_a[1] = bt_alarge;
 
2031
      bt_alen = 2;
 
2032
    }
 
2033
  } else {
 
2034
    if (bdytail == 0.0) {
 
2035
      Two_Product(bdxtail, cdy, bt_clarge, bt_c[0]);
 
2036
      bt_c[1] = bt_clarge;
 
2037
      bt_clen = 2;
 
2038
      negate = -bdxtail;
 
2039
      Two_Product(negate, ady, bt_alarge, bt_a[0]);
 
2040
      bt_a[1] = bt_alarge;
 
2041
      bt_alen = 2;
 
2042
    } else {
 
2043
      Two_Product(bdxtail, cdy, bdxt_cdy1, bdxt_cdy0);
 
2044
      Two_Product(bdytail, cdx, bdyt_cdx1, bdyt_cdx0);
 
2045
      Two_Two_Diff(bdxt_cdy1, bdxt_cdy0, bdyt_cdx1, bdyt_cdx0,
 
2046
                   bt_clarge, bt_c[2], bt_c[1], bt_c[0]);
 
2047
      bt_c[3] = bt_clarge;
 
2048
      bt_clen = 4;
 
2049
      Two_Product(bdytail, adx, bdyt_adx1, bdyt_adx0);
 
2050
      Two_Product(bdxtail, ady, bdxt_ady1, bdxt_ady0);
 
2051
      Two_Two_Diff(bdyt_adx1, bdyt_adx0, bdxt_ady1, bdxt_ady0,
 
2052
                  bt_alarge, bt_a[2], bt_a[1], bt_a[0]);
 
2053
      bt_a[3] = bt_alarge;
 
2054
      bt_alen = 4;
 
2055
    }
 
2056
  }
 
2057
  if (cdxtail == 0.0) {
 
2058
    if (cdytail == 0.0) {
 
2059
      ct_a[0] = 0.0;
 
2060
      ct_alen = 1;
 
2061
      ct_b[0] = 0.0;
 
2062
      ct_blen = 1;
 
2063
    } else {
 
2064
      negate = -cdytail;
 
2065
      Two_Product(negate, adx, ct_alarge, ct_a[0]);
 
2066
      ct_a[1] = ct_alarge;
 
2067
      ct_alen = 2;
 
2068
      Two_Product(cdytail, bdx, ct_blarge, ct_b[0]);
 
2069
      ct_b[1] = ct_blarge;
 
2070
      ct_blen = 2;
 
2071
    }
 
2072
  } else {
 
2073
    if (cdytail == 0.0) {
 
2074
      Two_Product(cdxtail, ady, ct_alarge, ct_a[0]);
 
2075
      ct_a[1] = ct_alarge;
 
2076
      ct_alen = 2;
 
2077
      negate = -cdxtail;
 
2078
      Two_Product(negate, bdy, ct_blarge, ct_b[0]);
 
2079
      ct_b[1] = ct_blarge;
 
2080
      ct_blen = 2;
 
2081
    } else {
 
2082
      Two_Product(cdxtail, ady, cdxt_ady1, cdxt_ady0);
 
2083
      Two_Product(cdytail, adx, cdyt_adx1, cdyt_adx0);
 
2084
      Two_Two_Diff(cdxt_ady1, cdxt_ady0, cdyt_adx1, cdyt_adx0,
 
2085
                   ct_alarge, ct_a[2], ct_a[1], ct_a[0]);
 
2086
      ct_a[3] = ct_alarge;
 
2087
      ct_alen = 4;
 
2088
      Two_Product(cdytail, bdx, cdyt_bdx1, cdyt_bdx0);
 
2089
      Two_Product(cdxtail, bdy, cdxt_bdy1, cdxt_bdy0);
 
2090
      Two_Two_Diff(cdyt_bdx1, cdyt_bdx0, cdxt_bdy1, cdxt_bdy0,
 
2091
                   ct_blarge, ct_b[2], ct_b[1], ct_b[0]);
 
2092
      ct_b[3] = ct_blarge;
 
2093
      ct_blen = 4;
 
2094
    }
 
2095
  }
 
2096
 
 
2097
  bctlen = fast_expansion_sum_zeroelim(bt_clen, bt_c, ct_blen, ct_b, bct);
 
2098
  wlength = scale_expansion_zeroelim(bctlen, bct, adz, w);
 
2099
  finlength = fast_expansion_sum_zeroelim(finlength, finnow, wlength, w,
 
2100
                                          finother);
 
2101
  finswap = finnow; finnow = finother; finother = finswap;
 
2102
 
 
2103
  catlen = fast_expansion_sum_zeroelim(ct_alen, ct_a, at_clen, at_c, cat);
 
2104
  wlength = scale_expansion_zeroelim(catlen, cat, bdz, w);
 
2105
  finlength = fast_expansion_sum_zeroelim(finlength, finnow, wlength, w,
 
2106
                                          finother);
 
2107
  finswap = finnow; finnow = finother; finother = finswap;
 
2108
 
 
2109
  abtlen = fast_expansion_sum_zeroelim(at_blen, at_b, bt_alen, bt_a, abt);
 
2110
  wlength = scale_expansion_zeroelim(abtlen, abt, cdz, w);
 
2111
  finlength = fast_expansion_sum_zeroelim(finlength, finnow, wlength, w,
 
2112
                                          finother);
 
2113
  finswap = finnow; finnow = finother; finother = finswap;
 
2114
 
 
2115
  if (adztail != 0.0) {
 
2116
    vlength = scale_expansion_zeroelim(4, bc, adztail, v);
 
2117
    finlength = fast_expansion_sum_zeroelim(finlength, finnow, vlength, v,
 
2118
                                            finother);
 
2119
    finswap = finnow; finnow = finother; finother = finswap;
 
2120
  }
 
2121
  if (bdztail != 0.0) {
 
2122
    vlength = scale_expansion_zeroelim(4, ca, bdztail, v);
 
2123
    finlength = fast_expansion_sum_zeroelim(finlength, finnow, vlength, v,
 
2124
                                            finother);
 
2125
    finswap = finnow; finnow = finother; finother = finswap;
 
2126
  }
 
2127
  if (cdztail != 0.0) {
 
2128
    vlength = scale_expansion_zeroelim(4, ab, cdztail, v);
 
2129
    finlength = fast_expansion_sum_zeroelim(finlength, finnow, vlength, v,
 
2130
                                            finother);
 
2131
    finswap = finnow; finnow = finother; finother = finswap;
 
2132
  }
 
2133
 
 
2134
  if (adxtail != 0.0) {
 
2135
    if (bdytail != 0.0) {
 
2136
      Two_Product(adxtail, bdytail, adxt_bdyt1, adxt_bdyt0);
 
2137
      Two_One_Product(adxt_bdyt1, adxt_bdyt0, cdz, u3, u[2], u[1], u[0]);
 
2138
      u[3] = u3;
 
2139
      finlength = fast_expansion_sum_zeroelim(finlength, finnow, 4, u,
 
2140
                                              finother);
 
2141
      finswap = finnow; finnow = finother; finother = finswap;
 
2142
      if (cdztail != 0.0) {
 
2143
        Two_One_Product(adxt_bdyt1, adxt_bdyt0, cdztail, u3, u[2], u[1], u[0]);
 
2144
        u[3] = u3;
 
2145
        finlength = fast_expansion_sum_zeroelim(finlength, finnow, 4, u,
 
2146
                                                finother);
 
2147
        finswap = finnow; finnow = finother; finother = finswap;
 
2148
      }
 
2149
    }
 
2150
    if (cdytail != 0.0) {
 
2151
      negate = -adxtail;
 
2152
      Two_Product(negate, cdytail, adxt_cdyt1, adxt_cdyt0);
 
2153
      Two_One_Product(adxt_cdyt1, adxt_cdyt0, bdz, u3, u[2], u[1], u[0]);
 
2154
      u[3] = u3;
 
2155
      finlength = fast_expansion_sum_zeroelim(finlength, finnow, 4, u,
 
2156
                                              finother);
 
2157
      finswap = finnow; finnow = finother; finother = finswap;
 
2158
      if (bdztail != 0.0) {
 
2159
        Two_One_Product(adxt_cdyt1, adxt_cdyt0, bdztail, u3, u[2], u[1], u[0]);
 
2160
        u[3] = u3;
 
2161
        finlength = fast_expansion_sum_zeroelim(finlength, finnow, 4, u,
 
2162
                                                finother);
 
2163
        finswap = finnow; finnow = finother; finother = finswap;
 
2164
      }
 
2165
    }
 
2166
  }
 
2167
  if (bdxtail != 0.0) {
 
2168
    if (cdytail != 0.0) {
 
2169
      Two_Product(bdxtail, cdytail, bdxt_cdyt1, bdxt_cdyt0);
 
2170
      Two_One_Product(bdxt_cdyt1, bdxt_cdyt0, adz, u3, u[2], u[1], u[0]);
 
2171
      u[3] = u3;
 
2172
      finlength = fast_expansion_sum_zeroelim(finlength, finnow, 4, u,
 
2173
                                              finother);
 
2174
      finswap = finnow; finnow = finother; finother = finswap;
 
2175
      if (adztail != 0.0) {
 
2176
        Two_One_Product(bdxt_cdyt1, bdxt_cdyt0, adztail, u3, u[2], u[1], u[0]);
 
2177
        u[3] = u3;
 
2178
        finlength = fast_expansion_sum_zeroelim(finlength, finnow, 4, u,
 
2179
                                                finother);
 
2180
        finswap = finnow; finnow = finother; finother = finswap;
 
2181
      }
 
2182
    }
 
2183
    if (adytail != 0.0) {
 
2184
      negate = -bdxtail;
 
2185
      Two_Product(negate, adytail, bdxt_adyt1, bdxt_adyt0);
 
2186
      Two_One_Product(bdxt_adyt1, bdxt_adyt0, cdz, u3, u[2], u[1], u[0]);
 
2187
      u[3] = u3;
 
2188
      finlength = fast_expansion_sum_zeroelim(finlength, finnow, 4, u,
 
2189
                                              finother);
 
2190
      finswap = finnow; finnow = finother; finother = finswap;
 
2191
      if (cdztail != 0.0) {
 
2192
        Two_One_Product(bdxt_adyt1, bdxt_adyt0, cdztail, u3, u[2], u[1], u[0]);
 
2193
        u[3] = u3;
 
2194
        finlength = fast_expansion_sum_zeroelim(finlength, finnow, 4, u,
 
2195
                                                finother);
 
2196
        finswap = finnow; finnow = finother; finother = finswap;
 
2197
      }
 
2198
    }
 
2199
  }
 
2200
  if (cdxtail != 0.0) {
 
2201
    if (adytail != 0.0) {
 
2202
      Two_Product(cdxtail, adytail, cdxt_adyt1, cdxt_adyt0);
 
2203
      Two_One_Product(cdxt_adyt1, cdxt_adyt0, bdz, u3, u[2], u[1], u[0]);
 
2204
      u[3] = u3;
 
2205
      finlength = fast_expansion_sum_zeroelim(finlength, finnow, 4, u,
 
2206
                                              finother);
 
2207
      finswap = finnow; finnow = finother; finother = finswap;
 
2208
      if (bdztail != 0.0) {
 
2209
        Two_One_Product(cdxt_adyt1, cdxt_adyt0, bdztail, u3, u[2], u[1], u[0]);
 
2210
        u[3] = u3;
 
2211
        finlength = fast_expansion_sum_zeroelim(finlength, finnow, 4, u,
 
2212
                                                finother);
 
2213
        finswap = finnow; finnow = finother; finother = finswap;
 
2214
      }
 
2215
    }
 
2216
    if (bdytail != 0.0) {
 
2217
      negate = -cdxtail;
 
2218
      Two_Product(negate, bdytail, cdxt_bdyt1, cdxt_bdyt0);
 
2219
      Two_One_Product(cdxt_bdyt1, cdxt_bdyt0, adz, u3, u[2], u[1], u[0]);
 
2220
      u[3] = u3;
 
2221
      finlength = fast_expansion_sum_zeroelim(finlength, finnow, 4, u,
 
2222
                                              finother);
 
2223
      finswap = finnow; finnow = finother; finother = finswap;
 
2224
      if (adztail != 0.0) {
 
2225
        Two_One_Product(cdxt_bdyt1, cdxt_bdyt0, adztail, u3, u[2], u[1], u[0]);
 
2226
        u[3] = u3;
 
2227
        finlength = fast_expansion_sum_zeroelim(finlength, finnow, 4, u,
 
2228
                                                finother);
 
2229
        finswap = finnow; finnow = finother; finother = finswap;
 
2230
      }
 
2231
    }
 
2232
  }
 
2233
 
 
2234
  if (adztail != 0.0) {
 
2235
    wlength = scale_expansion_zeroelim(bctlen, bct, adztail, w);
 
2236
    finlength = fast_expansion_sum_zeroelim(finlength, finnow, wlength, w,
 
2237
                                            finother);
 
2238
    finswap = finnow; finnow = finother; finother = finswap;
 
2239
  }
 
2240
  if (bdztail != 0.0) {
 
2241
    wlength = scale_expansion_zeroelim(catlen, cat, bdztail, w);
 
2242
    finlength = fast_expansion_sum_zeroelim(finlength, finnow, wlength, w,
 
2243
                                            finother);
 
2244
    finswap = finnow; finnow = finother; finother = finswap;
 
2245
  }
 
2246
  if (cdztail != 0.0) {
 
2247
    wlength = scale_expansion_zeroelim(abtlen, abt, cdztail, w);
 
2248
    finlength = fast_expansion_sum_zeroelim(finlength, finnow, wlength, w,
 
2249
                                            finother);
 
2250
    finswap = finnow; finnow = finother; finother = finswap;
 
2251
  }
 
2252
 
 
2253
  return finnow[finlength - 1];
 
2254
}
 
2255
 
 
2256
REAL orient3d(REAL *pa, REAL *pb, REAL *pc, REAL *pd)
 
2257
{
 
2258
  REAL adx, bdx, cdx, ady, bdy, cdy, adz, bdz, cdz;
 
2259
  REAL bdxcdy, cdxbdy, cdxady, adxcdy, adxbdy, bdxady;
 
2260
  REAL det;
 
2261
  REAL permanent, errbound;
 
2262
 
 
2263
  adx = pa[0] - pd[0];
 
2264
  bdx = pb[0] - pd[0];
 
2265
  cdx = pc[0] - pd[0];
 
2266
  ady = pa[1] - pd[1];
 
2267
  bdy = pb[1] - pd[1];
 
2268
  cdy = pc[1] - pd[1];
 
2269
  adz = pa[2] - pd[2];
 
2270
  bdz = pb[2] - pd[2];
 
2271
  cdz = pc[2] - pd[2];
 
2272
 
 
2273
  bdxcdy = bdx * cdy;
 
2274
  cdxbdy = cdx * bdy;
 
2275
 
 
2276
  cdxady = cdx * ady;
 
2277
  adxcdy = adx * cdy;
 
2278
 
 
2279
  adxbdy = adx * bdy;
 
2280
  bdxady = bdx * ady;
 
2281
 
 
2282
  det = adz * (bdxcdy - cdxbdy) 
 
2283
      + bdz * (cdxady - adxcdy)
 
2284
      + cdz * (adxbdy - bdxady);
 
2285
 
 
2286
  permanent = (Absolute(bdxcdy) + Absolute(cdxbdy)) * Absolute(adz)
 
2287
            + (Absolute(cdxady) + Absolute(adxcdy)) * Absolute(bdz)
 
2288
            + (Absolute(adxbdy) + Absolute(bdxady)) * Absolute(cdz);
 
2289
  errbound = o3derrboundA * permanent;
 
2290
  if ((det > errbound) || (-det > errbound)) {
 
2291
    return det;
 
2292
  }
 
2293
 
 
2294
  return orient3dadapt(pa, pb, pc, pd, permanent);
 
2295
}
 
2296
 
 
2297
/*****************************************************************************/
 
2298
/*                                                                           */
 
2299
/*  incirclefast()   Approximate 2D incircle test.  Nonrobust.               */
 
2300
/*  incircleexact()   Exact 2D incircle test.  Robust.                       */
 
2301
/*  incircleslow()   Another exact 2D incircle test.  Robust.                */
 
2302
/*  incircle()   Adaptive exact 2D incircle test.  Robust.                   */
 
2303
/*                                                                           */
 
2304
/*               Return a positive value if the point pd lies inside the     */
 
2305
/*               circle passing through pa, pb, and pc; a negative value if  */
 
2306
/*               it lies outside; and zero if the four points are cocircular.*/
 
2307
/*               The points pa, pb, and pc must be in counterclockwise       */
 
2308
/*               order, or the sign of the result will be reversed.          */
 
2309
/*                                                                           */
 
2310
/*  Only the first and last routine should be used; the middle two are for   */
 
2311
/*  timings.                                                                 */
 
2312
/*                                                                           */
 
2313
/*  The last three use exact arithmetic to ensure a correct answer.  The     */
 
2314
/*  result returned is the determinant of a matrix.  In incircle() only,     */
 
2315
/*  this determinant is computed adaptively, in the sense that exact         */
 
2316
/*  arithmetic is used only to the degree it is needed to ensure that the    */
 
2317
/*  returned value has the correct sign.  Hence, incircle() is usually quite */
 
2318
/*  fast, but will run more slowly when the input points are cocircular or   */
 
2319
/*  nearly so.                                                               */
 
2320
/*                                                                           */
 
2321
/*****************************************************************************/
 
2322
 
 
2323
REAL incirclefast(REAL *pa, REAL *pb, REAL *pc, REAL *pd)
 
2324
{
 
2325
  REAL adx, ady, bdx, bdy, cdx, cdy;
 
2326
  REAL abdet, bcdet, cadet;
 
2327
  REAL alift, blift, clift;
 
2328
 
 
2329
  adx = pa[0] - pd[0];
 
2330
  ady = pa[1] - pd[1];
 
2331
  bdx = pb[0] - pd[0];
 
2332
  bdy = pb[1] - pd[1];
 
2333
  cdx = pc[0] - pd[0];
 
2334
  cdy = pc[1] - pd[1];
 
2335
 
 
2336
  abdet = adx * bdy - bdx * ady;
 
2337
  bcdet = bdx * cdy - cdx * bdy;
 
2338
  cadet = cdx * ady - adx * cdy;
 
2339
  alift = adx * adx + ady * ady;
 
2340
  blift = bdx * bdx + bdy * bdy;
 
2341
  clift = cdx * cdx + cdy * cdy;
 
2342
 
 
2343
  return alift * bcdet + blift * cadet + clift * abdet;
 
2344
}
 
2345
 
 
2346
REAL incircleexact(REAL *pa, REAL *pb, REAL *pc, REAL *pd)
 
2347
{
 
2348
  INEXACT REAL axby1, bxcy1, cxdy1, dxay1, axcy1, bxdy1;
 
2349
  INEXACT REAL bxay1, cxby1, dxcy1, axdy1, cxay1, dxby1;
 
2350
  REAL axby0, bxcy0, cxdy0, dxay0, axcy0, bxdy0;
 
2351
  REAL bxay0, cxby0, dxcy0, axdy0, cxay0, dxby0;
 
2352
  REAL ab[4], bc[4], cd[4], da[4], ac[4], bd[4];
 
2353
  REAL temp8[8];
 
2354
  int templen;
 
2355
  REAL abc[12], bcd[12], cda[12], dab[12];
 
2356
  int abclen, bcdlen, cdalen, dablen;
 
2357
  REAL det24x[24], det24y[24], det48x[48], det48y[48];
 
2358
  int xlen, ylen;
 
2359
  REAL adet[96], bdet[96], cdet[96], ddet[96];
 
2360
  int alen, blen, clen, dlen;
 
2361
  REAL abdet[192], cddet[192];
 
2362
  int ablen, cdlen;
 
2363
  REAL deter[384];
 
2364
  int deterlen;
 
2365
  int i;
 
2366
 
 
2367
  INEXACT REAL bvirt;
 
2368
  REAL avirt, bround, around;
 
2369
  INEXACT REAL c;
 
2370
  INEXACT REAL abig;
 
2371
  REAL ahi, alo, bhi, blo;
 
2372
  REAL err1, err2, err3;
 
2373
  INEXACT REAL _i, _j;
 
2374
  REAL _0;
 
2375
 
 
2376
  Two_Product(pa[0], pb[1], axby1, axby0);
 
2377
  Two_Product(pb[0], pa[1], bxay1, bxay0);
 
2378
  Two_Two_Diff(axby1, axby0, bxay1, bxay0, ab[3], ab[2], ab[1], ab[0]);
 
2379
 
 
2380
  Two_Product(pb[0], pc[1], bxcy1, bxcy0);
 
2381
  Two_Product(pc[0], pb[1], cxby1, cxby0);
 
2382
  Two_Two_Diff(bxcy1, bxcy0, cxby1, cxby0, bc[3], bc[2], bc[1], bc[0]);
 
2383
 
 
2384
  Two_Product(pc[0], pd[1], cxdy1, cxdy0);
 
2385
  Two_Product(pd[0], pc[1], dxcy1, dxcy0);
 
2386
  Two_Two_Diff(cxdy1, cxdy0, dxcy1, dxcy0, cd[3], cd[2], cd[1], cd[0]);
 
2387
 
 
2388
  Two_Product(pd[0], pa[1], dxay1, dxay0);
 
2389
  Two_Product(pa[0], pd[1], axdy1, axdy0);
 
2390
  Two_Two_Diff(dxay1, dxay0, axdy1, axdy0, da[3], da[2], da[1], da[0]);
 
2391
 
 
2392
  Two_Product(pa[0], pc[1], axcy1, axcy0);
 
2393
  Two_Product(pc[0], pa[1], cxay1, cxay0);
 
2394
  Two_Two_Diff(axcy1, axcy0, cxay1, cxay0, ac[3], ac[2], ac[1], ac[0]);
 
2395
 
 
2396
  Two_Product(pb[0], pd[1], bxdy1, bxdy0);
 
2397
  Two_Product(pd[0], pb[1], dxby1, dxby0);
 
2398
  Two_Two_Diff(bxdy1, bxdy0, dxby1, dxby0, bd[3], bd[2], bd[1], bd[0]);
 
2399
 
 
2400
  templen = fast_expansion_sum_zeroelim(4, cd, 4, da, temp8);
 
2401
  cdalen = fast_expansion_sum_zeroelim(templen, temp8, 4, ac, cda);
 
2402
  templen = fast_expansion_sum_zeroelim(4, da, 4, ab, temp8);
 
2403
  dablen = fast_expansion_sum_zeroelim(templen, temp8, 4, bd, dab);
 
2404
  for (i = 0; i < 4; i++) {
 
2405
    bd[i] = -bd[i];
 
2406
    ac[i] = -ac[i];
 
2407
  }
 
2408
  templen = fast_expansion_sum_zeroelim(4, ab, 4, bc, temp8);
 
2409
  abclen = fast_expansion_sum_zeroelim(templen, temp8, 4, ac, abc);
 
2410
  templen = fast_expansion_sum_zeroelim(4, bc, 4, cd, temp8);
 
2411
  bcdlen = fast_expansion_sum_zeroelim(templen, temp8, 4, bd, bcd);
 
2412
 
 
2413
  xlen = scale_expansion_zeroelim(bcdlen, bcd, pa[0], det24x);
 
2414
  xlen = scale_expansion_zeroelim(xlen, det24x, pa[0], det48x);
 
2415
  ylen = scale_expansion_zeroelim(bcdlen, bcd, pa[1], det24y);
 
2416
  ylen = scale_expansion_zeroelim(ylen, det24y, pa[1], det48y);
 
2417
  alen = fast_expansion_sum_zeroelim(xlen, det48x, ylen, det48y, adet);
 
2418
 
 
2419
  xlen = scale_expansion_zeroelim(cdalen, cda, pb[0], det24x);
 
2420
  xlen = scale_expansion_zeroelim(xlen, det24x, -pb[0], det48x);
 
2421
  ylen = scale_expansion_zeroelim(cdalen, cda, pb[1], det24y);
 
2422
  ylen = scale_expansion_zeroelim(ylen, det24y, -pb[1], det48y);
 
2423
  blen = fast_expansion_sum_zeroelim(xlen, det48x, ylen, det48y, bdet);
 
2424
 
 
2425
  xlen = scale_expansion_zeroelim(dablen, dab, pc[0], det24x);
 
2426
  xlen = scale_expansion_zeroelim(xlen, det24x, pc[0], det48x);
 
2427
  ylen = scale_expansion_zeroelim(dablen, dab, pc[1], det24y);
 
2428
  ylen = scale_expansion_zeroelim(ylen, det24y, pc[1], det48y);
 
2429
  clen = fast_expansion_sum_zeroelim(xlen, det48x, ylen, det48y, cdet);
 
2430
 
 
2431
  xlen = scale_expansion_zeroelim(abclen, abc, pd[0], det24x);
 
2432
  xlen = scale_expansion_zeroelim(xlen, det24x, -pd[0], det48x);
 
2433
  ylen = scale_expansion_zeroelim(abclen, abc, pd[1], det24y);
 
2434
  ylen = scale_expansion_zeroelim(ylen, det24y, -pd[1], det48y);
 
2435
  dlen = fast_expansion_sum_zeroelim(xlen, det48x, ylen, det48y, ddet);
 
2436
 
 
2437
  ablen = fast_expansion_sum_zeroelim(alen, adet, blen, bdet, abdet);
 
2438
  cdlen = fast_expansion_sum_zeroelim(clen, cdet, dlen, ddet, cddet);
 
2439
  deterlen = fast_expansion_sum_zeroelim(ablen, abdet, cdlen, cddet, deter);
 
2440
 
 
2441
  return deter[deterlen - 1];
 
2442
}
 
2443
 
 
2444
REAL incircleslow(REAL *pa, REAL *pb, REAL *pc, REAL *pd)
 
2445
{
 
2446
  INEXACT REAL adx, bdx, cdx, ady, bdy, cdy;
 
2447
  REAL adxtail, bdxtail, cdxtail;
 
2448
  REAL adytail, bdytail, cdytail;
 
2449
  REAL negate, negatetail;
 
2450
  INEXACT REAL axby7, bxcy7, axcy7, bxay7, cxby7, cxay7;
 
2451
  REAL axby[8], bxcy[8], axcy[8], bxay[8], cxby[8], cxay[8];
 
2452
  REAL temp16[16];
 
2453
  int temp16len;
 
2454
  REAL detx[32], detxx[64], detxt[32], detxxt[64], detxtxt[64];
 
2455
  int xlen, xxlen, xtlen, xxtlen, xtxtlen;
 
2456
  REAL x1[128], x2[192];
 
2457
  int x1len, x2len;
 
2458
  REAL dety[32], detyy[64], detyt[32], detyyt[64], detytyt[64];
 
2459
  int ylen, yylen, ytlen, yytlen, ytytlen;
 
2460
  REAL y1[128], y2[192];
 
2461
  int y1len, y2len;
 
2462
  REAL adet[384], bdet[384], cdet[384], abdet[768], deter[1152];
 
2463
  int alen, blen, clen, ablen, deterlen;
 
2464
  int i;
 
2465
 
 
2466
  INEXACT REAL bvirt;
 
2467
  REAL avirt, bround, around;
 
2468
  INEXACT REAL c;
 
2469
  INEXACT REAL abig;
 
2470
  REAL a0hi, a0lo, a1hi, a1lo, bhi, blo;
 
2471
  REAL err1, err2, err3;
 
2472
  INEXACT REAL _i, _j, _k, _l, _m, _n;
 
2473
  REAL _0, _1, _2;
 
2474
 
 
2475
  Two_Diff(pa[0], pd[0], adx, adxtail);
 
2476
  Two_Diff(pa[1], pd[1], ady, adytail);
 
2477
  Two_Diff(pb[0], pd[0], bdx, bdxtail);
 
2478
  Two_Diff(pb[1], pd[1], bdy, bdytail);
 
2479
  Two_Diff(pc[0], pd[0], cdx, cdxtail);
 
2480
  Two_Diff(pc[1], pd[1], cdy, cdytail);
 
2481
 
 
2482
  Two_Two_Product(adx, adxtail, bdy, bdytail,
 
2483
                  axby7, axby[6], axby[5], axby[4],
 
2484
                  axby[3], axby[2], axby[1], axby[0]);
 
2485
  axby[7] = axby7;
 
2486
  negate = -ady;
 
2487
  negatetail = -adytail;
 
2488
  Two_Two_Product(bdx, bdxtail, negate, negatetail,
 
2489
                  bxay7, bxay[6], bxay[5], bxay[4],
 
2490
                  bxay[3], bxay[2], bxay[1], bxay[0]);
 
2491
  bxay[7] = bxay7;
 
2492
  Two_Two_Product(bdx, bdxtail, cdy, cdytail,
 
2493
                  bxcy7, bxcy[6], bxcy[5], bxcy[4],
 
2494
                  bxcy[3], bxcy[2], bxcy[1], bxcy[0]);
 
2495
  bxcy[7] = bxcy7;
 
2496
  negate = -bdy;
 
2497
  negatetail = -bdytail;
 
2498
  Two_Two_Product(cdx, cdxtail, negate, negatetail,
 
2499
                  cxby7, cxby[6], cxby[5], cxby[4],
 
2500
                  cxby[3], cxby[2], cxby[1], cxby[0]);
 
2501
  cxby[7] = cxby7;
 
2502
  Two_Two_Product(cdx, cdxtail, ady, adytail,
 
2503
                  cxay7, cxay[6], cxay[5], cxay[4],
 
2504
                  cxay[3], cxay[2], cxay[1], cxay[0]);
 
2505
  cxay[7] = cxay7;
 
2506
  negate = -cdy;
 
2507
  negatetail = -cdytail;
 
2508
  Two_Two_Product(adx, adxtail, negate, negatetail,
 
2509
                  axcy7, axcy[6], axcy[5], axcy[4],
 
2510
                  axcy[3], axcy[2], axcy[1], axcy[0]);
 
2511
  axcy[7] = axcy7;
 
2512
 
 
2513
 
 
2514
  temp16len = fast_expansion_sum_zeroelim(8, bxcy, 8, cxby, temp16);
 
2515
 
 
2516
  xlen = scale_expansion_zeroelim(temp16len, temp16, adx, detx);
 
2517
  xxlen = scale_expansion_zeroelim(xlen, detx, adx, detxx);
 
2518
  xtlen = scale_expansion_zeroelim(temp16len, temp16, adxtail, detxt);
 
2519
  xxtlen = scale_expansion_zeroelim(xtlen, detxt, adx, detxxt);
 
2520
  for (i = 0; i < xxtlen; i++) {
 
2521
    detxxt[i] *= 2.0;
 
2522
  }
 
2523
  xtxtlen = scale_expansion_zeroelim(xtlen, detxt, adxtail, detxtxt);
 
2524
  x1len = fast_expansion_sum_zeroelim(xxlen, detxx, xxtlen, detxxt, x1);
 
2525
  x2len = fast_expansion_sum_zeroelim(x1len, x1, xtxtlen, detxtxt, x2);
 
2526
 
 
2527
  ylen = scale_expansion_zeroelim(temp16len, temp16, ady, dety);
 
2528
  yylen = scale_expansion_zeroelim(ylen, dety, ady, detyy);
 
2529
  ytlen = scale_expansion_zeroelim(temp16len, temp16, adytail, detyt);
 
2530
  yytlen = scale_expansion_zeroelim(ytlen, detyt, ady, detyyt);
 
2531
  for (i = 0; i < yytlen; i++) {
 
2532
    detyyt[i] *= 2.0;
 
2533
  }
 
2534
  ytytlen = scale_expansion_zeroelim(ytlen, detyt, adytail, detytyt);
 
2535
  y1len = fast_expansion_sum_zeroelim(yylen, detyy, yytlen, detyyt, y1);
 
2536
  y2len = fast_expansion_sum_zeroelim(y1len, y1, ytytlen, detytyt, y2);
 
2537
 
 
2538
  alen = fast_expansion_sum_zeroelim(x2len, x2, y2len, y2, adet);
 
2539
 
 
2540
 
 
2541
  temp16len = fast_expansion_sum_zeroelim(8, cxay, 8, axcy, temp16);
 
2542
 
 
2543
  xlen = scale_expansion_zeroelim(temp16len, temp16, bdx, detx);
 
2544
  xxlen = scale_expansion_zeroelim(xlen, detx, bdx, detxx);
 
2545
  xtlen = scale_expansion_zeroelim(temp16len, temp16, bdxtail, detxt);
 
2546
  xxtlen = scale_expansion_zeroelim(xtlen, detxt, bdx, detxxt);
 
2547
  for (i = 0; i < xxtlen; i++) {
 
2548
    detxxt[i] *= 2.0;
 
2549
  }
 
2550
  xtxtlen = scale_expansion_zeroelim(xtlen, detxt, bdxtail, detxtxt);
 
2551
  x1len = fast_expansion_sum_zeroelim(xxlen, detxx, xxtlen, detxxt, x1);
 
2552
  x2len = fast_expansion_sum_zeroelim(x1len, x1, xtxtlen, detxtxt, x2);
 
2553
 
 
2554
  ylen = scale_expansion_zeroelim(temp16len, temp16, bdy, dety);
 
2555
  yylen = scale_expansion_zeroelim(ylen, dety, bdy, detyy);
 
2556
  ytlen = scale_expansion_zeroelim(temp16len, temp16, bdytail, detyt);
 
2557
  yytlen = scale_expansion_zeroelim(ytlen, detyt, bdy, detyyt);
 
2558
  for (i = 0; i < yytlen; i++) {
 
2559
    detyyt[i] *= 2.0;
 
2560
  }
 
2561
  ytytlen = scale_expansion_zeroelim(ytlen, detyt, bdytail, detytyt);
 
2562
  y1len = fast_expansion_sum_zeroelim(yylen, detyy, yytlen, detyyt, y1);
 
2563
  y2len = fast_expansion_sum_zeroelim(y1len, y1, ytytlen, detytyt, y2);
 
2564
 
 
2565
  blen = fast_expansion_sum_zeroelim(x2len, x2, y2len, y2, bdet);
 
2566
 
 
2567
 
 
2568
  temp16len = fast_expansion_sum_zeroelim(8, axby, 8, bxay, temp16);
 
2569
 
 
2570
  xlen = scale_expansion_zeroelim(temp16len, temp16, cdx, detx);
 
2571
  xxlen = scale_expansion_zeroelim(xlen, detx, cdx, detxx);
 
2572
  xtlen = scale_expansion_zeroelim(temp16len, temp16, cdxtail, detxt);
 
2573
  xxtlen = scale_expansion_zeroelim(xtlen, detxt, cdx, detxxt);
 
2574
  for (i = 0; i < xxtlen; i++) {
 
2575
    detxxt[i] *= 2.0;
 
2576
  }
 
2577
  xtxtlen = scale_expansion_zeroelim(xtlen, detxt, cdxtail, detxtxt);
 
2578
  x1len = fast_expansion_sum_zeroelim(xxlen, detxx, xxtlen, detxxt, x1);
 
2579
  x2len = fast_expansion_sum_zeroelim(x1len, x1, xtxtlen, detxtxt, x2);
 
2580
 
 
2581
  ylen = scale_expansion_zeroelim(temp16len, temp16, cdy, dety);
 
2582
  yylen = scale_expansion_zeroelim(ylen, dety, cdy, detyy);
 
2583
  ytlen = scale_expansion_zeroelim(temp16len, temp16, cdytail, detyt);
 
2584
  yytlen = scale_expansion_zeroelim(ytlen, detyt, cdy, detyyt);
 
2585
  for (i = 0; i < yytlen; i++) {
 
2586
    detyyt[i] *= 2.0;
 
2587
  }
 
2588
  ytytlen = scale_expansion_zeroelim(ytlen, detyt, cdytail, detytyt);
 
2589
  y1len = fast_expansion_sum_zeroelim(yylen, detyy, yytlen, detyyt, y1);
 
2590
  y2len = fast_expansion_sum_zeroelim(y1len, y1, ytytlen, detytyt, y2);
 
2591
 
 
2592
  clen = fast_expansion_sum_zeroelim(x2len, x2, y2len, y2, cdet);
 
2593
 
 
2594
  ablen = fast_expansion_sum_zeroelim(alen, adet, blen, bdet, abdet);
 
2595
  deterlen = fast_expansion_sum_zeroelim(ablen, abdet, clen, cdet, deter);
 
2596
 
 
2597
  return deter[deterlen - 1];
 
2598
}
 
2599
 
 
2600
REAL incircleadapt(REAL *pa, REAL *pb, REAL *pc, REAL *pd, REAL permanent)
 
2601
{
 
2602
  INEXACT REAL adx, bdx, cdx, ady, bdy, cdy;
 
2603
  REAL det, errbound;
 
2604
 
 
2605
  INEXACT REAL bdxcdy1, cdxbdy1, cdxady1, adxcdy1, adxbdy1, bdxady1;
 
2606
  REAL bdxcdy0, cdxbdy0, cdxady0, adxcdy0, adxbdy0, bdxady0;
 
2607
  REAL bc[4], ca[4], ab[4];
 
2608
  INEXACT REAL bc3, ca3, ab3;
 
2609
  REAL axbc[8], axxbc[16], aybc[8], ayybc[16], adet[32];
 
2610
  int axbclen, axxbclen, aybclen, ayybclen, alen;
 
2611
  REAL bxca[8], bxxca[16], byca[8], byyca[16], bdet[32];
 
2612
  int bxcalen, bxxcalen, bycalen, byycalen, blen;
 
2613
  REAL cxab[8], cxxab[16], cyab[8], cyyab[16], cdet[32];
 
2614
  int cxablen, cxxablen, cyablen, cyyablen, clen;
 
2615
  REAL abdet[64];
 
2616
  int ablen;
 
2617
  REAL fin1[1152], fin2[1152];
 
2618
  REAL *finnow, *finother, *finswap;
 
2619
  int finlength;
 
2620
 
 
2621
  REAL adxtail, bdxtail, cdxtail, adytail, bdytail, cdytail;
 
2622
  INEXACT REAL adxadx1, adyady1, bdxbdx1, bdybdy1, cdxcdx1, cdycdy1;
 
2623
  REAL adxadx0, adyady0, bdxbdx0, bdybdy0, cdxcdx0, cdycdy0;
 
2624
  REAL aa[4], bb[4], cc[4];
 
2625
  INEXACT REAL aa3, bb3, cc3;
 
2626
  INEXACT REAL ti1, tj1;
 
2627
  REAL ti0, tj0;
 
2628
  REAL u[4], v[4];
 
2629
  INEXACT REAL u3, v3;
 
2630
  REAL temp8[8], temp16a[16], temp16b[16], temp16c[16];
 
2631
  REAL temp32a[32], temp32b[32], temp48[48], temp64[64];
 
2632
  int temp8len, temp16alen, temp16blen, temp16clen;
 
2633
  int temp32alen, temp32blen, temp48len, temp64len;
 
2634
  REAL axtbb[8], axtcc[8], aytbb[8], aytcc[8];
 
2635
  int axtbblen, axtcclen, aytbblen, aytcclen;
 
2636
  REAL bxtaa[8], bxtcc[8], bytaa[8], bytcc[8];
 
2637
  int bxtaalen, bxtcclen, bytaalen, bytcclen;
 
2638
  REAL cxtaa[8], cxtbb[8], cytaa[8], cytbb[8];
 
2639
  int cxtaalen, cxtbblen, cytaalen, cytbblen;
 
2640
  REAL axtbc[8], aytbc[8], bxtca[8], bytca[8], cxtab[8], cytab[8];
 
2641
  int axtbclen=0, aytbclen=0, bxtcalen=0, bytcalen=0, cxtablen=0, cytablen=0;
 
2642
  REAL axtbct[16], aytbct[16], bxtcat[16], bytcat[16], cxtabt[16], cytabt[16];
 
2643
  int axtbctlen, aytbctlen, bxtcatlen, bytcatlen, cxtabtlen, cytabtlen;
 
2644
  REAL axtbctt[8], aytbctt[8], bxtcatt[8];
 
2645
  REAL bytcatt[8], cxtabtt[8], cytabtt[8];
 
2646
  int axtbcttlen, aytbcttlen, bxtcattlen, bytcattlen, cxtabttlen, cytabttlen;
 
2647
  REAL abt[8], bct[8], cat[8];
 
2648
  int abtlen, bctlen, catlen;
 
2649
  REAL abtt[4], bctt[4], catt[4];
 
2650
  int abttlen, bcttlen, cattlen;
 
2651
  INEXACT REAL abtt3, bctt3, catt3;
 
2652
  REAL negate;
 
2653
 
 
2654
  INEXACT REAL bvirt;
 
2655
  REAL avirt, bround, around;
 
2656
  INEXACT REAL c;
 
2657
  INEXACT REAL abig;
 
2658
  REAL ahi, alo, bhi, blo;
 
2659
  REAL err1, err2, err3;
 
2660
  INEXACT REAL _i, _j;
 
2661
  REAL _0;
 
2662
 
 
2663
  adx = (REAL) (pa[0] - pd[0]);
 
2664
  bdx = (REAL) (pb[0] - pd[0]);
 
2665
  cdx = (REAL) (pc[0] - pd[0]);
 
2666
  ady = (REAL) (pa[1] - pd[1]);
 
2667
  bdy = (REAL) (pb[1] - pd[1]);
 
2668
  cdy = (REAL) (pc[1] - pd[1]);
 
2669
 
 
2670
  Two_Product(bdx, cdy, bdxcdy1, bdxcdy0);
 
2671
  Two_Product(cdx, bdy, cdxbdy1, cdxbdy0);
 
2672
  Two_Two_Diff(bdxcdy1, bdxcdy0, cdxbdy1, cdxbdy0, bc3, bc[2], bc[1], bc[0]);
 
2673
  bc[3] = bc3;
 
2674
  axbclen = scale_expansion_zeroelim(4, bc, adx, axbc);
 
2675
  axxbclen = scale_expansion_zeroelim(axbclen, axbc, adx, axxbc);
 
2676
  aybclen = scale_expansion_zeroelim(4, bc, ady, aybc);
 
2677
  ayybclen = scale_expansion_zeroelim(aybclen, aybc, ady, ayybc);
 
2678
  alen = fast_expansion_sum_zeroelim(axxbclen, axxbc, ayybclen, ayybc, adet);
 
2679
 
 
2680
  Two_Product(cdx, ady, cdxady1, cdxady0);
 
2681
  Two_Product(adx, cdy, adxcdy1, adxcdy0);
 
2682
  Two_Two_Diff(cdxady1, cdxady0, adxcdy1, adxcdy0, ca3, ca[2], ca[1], ca[0]);
 
2683
  ca[3] = ca3;
 
2684
  bxcalen = scale_expansion_zeroelim(4, ca, bdx, bxca);
 
2685
  bxxcalen = scale_expansion_zeroelim(bxcalen, bxca, bdx, bxxca);
 
2686
  bycalen = scale_expansion_zeroelim(4, ca, bdy, byca);
 
2687
  byycalen = scale_expansion_zeroelim(bycalen, byca, bdy, byyca);
 
2688
  blen = fast_expansion_sum_zeroelim(bxxcalen, bxxca, byycalen, byyca, bdet);
 
2689
 
 
2690
  Two_Product(adx, bdy, adxbdy1, adxbdy0);
 
2691
  Two_Product(bdx, ady, bdxady1, bdxady0);
 
2692
  Two_Two_Diff(adxbdy1, adxbdy0, bdxady1, bdxady0, ab3, ab[2], ab[1], ab[0]);
 
2693
  ab[3] = ab3;
 
2694
  cxablen = scale_expansion_zeroelim(4, ab, cdx, cxab);
 
2695
  cxxablen = scale_expansion_zeroelim(cxablen, cxab, cdx, cxxab);
 
2696
  cyablen = scale_expansion_zeroelim(4, ab, cdy, cyab);
 
2697
  cyyablen = scale_expansion_zeroelim(cyablen, cyab, cdy, cyyab);
 
2698
  clen = fast_expansion_sum_zeroelim(cxxablen, cxxab, cyyablen, cyyab, cdet);
 
2699
 
 
2700
  ablen = fast_expansion_sum_zeroelim(alen, adet, blen, bdet, abdet);
 
2701
  finlength = fast_expansion_sum_zeroelim(ablen, abdet, clen, cdet, fin1);
 
2702
 
 
2703
  det = estimate(finlength, fin1);
 
2704
  errbound = iccerrboundB * permanent;
 
2705
  if ((det >= errbound) || (-det >= errbound)) {
 
2706
    return det;
 
2707
  }
 
2708
 
 
2709
  Two_Diff_Tail(pa[0], pd[0], adx, adxtail);
 
2710
  Two_Diff_Tail(pa[1], pd[1], ady, adytail);
 
2711
  Two_Diff_Tail(pb[0], pd[0], bdx, bdxtail);
 
2712
  Two_Diff_Tail(pb[1], pd[1], bdy, bdytail);
 
2713
  Two_Diff_Tail(pc[0], pd[0], cdx, cdxtail);
 
2714
  Two_Diff_Tail(pc[1], pd[1], cdy, cdytail);
 
2715
  if ((adxtail == 0.0) && (bdxtail == 0.0) && (cdxtail == 0.0)
 
2716
      && (adytail == 0.0) && (bdytail == 0.0) && (cdytail == 0.0)) {
 
2717
    return det;
 
2718
  }
 
2719
 
 
2720
  errbound = iccerrboundC * permanent + resulterrbound * Absolute(det);
 
2721
  det += ((adx * adx + ady * ady) * ((bdx * cdytail + cdy * bdxtail)
 
2722
                                     - (bdy * cdxtail + cdx * bdytail))
 
2723
          + 2.0 * (adx * adxtail + ady * adytail) * (bdx * cdy - bdy * cdx))
 
2724
       + ((bdx * bdx + bdy * bdy) * ((cdx * adytail + ady * cdxtail)
 
2725
                                     - (cdy * adxtail + adx * cdytail))
 
2726
          + 2.0 * (bdx * bdxtail + bdy * bdytail) * (cdx * ady - cdy * adx))
 
2727
       + ((cdx * cdx + cdy * cdy) * ((adx * bdytail + bdy * adxtail)
 
2728
                                     - (ady * bdxtail + bdx * adytail))
 
2729
          + 2.0 * (cdx * cdxtail + cdy * cdytail) * (adx * bdy - ady * bdx));
 
2730
  if ((det >= errbound) || (-det >= errbound)) {
 
2731
    return det;
 
2732
  }
 
2733
 
 
2734
  finnow = fin1;
 
2735
  finother = fin2;
 
2736
 
 
2737
  if ((bdxtail != 0.0) || (bdytail != 0.0)
 
2738
      || (cdxtail != 0.0) || (cdytail != 0.0)) {
 
2739
    Square(adx, adxadx1, adxadx0);
 
2740
    Square(ady, adyady1, adyady0);
 
2741
    Two_Two_Sum(adxadx1, adxadx0, adyady1, adyady0, aa3, aa[2], aa[1], aa[0]);
 
2742
    aa[3] = aa3;
 
2743
  }
 
2744
  if ((cdxtail != 0.0) || (cdytail != 0.0)
 
2745
      || (adxtail != 0.0) || (adytail != 0.0)) {
 
2746
    Square(bdx, bdxbdx1, bdxbdx0);
 
2747
    Square(bdy, bdybdy1, bdybdy0);
 
2748
    Two_Two_Sum(bdxbdx1, bdxbdx0, bdybdy1, bdybdy0, bb3, bb[2], bb[1], bb[0]);
 
2749
    bb[3] = bb3;
 
2750
  }
 
2751
  if ((adxtail != 0.0) || (adytail != 0.0)
 
2752
      || (bdxtail != 0.0) || (bdytail != 0.0)) {
 
2753
    Square(cdx, cdxcdx1, cdxcdx0);
 
2754
    Square(cdy, cdycdy1, cdycdy0);
 
2755
    Two_Two_Sum(cdxcdx1, cdxcdx0, cdycdy1, cdycdy0, cc3, cc[2], cc[1], cc[0]);
 
2756
    cc[3] = cc3;
 
2757
  }
 
2758
 
 
2759
  if (adxtail != 0.0) {
 
2760
    axtbclen = scale_expansion_zeroelim(4, bc, adxtail, axtbc);
 
2761
    temp16alen = scale_expansion_zeroelim(axtbclen, axtbc, 2.0 * adx,
 
2762
                                          temp16a);
 
2763
 
 
2764
    axtcclen = scale_expansion_zeroelim(4, cc, adxtail, axtcc);
 
2765
    temp16blen = scale_expansion_zeroelim(axtcclen, axtcc, bdy, temp16b);
 
2766
 
 
2767
    axtbblen = scale_expansion_zeroelim(4, bb, adxtail, axtbb);
 
2768
    temp16clen = scale_expansion_zeroelim(axtbblen, axtbb, -cdy, temp16c);
 
2769
 
 
2770
    temp32alen = fast_expansion_sum_zeroelim(temp16alen, temp16a,
 
2771
                                            temp16blen, temp16b, temp32a);
 
2772
    temp48len = fast_expansion_sum_zeroelim(temp16clen, temp16c,
 
2773
                                            temp32alen, temp32a, temp48);
 
2774
    finlength = fast_expansion_sum_zeroelim(finlength, finnow, temp48len,
 
2775
                                            temp48, finother);
 
2776
    finswap = finnow; finnow = finother; finother = finswap;
 
2777
  }
 
2778
  if (adytail != 0.0) {
 
2779
    aytbclen = scale_expansion_zeroelim(4, bc, adytail, aytbc);
 
2780
    temp16alen = scale_expansion_zeroelim(aytbclen, aytbc, 2.0 * ady,
 
2781
                                          temp16a);
 
2782
 
 
2783
    aytbblen = scale_expansion_zeroelim(4, bb, adytail, aytbb);
 
2784
    temp16blen = scale_expansion_zeroelim(aytbblen, aytbb, cdx, temp16b);
 
2785
 
 
2786
    aytcclen = scale_expansion_zeroelim(4, cc, adytail, aytcc);
 
2787
    temp16clen = scale_expansion_zeroelim(aytcclen, aytcc, -bdx, temp16c);
 
2788
 
 
2789
    temp32alen = fast_expansion_sum_zeroelim(temp16alen, temp16a,
 
2790
                                            temp16blen, temp16b, temp32a);
 
2791
    temp48len = fast_expansion_sum_zeroelim(temp16clen, temp16c,
 
2792
                                            temp32alen, temp32a, temp48);
 
2793
    finlength = fast_expansion_sum_zeroelim(finlength, finnow, temp48len,
 
2794
                                            temp48, finother);
 
2795
    finswap = finnow; finnow = finother; finother = finswap;
 
2796
  }
 
2797
  if (bdxtail != 0.0) {
 
2798
    bxtcalen = scale_expansion_zeroelim(4, ca, bdxtail, bxtca);
 
2799
    temp16alen = scale_expansion_zeroelim(bxtcalen, bxtca, 2.0 * bdx,
 
2800
                                          temp16a);
 
2801
 
 
2802
    bxtaalen = scale_expansion_zeroelim(4, aa, bdxtail, bxtaa);
 
2803
    temp16blen = scale_expansion_zeroelim(bxtaalen, bxtaa, cdy, temp16b);
 
2804
 
 
2805
    bxtcclen = scale_expansion_zeroelim(4, cc, bdxtail, bxtcc);
 
2806
    temp16clen = scale_expansion_zeroelim(bxtcclen, bxtcc, -ady, temp16c);
 
2807
 
 
2808
    temp32alen = fast_expansion_sum_zeroelim(temp16alen, temp16a,
 
2809
                                            temp16blen, temp16b, temp32a);
 
2810
    temp48len = fast_expansion_sum_zeroelim(temp16clen, temp16c,
 
2811
                                            temp32alen, temp32a, temp48);
 
2812
    finlength = fast_expansion_sum_zeroelim(finlength, finnow, temp48len,
 
2813
                                            temp48, finother);
 
2814
    finswap = finnow; finnow = finother; finother = finswap;
 
2815
  }
 
2816
  if (bdytail != 0.0) {
 
2817
    bytcalen = scale_expansion_zeroelim(4, ca, bdytail, bytca);
 
2818
    temp16alen = scale_expansion_zeroelim(bytcalen, bytca, 2.0 * bdy,
 
2819
                                          temp16a);
 
2820
 
 
2821
    bytcclen = scale_expansion_zeroelim(4, cc, bdytail, bytcc);
 
2822
    temp16blen = scale_expansion_zeroelim(bytcclen, bytcc, adx, temp16b);
 
2823
 
 
2824
    bytaalen = scale_expansion_zeroelim(4, aa, bdytail, bytaa);
 
2825
    temp16clen = scale_expansion_zeroelim(bytaalen, bytaa, -cdx, temp16c);
 
2826
 
 
2827
    temp32alen = fast_expansion_sum_zeroelim(temp16alen, temp16a,
 
2828
                                            temp16blen, temp16b, temp32a);
 
2829
    temp48len = fast_expansion_sum_zeroelim(temp16clen, temp16c,
 
2830
                                            temp32alen, temp32a, temp48);
 
2831
    finlength = fast_expansion_sum_zeroelim(finlength, finnow, temp48len,
 
2832
                                            temp48, finother);
 
2833
    finswap = finnow; finnow = finother; finother = finswap;
 
2834
  }
 
2835
  if (cdxtail != 0.0) {
 
2836
    cxtablen = scale_expansion_zeroelim(4, ab, cdxtail, cxtab);
 
2837
    temp16alen = scale_expansion_zeroelim(cxtablen, cxtab, 2.0 * cdx,
 
2838
                                          temp16a);
 
2839
 
 
2840
    cxtbblen = scale_expansion_zeroelim(4, bb, cdxtail, cxtbb);
 
2841
    temp16blen = scale_expansion_zeroelim(cxtbblen, cxtbb, ady, temp16b);
 
2842
 
 
2843
    cxtaalen = scale_expansion_zeroelim(4, aa, cdxtail, cxtaa);
 
2844
    temp16clen = scale_expansion_zeroelim(cxtaalen, cxtaa, -bdy, temp16c);
 
2845
 
 
2846
    temp32alen = fast_expansion_sum_zeroelim(temp16alen, temp16a,
 
2847
                                            temp16blen, temp16b, temp32a);
 
2848
    temp48len = fast_expansion_sum_zeroelim(temp16clen, temp16c,
 
2849
                                            temp32alen, temp32a, temp48);
 
2850
    finlength = fast_expansion_sum_zeroelim(finlength, finnow, temp48len,
 
2851
                                            temp48, finother);
 
2852
    finswap = finnow; finnow = finother; finother = finswap;
 
2853
  }
 
2854
  if (cdytail != 0.0) {
 
2855
    cytablen = scale_expansion_zeroelim(4, ab, cdytail, cytab);
 
2856
    temp16alen = scale_expansion_zeroelim(cytablen, cytab, 2.0 * cdy,
 
2857
                                          temp16a);
 
2858
 
 
2859
    cytaalen = scale_expansion_zeroelim(4, aa, cdytail, cytaa);
 
2860
    temp16blen = scale_expansion_zeroelim(cytaalen, cytaa, bdx, temp16b);
 
2861
 
 
2862
    cytbblen = scale_expansion_zeroelim(4, bb, cdytail, cytbb);
 
2863
    temp16clen = scale_expansion_zeroelim(cytbblen, cytbb, -adx, temp16c);
 
2864
 
 
2865
    temp32alen = fast_expansion_sum_zeroelim(temp16alen, temp16a,
 
2866
                                            temp16blen, temp16b, temp32a);
 
2867
    temp48len = fast_expansion_sum_zeroelim(temp16clen, temp16c,
 
2868
                                            temp32alen, temp32a, temp48);
 
2869
    finlength = fast_expansion_sum_zeroelim(finlength, finnow, temp48len,
 
2870
                                            temp48, finother);
 
2871
    finswap = finnow; finnow = finother; finother = finswap;
 
2872
  }
 
2873
 
 
2874
  if ((adxtail != 0.0) || (adytail != 0.0)) {
 
2875
    if ((bdxtail != 0.0) || (bdytail != 0.0)
 
2876
        || (cdxtail != 0.0) || (cdytail != 0.0)) {
 
2877
      Two_Product(bdxtail, cdy, ti1, ti0);
 
2878
      Two_Product(bdx, cdytail, tj1, tj0);
 
2879
      Two_Two_Sum(ti1, ti0, tj1, tj0, u3, u[2], u[1], u[0]);
 
2880
      u[3] = u3;
 
2881
      negate = -bdy;
 
2882
      Two_Product(cdxtail, negate, ti1, ti0);
 
2883
      negate = -bdytail;
 
2884
      Two_Product(cdx, negate, tj1, tj0);
 
2885
      Two_Two_Sum(ti1, ti0, tj1, tj0, v3, v[2], v[1], v[0]);
 
2886
      v[3] = v3;
 
2887
      bctlen = fast_expansion_sum_zeroelim(4, u, 4, v, bct);
 
2888
 
 
2889
      Two_Product(bdxtail, cdytail, ti1, ti0);
 
2890
      Two_Product(cdxtail, bdytail, tj1, tj0);
 
2891
      Two_Two_Diff(ti1, ti0, tj1, tj0, bctt3, bctt[2], bctt[1], bctt[0]);
 
2892
      bctt[3] = bctt3;
 
2893
      bcttlen = 4;
 
2894
    } else {
 
2895
      bct[0] = 0.0;
 
2896
      bctlen = 1;
 
2897
      bctt[0] = 0.0;
 
2898
      bcttlen = 1;
 
2899
    }
 
2900
 
 
2901
    if (adxtail != 0.0) {
 
2902
      temp16alen = scale_expansion_zeroelim(axtbclen, axtbc, adxtail, temp16a);
 
2903
      axtbctlen = scale_expansion_zeroelim(bctlen, bct, adxtail, axtbct);
 
2904
      temp32alen = scale_expansion_zeroelim(axtbctlen, axtbct, 2.0 * adx,
 
2905
                                            temp32a);
 
2906
      temp48len = fast_expansion_sum_zeroelim(temp16alen, temp16a,
 
2907
                                              temp32alen, temp32a, temp48);
 
2908
      finlength = fast_expansion_sum_zeroelim(finlength, finnow, temp48len,
 
2909
                                              temp48, finother);
 
2910
      finswap = finnow; finnow = finother; finother = finswap;
 
2911
      if (bdytail != 0.0) {
 
2912
        temp8len = scale_expansion_zeroelim(4, cc, adxtail, temp8);
 
2913
        temp16alen = scale_expansion_zeroelim(temp8len, temp8, bdytail,
 
2914
                                              temp16a);
 
2915
        finlength = fast_expansion_sum_zeroelim(finlength, finnow, temp16alen,
 
2916
                                                temp16a, finother);
 
2917
        finswap = finnow; finnow = finother; finother = finswap;
 
2918
      }
 
2919
      if (cdytail != 0.0) {
 
2920
        temp8len = scale_expansion_zeroelim(4, bb, -adxtail, temp8);
 
2921
        temp16alen = scale_expansion_zeroelim(temp8len, temp8, cdytail,
 
2922
                                              temp16a);
 
2923
        finlength = fast_expansion_sum_zeroelim(finlength, finnow, temp16alen,
 
2924
                                                temp16a, finother);
 
2925
        finswap = finnow; finnow = finother; finother = finswap;
 
2926
      }
 
2927
 
 
2928
      temp32alen = scale_expansion_zeroelim(axtbctlen, axtbct, adxtail,
 
2929
                                            temp32a);
 
2930
      axtbcttlen = scale_expansion_zeroelim(bcttlen, bctt, adxtail, axtbctt);
 
2931
      temp16alen = scale_expansion_zeroelim(axtbcttlen, axtbctt, 2.0 * adx,
 
2932
                                            temp16a);
 
2933
      temp16blen = scale_expansion_zeroelim(axtbcttlen, axtbctt, adxtail,
 
2934
                                            temp16b);
 
2935
      temp32blen = fast_expansion_sum_zeroelim(temp16alen, temp16a,
 
2936
                                              temp16blen, temp16b, temp32b);
 
2937
      temp64len = fast_expansion_sum_zeroelim(temp32alen, temp32a,
 
2938
                                              temp32blen, temp32b, temp64);
 
2939
      finlength = fast_expansion_sum_zeroelim(finlength, finnow, temp64len,
 
2940
                                              temp64, finother);
 
2941
      finswap = finnow; finnow = finother; finother = finswap;
 
2942
    }
 
2943
    if (adytail != 0.0) {
 
2944
      temp16alen = scale_expansion_zeroelim(aytbclen, aytbc, adytail, temp16a);
 
2945
      aytbctlen = scale_expansion_zeroelim(bctlen, bct, adytail, aytbct);
 
2946
      temp32alen = scale_expansion_zeroelim(aytbctlen, aytbct, 2.0 * ady,
 
2947
                                            temp32a);
 
2948
      temp48len = fast_expansion_sum_zeroelim(temp16alen, temp16a,
 
2949
                                              temp32alen, temp32a, temp48);
 
2950
      finlength = fast_expansion_sum_zeroelim(finlength, finnow, temp48len,
 
2951
                                              temp48, finother);
 
2952
      finswap = finnow; finnow = finother; finother = finswap;
 
2953
 
 
2954
 
 
2955
      temp32alen = scale_expansion_zeroelim(aytbctlen, aytbct, adytail,
 
2956
                                            temp32a);
 
2957
      aytbcttlen = scale_expansion_zeroelim(bcttlen, bctt, adytail, aytbctt);
 
2958
      temp16alen = scale_expansion_zeroelim(aytbcttlen, aytbctt, 2.0 * ady,
 
2959
                                            temp16a);
 
2960
      temp16blen = scale_expansion_zeroelim(aytbcttlen, aytbctt, adytail,
 
2961
                                            temp16b);
 
2962
      temp32blen = fast_expansion_sum_zeroelim(temp16alen, temp16a,
 
2963
                                              temp16blen, temp16b, temp32b);
 
2964
      temp64len = fast_expansion_sum_zeroelim(temp32alen, temp32a,
 
2965
                                              temp32blen, temp32b, temp64);
 
2966
      finlength = fast_expansion_sum_zeroelim(finlength, finnow, temp64len,
 
2967
                                              temp64, finother);
 
2968
      finswap = finnow; finnow = finother; finother = finswap;
 
2969
    }
 
2970
  }
 
2971
  if ((bdxtail != 0.0) || (bdytail != 0.0)) {
 
2972
    if ((cdxtail != 0.0) || (cdytail != 0.0)
 
2973
        || (adxtail != 0.0) || (adytail != 0.0)) {
 
2974
      Two_Product(cdxtail, ady, ti1, ti0);
 
2975
      Two_Product(cdx, adytail, tj1, tj0);
 
2976
      Two_Two_Sum(ti1, ti0, tj1, tj0, u3, u[2], u[1], u[0]);
 
2977
      u[3] = u3;
 
2978
      negate = -cdy;
 
2979
      Two_Product(adxtail, negate, ti1, ti0);
 
2980
      negate = -cdytail;
 
2981
      Two_Product(adx, negate, tj1, tj0);
 
2982
      Two_Two_Sum(ti1, ti0, tj1, tj0, v3, v[2], v[1], v[0]);
 
2983
      v[3] = v3;
 
2984
      catlen = fast_expansion_sum_zeroelim(4, u, 4, v, cat);
 
2985
 
 
2986
      Two_Product(cdxtail, adytail, ti1, ti0);
 
2987
      Two_Product(adxtail, cdytail, tj1, tj0);
 
2988
      Two_Two_Diff(ti1, ti0, tj1, tj0, catt3, catt[2], catt[1], catt[0]);
 
2989
      catt[3] = catt3;
 
2990
      cattlen = 4;
 
2991
    } else {
 
2992
      cat[0] = 0.0;
 
2993
      catlen = 1;
 
2994
      catt[0] = 0.0;
 
2995
      cattlen = 1;
 
2996
    }
 
2997
 
 
2998
    if (bdxtail != 0.0) {
 
2999
      temp16alen = scale_expansion_zeroelim(bxtcalen, bxtca, bdxtail, temp16a);
 
3000
      bxtcatlen = scale_expansion_zeroelim(catlen, cat, bdxtail, bxtcat);
 
3001
      temp32alen = scale_expansion_zeroelim(bxtcatlen, bxtcat, 2.0 * bdx,
 
3002
                                            temp32a);
 
3003
      temp48len = fast_expansion_sum_zeroelim(temp16alen, temp16a,
 
3004
                                              temp32alen, temp32a, temp48);
 
3005
      finlength = fast_expansion_sum_zeroelim(finlength, finnow, temp48len,
 
3006
                                              temp48, finother);
 
3007
      finswap = finnow; finnow = finother; finother = finswap;
 
3008
      if (cdytail != 0.0) {
 
3009
        temp8len = scale_expansion_zeroelim(4, aa, bdxtail, temp8);
 
3010
        temp16alen = scale_expansion_zeroelim(temp8len, temp8, cdytail,
 
3011
                                              temp16a);
 
3012
        finlength = fast_expansion_sum_zeroelim(finlength, finnow, temp16alen,
 
3013
                                                temp16a, finother);
 
3014
        finswap = finnow; finnow = finother; finother = finswap;
 
3015
      }
 
3016
      if (adytail != 0.0) {
 
3017
        temp8len = scale_expansion_zeroelim(4, cc, -bdxtail, temp8);
 
3018
        temp16alen = scale_expansion_zeroelim(temp8len, temp8, adytail,
 
3019
                                              temp16a);
 
3020
        finlength = fast_expansion_sum_zeroelim(finlength, finnow, temp16alen,
 
3021
                                                temp16a, finother);
 
3022
        finswap = finnow; finnow = finother; finother = finswap;
 
3023
      }
 
3024
 
 
3025
      temp32alen = scale_expansion_zeroelim(bxtcatlen, bxtcat, bdxtail,
 
3026
                                            temp32a);
 
3027
      bxtcattlen = scale_expansion_zeroelim(cattlen, catt, bdxtail, bxtcatt);
 
3028
      temp16alen = scale_expansion_zeroelim(bxtcattlen, bxtcatt, 2.0 * bdx,
 
3029
                                            temp16a);
 
3030
      temp16blen = scale_expansion_zeroelim(bxtcattlen, bxtcatt, bdxtail,
 
3031
                                            temp16b);
 
3032
      temp32blen = fast_expansion_sum_zeroelim(temp16alen, temp16a,
 
3033
                                              temp16blen, temp16b, temp32b);
 
3034
      temp64len = fast_expansion_sum_zeroelim(temp32alen, temp32a,
 
3035
                                              temp32blen, temp32b, temp64);
 
3036
      finlength = fast_expansion_sum_zeroelim(finlength, finnow, temp64len,
 
3037
                                              temp64, finother);
 
3038
      finswap = finnow; finnow = finother; finother = finswap;
 
3039
    }
 
3040
    if (bdytail != 0.0) {
 
3041
      temp16alen = scale_expansion_zeroelim(bytcalen, bytca, bdytail, temp16a);
 
3042
      bytcatlen = scale_expansion_zeroelim(catlen, cat, bdytail, bytcat);
 
3043
      temp32alen = scale_expansion_zeroelim(bytcatlen, bytcat, 2.0 * bdy,
 
3044
                                            temp32a);
 
3045
      temp48len = fast_expansion_sum_zeroelim(temp16alen, temp16a,
 
3046
                                              temp32alen, temp32a, temp48);
 
3047
      finlength = fast_expansion_sum_zeroelim(finlength, finnow, temp48len,
 
3048
                                              temp48, finother);
 
3049
      finswap = finnow; finnow = finother; finother = finswap;
 
3050
 
 
3051
 
 
3052
      temp32alen = scale_expansion_zeroelim(bytcatlen, bytcat, bdytail,
 
3053
                                            temp32a);
 
3054
      bytcattlen = scale_expansion_zeroelim(cattlen, catt, bdytail, bytcatt);
 
3055
      temp16alen = scale_expansion_zeroelim(bytcattlen, bytcatt, 2.0 * bdy,
 
3056
                                            temp16a);
 
3057
      temp16blen = scale_expansion_zeroelim(bytcattlen, bytcatt, bdytail,
 
3058
                                            temp16b);
 
3059
      temp32blen = fast_expansion_sum_zeroelim(temp16alen, temp16a,
 
3060
                                              temp16blen, temp16b, temp32b);
 
3061
      temp64len = fast_expansion_sum_zeroelim(temp32alen, temp32a,
 
3062
                                              temp32blen, temp32b, temp64);
 
3063
      finlength = fast_expansion_sum_zeroelim(finlength, finnow, temp64len,
 
3064
                                              temp64, finother);
 
3065
      finswap = finnow; finnow = finother; finother = finswap;
 
3066
    }
 
3067
  }
 
3068
  if ((cdxtail != 0.0) || (cdytail != 0.0)) {
 
3069
    if ((adxtail != 0.0) || (adytail != 0.0)
 
3070
        || (bdxtail != 0.0) || (bdytail != 0.0)) {
 
3071
      Two_Product(adxtail, bdy, ti1, ti0);
 
3072
      Two_Product(adx, bdytail, tj1, tj0);
 
3073
      Two_Two_Sum(ti1, ti0, tj1, tj0, u3, u[2], u[1], u[0]);
 
3074
      u[3] = u3;
 
3075
      negate = -ady;
 
3076
      Two_Product(bdxtail, negate, ti1, ti0);
 
3077
      negate = -adytail;
 
3078
      Two_Product(bdx, negate, tj1, tj0);
 
3079
      Two_Two_Sum(ti1, ti0, tj1, tj0, v3, v[2], v[1], v[0]);
 
3080
      v[3] = v3;
 
3081
      abtlen = fast_expansion_sum_zeroelim(4, u, 4, v, abt);
 
3082
 
 
3083
      Two_Product(adxtail, bdytail, ti1, ti0);
 
3084
      Two_Product(bdxtail, adytail, tj1, tj0);
 
3085
      Two_Two_Diff(ti1, ti0, tj1, tj0, abtt3, abtt[2], abtt[1], abtt[0]);
 
3086
      abtt[3] = abtt3;
 
3087
      abttlen = 4;
 
3088
    } else {
 
3089
      abt[0] = 0.0;
 
3090
      abtlen = 1;
 
3091
      abtt[0] = 0.0;
 
3092
      abttlen = 1;
 
3093
    }
 
3094
 
 
3095
    if (cdxtail != 0.0) {
 
3096
      temp16alen = scale_expansion_zeroelim(cxtablen, cxtab, cdxtail, temp16a);
 
3097
      cxtabtlen = scale_expansion_zeroelim(abtlen, abt, cdxtail, cxtabt);
 
3098
      temp32alen = scale_expansion_zeroelim(cxtabtlen, cxtabt, 2.0 * cdx,
 
3099
                                            temp32a);
 
3100
      temp48len = fast_expansion_sum_zeroelim(temp16alen, temp16a,
 
3101
                                              temp32alen, temp32a, temp48);
 
3102
      finlength = fast_expansion_sum_zeroelim(finlength, finnow, temp48len,
 
3103
                                              temp48, finother);
 
3104
      finswap = finnow; finnow = finother; finother = finswap;
 
3105
      if (adytail != 0.0) {
 
3106
        temp8len = scale_expansion_zeroelim(4, bb, cdxtail, temp8);
 
3107
        temp16alen = scale_expansion_zeroelim(temp8len, temp8, adytail,
 
3108
                                              temp16a);
 
3109
        finlength = fast_expansion_sum_zeroelim(finlength, finnow, temp16alen,
 
3110
                                                temp16a, finother);
 
3111
        finswap = finnow; finnow = finother; finother = finswap;
 
3112
      }
 
3113
      if (bdytail != 0.0) {
 
3114
        temp8len = scale_expansion_zeroelim(4, aa, -cdxtail, temp8);
 
3115
        temp16alen = scale_expansion_zeroelim(temp8len, temp8, bdytail,
 
3116
                                              temp16a);
 
3117
        finlength = fast_expansion_sum_zeroelim(finlength, finnow, temp16alen,
 
3118
                                                temp16a, finother);
 
3119
        finswap = finnow; finnow = finother; finother = finswap;
 
3120
      }
 
3121
 
 
3122
      temp32alen = scale_expansion_zeroelim(cxtabtlen, cxtabt, cdxtail,
 
3123
                                            temp32a);
 
3124
      cxtabttlen = scale_expansion_zeroelim(abttlen, abtt, cdxtail, cxtabtt);
 
3125
      temp16alen = scale_expansion_zeroelim(cxtabttlen, cxtabtt, 2.0 * cdx,
 
3126
                                            temp16a);
 
3127
      temp16blen = scale_expansion_zeroelim(cxtabttlen, cxtabtt, cdxtail,
 
3128
                                            temp16b);
 
3129
      temp32blen = fast_expansion_sum_zeroelim(temp16alen, temp16a,
 
3130
                                              temp16blen, temp16b, temp32b);
 
3131
      temp64len = fast_expansion_sum_zeroelim(temp32alen, temp32a,
 
3132
                                              temp32blen, temp32b, temp64);
 
3133
      finlength = fast_expansion_sum_zeroelim(finlength, finnow, temp64len,
 
3134
                                              temp64, finother);
 
3135
      finswap = finnow; finnow = finother; finother = finswap;
 
3136
    }
 
3137
    if (cdytail != 0.0) {
 
3138
      temp16alen = scale_expansion_zeroelim(cytablen, cytab, cdytail, temp16a);
 
3139
      cytabtlen = scale_expansion_zeroelim(abtlen, abt, cdytail, cytabt);
 
3140
      temp32alen = scale_expansion_zeroelim(cytabtlen, cytabt, 2.0 * cdy,
 
3141
                                            temp32a);
 
3142
      temp48len = fast_expansion_sum_zeroelim(temp16alen, temp16a,
 
3143
                                              temp32alen, temp32a, temp48);
 
3144
      finlength = fast_expansion_sum_zeroelim(finlength, finnow, temp48len,
 
3145
                                              temp48, finother);
 
3146
      finswap = finnow; finnow = finother; finother = finswap;
 
3147
 
 
3148
 
 
3149
      temp32alen = scale_expansion_zeroelim(cytabtlen, cytabt, cdytail,
 
3150
                                            temp32a);
 
3151
      cytabttlen = scale_expansion_zeroelim(abttlen, abtt, cdytail, cytabtt);
 
3152
      temp16alen = scale_expansion_zeroelim(cytabttlen, cytabtt, 2.0 * cdy,
 
3153
                                            temp16a);
 
3154
      temp16blen = scale_expansion_zeroelim(cytabttlen, cytabtt, cdytail,
 
3155
                                            temp16b);
 
3156
      temp32blen = fast_expansion_sum_zeroelim(temp16alen, temp16a,
 
3157
                                              temp16blen, temp16b, temp32b);
 
3158
      temp64len = fast_expansion_sum_zeroelim(temp32alen, temp32a,
 
3159
                                              temp32blen, temp32b, temp64);
 
3160
      finlength = fast_expansion_sum_zeroelim(finlength, finnow, temp64len,
 
3161
                                              temp64, finother);
 
3162
      finswap = finnow; finnow = finother; finother = finswap;
 
3163
    }
 
3164
  }
 
3165
 
 
3166
  return finnow[finlength - 1];
 
3167
}
 
3168
 
 
3169
REAL incircle(REAL *pa, REAL *pb, REAL *pc, REAL *pd)
 
3170
{
 
3171
  REAL adx, bdx, cdx, ady, bdy, cdy;
 
3172
  REAL bdxcdy, cdxbdy, cdxady, adxcdy, adxbdy, bdxady;
 
3173
  REAL alift, blift, clift;
 
3174
  REAL det;
 
3175
  REAL permanent, errbound;
 
3176
 
 
3177
  adx = pa[0] - pd[0];
 
3178
  bdx = pb[0] - pd[0];
 
3179
  cdx = pc[0] - pd[0];
 
3180
  ady = pa[1] - pd[1];
 
3181
  bdy = pb[1] - pd[1];
 
3182
  cdy = pc[1] - pd[1];
 
3183
 
 
3184
  bdxcdy = bdx * cdy;
 
3185
  cdxbdy = cdx * bdy;
 
3186
  alift = adx * adx + ady * ady;
 
3187
 
 
3188
  cdxady = cdx * ady;
 
3189
  adxcdy = adx * cdy;
 
3190
  blift = bdx * bdx + bdy * bdy;
 
3191
 
 
3192
  adxbdy = adx * bdy;
 
3193
  bdxady = bdx * ady;
 
3194
  clift = cdx * cdx + cdy * cdy;
 
3195
 
 
3196
  det = alift * (bdxcdy - cdxbdy)
 
3197
      + blift * (cdxady - adxcdy)
 
3198
      + clift * (adxbdy - bdxady);
 
3199
 
 
3200
  permanent = (Absolute(bdxcdy) + Absolute(cdxbdy)) * alift
 
3201
            + (Absolute(cdxady) + Absolute(adxcdy)) * blift
 
3202
            + (Absolute(adxbdy) + Absolute(bdxady)) * clift;
 
3203
  errbound = iccerrboundA * permanent;
 
3204
  if ((det > errbound) || (-det > errbound)) {
 
3205
    return det;
 
3206
  }
 
3207
 
 
3208
  return incircleadapt(pa, pb, pc, pd, permanent);
 
3209
}
 
3210
 
 
3211
/*****************************************************************************/
 
3212
/*                                                                           */
 
3213
/*  inspherefast()   Approximate 3D insphere test.  Nonrobust.               */
 
3214
/*  insphereexact()   Exact 3D insphere test.  Robust.                       */
 
3215
/*  insphereslow()   Another exact 3D insphere test.  Robust.                */
 
3216
/*  insphere()   Adaptive exact 3D insphere test.  Robust.                   */
 
3217
/*                                                                           */
 
3218
/*               Return a positive value if the point pe lies inside the     */
 
3219
/*               sphere passing through pa, pb, pc, and pd; a negative value */
 
3220
/*               if it lies outside; and zero if the five points are         */
 
3221
/*               cospherical.  The points pa, pb, pc, and pd must be ordered */
 
3222
/*               so that they have a positive orientation (as defined by     */
 
3223
/*               orient3d()), or the sign of the result will be reversed.    */
 
3224
/*                                                                           */
 
3225
/*  Only the first and last routine should be used; the middle two are for   */
 
3226
/*  timings.                                                                 */
 
3227
/*                                                                           */
 
3228
/*  The last three use exact arithmetic to ensure a correct answer.  The     */
 
3229
/*  result returned is the determinant of a matrix.  In insphere() only,     */
 
3230
/*  this determinant is computed adaptively, in the sense that exact         */
 
3231
/*  arithmetic is used only to the degree it is needed to ensure that the    */
 
3232
/*  returned value has the correct sign.  Hence, insphere() is usually quite */
 
3233
/*  fast, but will run more slowly when the input points are cospherical or  */
 
3234
/*  nearly so.                                                               */
 
3235
/*                                                                           */
 
3236
/*****************************************************************************/
 
3237
 
 
3238
REAL inspherefast(REAL *pa, REAL *pb, REAL *pc, REAL *pd, REAL *pe)
 
3239
{
 
3240
  REAL aex, bex, cex, dex;
 
3241
  REAL aey, bey, cey, dey;
 
3242
  REAL aez, bez, cez, dez;
 
3243
  REAL alift, blift, clift, dlift;
 
3244
  REAL ab, bc, cd, da, ac, bd;
 
3245
  REAL abc, bcd, cda, dab;
 
3246
 
 
3247
  aex = pa[0] - pe[0];
 
3248
  bex = pb[0] - pe[0];
 
3249
  cex = pc[0] - pe[0];
 
3250
  dex = pd[0] - pe[0];
 
3251
  aey = pa[1] - pe[1];
 
3252
  bey = pb[1] - pe[1];
 
3253
  cey = pc[1] - pe[1];
 
3254
  dey = pd[1] - pe[1];
 
3255
  aez = pa[2] - pe[2];
 
3256
  bez = pb[2] - pe[2];
 
3257
  cez = pc[2] - pe[2];
 
3258
  dez = pd[2] - pe[2];
 
3259
 
 
3260
  ab = aex * bey - bex * aey;
 
3261
  bc = bex * cey - cex * bey;
 
3262
  cd = cex * dey - dex * cey;
 
3263
  da = dex * aey - aex * dey;
 
3264
 
 
3265
  ac = aex * cey - cex * aey;
 
3266
  bd = bex * dey - dex * bey;
 
3267
 
 
3268
  abc = aez * bc - bez * ac + cez * ab;
 
3269
  bcd = bez * cd - cez * bd + dez * bc;
 
3270
  cda = cez * da + dez * ac + aez * cd;
 
3271
  dab = dez * ab + aez * bd + bez * da;
 
3272
 
 
3273
  alift = aex * aex + aey * aey + aez * aez;
 
3274
  blift = bex * bex + bey * bey + bez * bez;
 
3275
  clift = cex * cex + cey * cey + cez * cez;
 
3276
  dlift = dex * dex + dey * dey + dez * dez;
 
3277
 
 
3278
  return (dlift * abc - clift * dab) + (blift * cda - alift * bcd);
 
3279
}
 
3280
 
 
3281
REAL insphereexact(REAL *pa, REAL *pb, REAL *pc, REAL *pd, REAL *pe)
 
3282
{
 
3283
  INEXACT REAL axby1, bxcy1, cxdy1, dxey1, exay1;
 
3284
  INEXACT REAL bxay1, cxby1, dxcy1, exdy1, axey1;
 
3285
  INEXACT REAL axcy1, bxdy1, cxey1, dxay1, exby1;
 
3286
  INEXACT REAL cxay1, dxby1, excy1, axdy1, bxey1;
 
3287
  REAL axby0, bxcy0, cxdy0, dxey0, exay0;
 
3288
  REAL bxay0, cxby0, dxcy0, exdy0, axey0;
 
3289
  REAL axcy0, bxdy0, cxey0, dxay0, exby0;
 
3290
  REAL cxay0, dxby0, excy0, axdy0, bxey0;
 
3291
  REAL ab[4], bc[4], cd[4], de[4], ea[4];
 
3292
  REAL ac[4], bd[4], ce[4], da[4], eb[4];
 
3293
  REAL temp8a[8], temp8b[8], temp16[16];
 
3294
  int temp8alen, temp8blen, temp16len;
 
3295
  REAL abc[24], bcd[24], cde[24], dea[24], eab[24];
 
3296
  REAL abd[24], bce[24], cda[24], deb[24], eac[24];
 
3297
  int abclen, bcdlen, cdelen, dealen, eablen;
 
3298
  int abdlen, bcelen, cdalen, deblen, eaclen;
 
3299
  REAL temp48a[48], temp48b[48];
 
3300
  int temp48alen, temp48blen;
 
3301
  REAL abcd[96], bcde[96], cdea[96], deab[96], eabc[96];
 
3302
  int abcdlen, bcdelen, cdealen, deablen, eabclen;
 
3303
  REAL temp192[192];
 
3304
  REAL det384x[384], det384y[384], det384z[384];
 
3305
  int xlen, ylen, zlen;
 
3306
  REAL detxy[768];
 
3307
  int xylen;
 
3308
  REAL adet[1152], bdet[1152], cdet[1152], ddet[1152], edet[1152];
 
3309
  int alen, blen, clen, dlen, elen;
 
3310
  REAL abdet[2304], cddet[2304], cdedet[3456];
 
3311
  int ablen, cdlen;
 
3312
  REAL deter[5760];
 
3313
  int deterlen;
 
3314
  int i;
 
3315
 
 
3316
  INEXACT REAL bvirt;
 
3317
  REAL avirt, bround, around;
 
3318
  INEXACT REAL c;
 
3319
  INEXACT REAL abig;
 
3320
  REAL ahi, alo, bhi, blo;
 
3321
  REAL err1, err2, err3;
 
3322
  INEXACT REAL _i, _j;
 
3323
  REAL _0;
 
3324
 
 
3325
  Two_Product(pa[0], pb[1], axby1, axby0);
 
3326
  Two_Product(pb[0], pa[1], bxay1, bxay0);
 
3327
  Two_Two_Diff(axby1, axby0, bxay1, bxay0, ab[3], ab[2], ab[1], ab[0]);
 
3328
 
 
3329
  Two_Product(pb[0], pc[1], bxcy1, bxcy0);
 
3330
  Two_Product(pc[0], pb[1], cxby1, cxby0);
 
3331
  Two_Two_Diff(bxcy1, bxcy0, cxby1, cxby0, bc[3], bc[2], bc[1], bc[0]);
 
3332
 
 
3333
  Two_Product(pc[0], pd[1], cxdy1, cxdy0);
 
3334
  Two_Product(pd[0], pc[1], dxcy1, dxcy0);
 
3335
  Two_Two_Diff(cxdy1, cxdy0, dxcy1, dxcy0, cd[3], cd[2], cd[1], cd[0]);
 
3336
 
 
3337
  Two_Product(pd[0], pe[1], dxey1, dxey0);
 
3338
  Two_Product(pe[0], pd[1], exdy1, exdy0);
 
3339
  Two_Two_Diff(dxey1, dxey0, exdy1, exdy0, de[3], de[2], de[1], de[0]);
 
3340
 
 
3341
  Two_Product(pe[0], pa[1], exay1, exay0);
 
3342
  Two_Product(pa[0], pe[1], axey1, axey0);
 
3343
  Two_Two_Diff(exay1, exay0, axey1, axey0, ea[3], ea[2], ea[1], ea[0]);
 
3344
 
 
3345
  Two_Product(pa[0], pc[1], axcy1, axcy0);
 
3346
  Two_Product(pc[0], pa[1], cxay1, cxay0);
 
3347
  Two_Two_Diff(axcy1, axcy0, cxay1, cxay0, ac[3], ac[2], ac[1], ac[0]);
 
3348
 
 
3349
  Two_Product(pb[0], pd[1], bxdy1, bxdy0);
 
3350
  Two_Product(pd[0], pb[1], dxby1, dxby0);
 
3351
  Two_Two_Diff(bxdy1, bxdy0, dxby1, dxby0, bd[3], bd[2], bd[1], bd[0]);
 
3352
 
 
3353
  Two_Product(pc[0], pe[1], cxey1, cxey0);
 
3354
  Two_Product(pe[0], pc[1], excy1, excy0);
 
3355
  Two_Two_Diff(cxey1, cxey0, excy1, excy0, ce[3], ce[2], ce[1], ce[0]);
 
3356
 
 
3357
  Two_Product(pd[0], pa[1], dxay1, dxay0);
 
3358
  Two_Product(pa[0], pd[1], axdy1, axdy0);
 
3359
  Two_Two_Diff(dxay1, dxay0, axdy1, axdy0, da[3], da[2], da[1], da[0]);
 
3360
 
 
3361
  Two_Product(pe[0], pb[1], exby1, exby0);
 
3362
  Two_Product(pb[0], pe[1], bxey1, bxey0);
 
3363
  Two_Two_Diff(exby1, exby0, bxey1, bxey0, eb[3], eb[2], eb[1], eb[0]);
 
3364
 
 
3365
  temp8alen = scale_expansion_zeroelim(4, bc, pa[2], temp8a);
 
3366
  temp8blen = scale_expansion_zeroelim(4, ac, -pb[2], temp8b);
 
3367
  temp16len = fast_expansion_sum_zeroelim(temp8alen, temp8a, temp8blen, temp8b,
 
3368
                                          temp16);
 
3369
  temp8alen = scale_expansion_zeroelim(4, ab, pc[2], temp8a);
 
3370
  abclen = fast_expansion_sum_zeroelim(temp8alen, temp8a, temp16len, temp16,
 
3371
                                       abc);
 
3372
 
 
3373
  temp8alen = scale_expansion_zeroelim(4, cd, pb[2], temp8a);
 
3374
  temp8blen = scale_expansion_zeroelim(4, bd, -pc[2], temp8b);
 
3375
  temp16len = fast_expansion_sum_zeroelim(temp8alen, temp8a, temp8blen, temp8b,
 
3376
                                          temp16);
 
3377
  temp8alen = scale_expansion_zeroelim(4, bc, pd[2], temp8a);
 
3378
  bcdlen = fast_expansion_sum_zeroelim(temp8alen, temp8a, temp16len, temp16,
 
3379
                                       bcd);
 
3380
 
 
3381
  temp8alen = scale_expansion_zeroelim(4, de, pc[2], temp8a);
 
3382
  temp8blen = scale_expansion_zeroelim(4, ce, -pd[2], temp8b);
 
3383
  temp16len = fast_expansion_sum_zeroelim(temp8alen, temp8a, temp8blen, temp8b,
 
3384
                                          temp16);
 
3385
  temp8alen = scale_expansion_zeroelim(4, cd, pe[2], temp8a);
 
3386
  cdelen = fast_expansion_sum_zeroelim(temp8alen, temp8a, temp16len, temp16,
 
3387
                                       cde);
 
3388
 
 
3389
  temp8alen = scale_expansion_zeroelim(4, ea, pd[2], temp8a);
 
3390
  temp8blen = scale_expansion_zeroelim(4, da, -pe[2], temp8b);
 
3391
  temp16len = fast_expansion_sum_zeroelim(temp8alen, temp8a, temp8blen, temp8b,
 
3392
                                          temp16);
 
3393
  temp8alen = scale_expansion_zeroelim(4, de, pa[2], temp8a);
 
3394
  dealen = fast_expansion_sum_zeroelim(temp8alen, temp8a, temp16len, temp16,
 
3395
                                       dea);
 
3396
 
 
3397
  temp8alen = scale_expansion_zeroelim(4, ab, pe[2], temp8a);
 
3398
  temp8blen = scale_expansion_zeroelim(4, eb, -pa[2], temp8b);
 
3399
  temp16len = fast_expansion_sum_zeroelim(temp8alen, temp8a, temp8blen, temp8b,
 
3400
                                          temp16);
 
3401
  temp8alen = scale_expansion_zeroelim(4, ea, pb[2], temp8a);
 
3402
  eablen = fast_expansion_sum_zeroelim(temp8alen, temp8a, temp16len, temp16,
 
3403
                                       eab);
 
3404
 
 
3405
  temp8alen = scale_expansion_zeroelim(4, bd, pa[2], temp8a);
 
3406
  temp8blen = scale_expansion_zeroelim(4, da, pb[2], temp8b);
 
3407
  temp16len = fast_expansion_sum_zeroelim(temp8alen, temp8a, temp8blen, temp8b,
 
3408
                                          temp16);
 
3409
  temp8alen = scale_expansion_zeroelim(4, ab, pd[2], temp8a);
 
3410
  abdlen = fast_expansion_sum_zeroelim(temp8alen, temp8a, temp16len, temp16,
 
3411
                                       abd);
 
3412
 
 
3413
  temp8alen = scale_expansion_zeroelim(4, ce, pb[2], temp8a);
 
3414
  temp8blen = scale_expansion_zeroelim(4, eb, pc[2], temp8b);
 
3415
  temp16len = fast_expansion_sum_zeroelim(temp8alen, temp8a, temp8blen, temp8b,
 
3416
                                          temp16);
 
3417
  temp8alen = scale_expansion_zeroelim(4, bc, pe[2], temp8a);
 
3418
  bcelen = fast_expansion_sum_zeroelim(temp8alen, temp8a, temp16len, temp16,
 
3419
                                       bce);
 
3420
 
 
3421
  temp8alen = scale_expansion_zeroelim(4, da, pc[2], temp8a);
 
3422
  temp8blen = scale_expansion_zeroelim(4, ac, pd[2], temp8b);
 
3423
  temp16len = fast_expansion_sum_zeroelim(temp8alen, temp8a, temp8blen, temp8b,
 
3424
                                          temp16);
 
3425
  temp8alen = scale_expansion_zeroelim(4, cd, pa[2], temp8a);
 
3426
  cdalen = fast_expansion_sum_zeroelim(temp8alen, temp8a, temp16len, temp16,
 
3427
                                       cda);
 
3428
 
 
3429
  temp8alen = scale_expansion_zeroelim(4, eb, pd[2], temp8a);
 
3430
  temp8blen = scale_expansion_zeroelim(4, bd, pe[2], temp8b);
 
3431
  temp16len = fast_expansion_sum_zeroelim(temp8alen, temp8a, temp8blen, temp8b,
 
3432
                                          temp16);
 
3433
  temp8alen = scale_expansion_zeroelim(4, de, pb[2], temp8a);
 
3434
  deblen = fast_expansion_sum_zeroelim(temp8alen, temp8a, temp16len, temp16,
 
3435
                                       deb);
 
3436
 
 
3437
  temp8alen = scale_expansion_zeroelim(4, ac, pe[2], temp8a);
 
3438
  temp8blen = scale_expansion_zeroelim(4, ce, pa[2], temp8b);
 
3439
  temp16len = fast_expansion_sum_zeroelim(temp8alen, temp8a, temp8blen, temp8b,
 
3440
                                          temp16);
 
3441
  temp8alen = scale_expansion_zeroelim(4, ea, pc[2], temp8a);
 
3442
  eaclen = fast_expansion_sum_zeroelim(temp8alen, temp8a, temp16len, temp16,
 
3443
                                       eac);
 
3444
 
 
3445
  temp48alen = fast_expansion_sum_zeroelim(cdelen, cde, bcelen, bce, temp48a);
 
3446
  temp48blen = fast_expansion_sum_zeroelim(deblen, deb, bcdlen, bcd, temp48b);
 
3447
  for (i = 0; i < temp48blen; i++) {
 
3448
    temp48b[i] = -temp48b[i];
 
3449
  }
 
3450
  bcdelen = fast_expansion_sum_zeroelim(temp48alen, temp48a,
 
3451
                                        temp48blen, temp48b, bcde);
 
3452
  xlen = scale_expansion_zeroelim(bcdelen, bcde, pa[0], temp192);
 
3453
  xlen = scale_expansion_zeroelim(xlen, temp192, pa[0], det384x);
 
3454
  ylen = scale_expansion_zeroelim(bcdelen, bcde, pa[1], temp192);
 
3455
  ylen = scale_expansion_zeroelim(ylen, temp192, pa[1], det384y);
 
3456
  zlen = scale_expansion_zeroelim(bcdelen, bcde, pa[2], temp192);
 
3457
  zlen = scale_expansion_zeroelim(zlen, temp192, pa[2], det384z);
 
3458
  xylen = fast_expansion_sum_zeroelim(xlen, det384x, ylen, det384y, detxy);
 
3459
  alen = fast_expansion_sum_zeroelim(xylen, detxy, zlen, det384z, adet);
 
3460
 
 
3461
  temp48alen = fast_expansion_sum_zeroelim(dealen, dea, cdalen, cda, temp48a);
 
3462
  temp48blen = fast_expansion_sum_zeroelim(eaclen, eac, cdelen, cde, temp48b);
 
3463
  for (i = 0; i < temp48blen; i++) {
 
3464
    temp48b[i] = -temp48b[i];
 
3465
  }
 
3466
  cdealen = fast_expansion_sum_zeroelim(temp48alen, temp48a,
 
3467
                                        temp48blen, temp48b, cdea);
 
3468
  xlen = scale_expansion_zeroelim(cdealen, cdea, pb[0], temp192);
 
3469
  xlen = scale_expansion_zeroelim(xlen, temp192, pb[0], det384x);
 
3470
  ylen = scale_expansion_zeroelim(cdealen, cdea, pb[1], temp192);
 
3471
  ylen = scale_expansion_zeroelim(ylen, temp192, pb[1], det384y);
 
3472
  zlen = scale_expansion_zeroelim(cdealen, cdea, pb[2], temp192);
 
3473
  zlen = scale_expansion_zeroelim(zlen, temp192, pb[2], det384z);
 
3474
  xylen = fast_expansion_sum_zeroelim(xlen, det384x, ylen, det384y, detxy);
 
3475
  blen = fast_expansion_sum_zeroelim(xylen, detxy, zlen, det384z, bdet);
 
3476
 
 
3477
  temp48alen = fast_expansion_sum_zeroelim(eablen, eab, deblen, deb, temp48a);
 
3478
  temp48blen = fast_expansion_sum_zeroelim(abdlen, abd, dealen, dea, temp48b);
 
3479
  for (i = 0; i < temp48blen; i++) {
 
3480
    temp48b[i] = -temp48b[i];
 
3481
  }
 
3482
  deablen = fast_expansion_sum_zeroelim(temp48alen, temp48a,
 
3483
                                        temp48blen, temp48b, deab);
 
3484
  xlen = scale_expansion_zeroelim(deablen, deab, pc[0], temp192);
 
3485
  xlen = scale_expansion_zeroelim(xlen, temp192, pc[0], det384x);
 
3486
  ylen = scale_expansion_zeroelim(deablen, deab, pc[1], temp192);
 
3487
  ylen = scale_expansion_zeroelim(ylen, temp192, pc[1], det384y);
 
3488
  zlen = scale_expansion_zeroelim(deablen, deab, pc[2], temp192);
 
3489
  zlen = scale_expansion_zeroelim(zlen, temp192, pc[2], det384z);
 
3490
  xylen = fast_expansion_sum_zeroelim(xlen, det384x, ylen, det384y, detxy);
 
3491
  clen = fast_expansion_sum_zeroelim(xylen, detxy, zlen, det384z, cdet);
 
3492
 
 
3493
  temp48alen = fast_expansion_sum_zeroelim(abclen, abc, eaclen, eac, temp48a);
 
3494
  temp48blen = fast_expansion_sum_zeroelim(bcelen, bce, eablen, eab, temp48b);
 
3495
  for (i = 0; i < temp48blen; i++) {
 
3496
    temp48b[i] = -temp48b[i];
 
3497
  }
 
3498
  eabclen = fast_expansion_sum_zeroelim(temp48alen, temp48a,
 
3499
                                        temp48blen, temp48b, eabc);
 
3500
  xlen = scale_expansion_zeroelim(eabclen, eabc, pd[0], temp192);
 
3501
  xlen = scale_expansion_zeroelim(xlen, temp192, pd[0], det384x);
 
3502
  ylen = scale_expansion_zeroelim(eabclen, eabc, pd[1], temp192);
 
3503
  ylen = scale_expansion_zeroelim(ylen, temp192, pd[1], det384y);
 
3504
  zlen = scale_expansion_zeroelim(eabclen, eabc, pd[2], temp192);
 
3505
  zlen = scale_expansion_zeroelim(zlen, temp192, pd[2], det384z);
 
3506
  xylen = fast_expansion_sum_zeroelim(xlen, det384x, ylen, det384y, detxy);
 
3507
  dlen = fast_expansion_sum_zeroelim(xylen, detxy, zlen, det384z, ddet);
 
3508
 
 
3509
  temp48alen = fast_expansion_sum_zeroelim(bcdlen, bcd, abdlen, abd, temp48a);
 
3510
  temp48blen = fast_expansion_sum_zeroelim(cdalen, cda, abclen, abc, temp48b);
 
3511
  for (i = 0; i < temp48blen; i++) {
 
3512
    temp48b[i] = -temp48b[i];
 
3513
  }
 
3514
  abcdlen = fast_expansion_sum_zeroelim(temp48alen, temp48a,
 
3515
                                        temp48blen, temp48b, abcd);
 
3516
  xlen = scale_expansion_zeroelim(abcdlen, abcd, pe[0], temp192);
 
3517
  xlen = scale_expansion_zeroelim(xlen, temp192, pe[0], det384x);
 
3518
  ylen = scale_expansion_zeroelim(abcdlen, abcd, pe[1], temp192);
 
3519
  ylen = scale_expansion_zeroelim(ylen, temp192, pe[1], det384y);
 
3520
  zlen = scale_expansion_zeroelim(abcdlen, abcd, pe[2], temp192);
 
3521
  zlen = scale_expansion_zeroelim(zlen, temp192, pe[2], det384z);
 
3522
  xylen = fast_expansion_sum_zeroelim(xlen, det384x, ylen, det384y, detxy);
 
3523
  elen = fast_expansion_sum_zeroelim(xylen, detxy, zlen, det384z, edet);
 
3524
 
 
3525
  ablen = fast_expansion_sum_zeroelim(alen, adet, blen, bdet, abdet);
 
3526
  cdlen = fast_expansion_sum_zeroelim(clen, cdet, dlen, ddet, cddet);
 
3527
  cdelen = fast_expansion_sum_zeroelim(cdlen, cddet, elen, edet, cdedet);
 
3528
  deterlen = fast_expansion_sum_zeroelim(ablen, abdet, cdelen, cdedet, deter);
 
3529
 
 
3530
  return deter[deterlen - 1];
 
3531
}
 
3532
 
 
3533
REAL insphereslow(REAL *pa, REAL *pb, REAL *pc, REAL *pd, REAL *pe)
 
3534
{
 
3535
  INEXACT REAL aex, bex, cex, dex, aey, bey, cey, dey, aez, bez, cez, dez;
 
3536
  REAL aextail, bextail, cextail, dextail;
 
3537
  REAL aeytail, beytail, ceytail, deytail;
 
3538
  REAL aeztail, beztail, ceztail, deztail;
 
3539
  REAL negate, negatetail;
 
3540
  INEXACT REAL axby7, bxcy7, cxdy7, dxay7, axcy7, bxdy7;
 
3541
  INEXACT REAL bxay7, cxby7, dxcy7, axdy7, cxay7, dxby7;
 
3542
  REAL axby[8], bxcy[8], cxdy[8], dxay[8], axcy[8], bxdy[8];
 
3543
  REAL bxay[8], cxby[8], dxcy[8], axdy[8], cxay[8], dxby[8];
 
3544
  REAL ab[16], bc[16], cd[16], da[16], ac[16], bd[16];
 
3545
  int ablen, bclen, cdlen, dalen, aclen, bdlen;
 
3546
  REAL temp32a[32], temp32b[32], temp64a[64], temp64b[64], temp64c[64];
 
3547
  int temp32alen, temp32blen, temp64alen, temp64blen, temp64clen;
 
3548
  REAL temp128[128], temp192[192];
 
3549
  int temp128len, temp192len;
 
3550
  REAL detx[384], detxx[768], detxt[384], detxxt[768], detxtxt[768];
 
3551
  int xlen, xxlen, xtlen, xxtlen, xtxtlen;
 
3552
  REAL x1[1536], x2[2304];
 
3553
  int x1len, x2len;
 
3554
  REAL dety[384], detyy[768], detyt[384], detyyt[768], detytyt[768];
 
3555
  int ylen, yylen, ytlen, yytlen, ytytlen;
 
3556
  REAL y1[1536], y2[2304];
 
3557
  int y1len, y2len;
 
3558
  REAL detz[384], detzz[768], detzt[384], detzzt[768], detztzt[768];
 
3559
  int zlen, zzlen, ztlen, zztlen, ztztlen;
 
3560
  REAL z1[1536], z2[2304];
 
3561
  int z1len, z2len;
 
3562
  REAL detxy[4608];
 
3563
  int xylen;
 
3564
  REAL adet[6912], bdet[6912], cdet[6912], ddet[6912];
 
3565
  int alen, blen, clen, dlen;
 
3566
  REAL abdet[13824], cddet[13824], deter[27648];
 
3567
  int deterlen;
 
3568
  int i;
 
3569
 
 
3570
  INEXACT REAL bvirt;
 
3571
  REAL avirt, bround, around;
 
3572
  INEXACT REAL c;
 
3573
  INEXACT REAL abig;
 
3574
  REAL a0hi, a0lo, a1hi, a1lo, bhi, blo;
 
3575
  REAL err1, err2, err3;
 
3576
  INEXACT REAL _i, _j, _k, _l, _m, _n;
 
3577
  REAL _0, _1, _2;
 
3578
 
 
3579
  Two_Diff(pa[0], pe[0], aex, aextail);
 
3580
  Two_Diff(pa[1], pe[1], aey, aeytail);
 
3581
  Two_Diff(pa[2], pe[2], aez, aeztail);
 
3582
  Two_Diff(pb[0], pe[0], bex, bextail);
 
3583
  Two_Diff(pb[1], pe[1], bey, beytail);
 
3584
  Two_Diff(pb[2], pe[2], bez, beztail);
 
3585
  Two_Diff(pc[0], pe[0], cex, cextail);
 
3586
  Two_Diff(pc[1], pe[1], cey, ceytail);
 
3587
  Two_Diff(pc[2], pe[2], cez, ceztail);
 
3588
  Two_Diff(pd[0], pe[0], dex, dextail);
 
3589
  Two_Diff(pd[1], pe[1], dey, deytail);
 
3590
  Two_Diff(pd[2], pe[2], dez, deztail);
 
3591
 
 
3592
  Two_Two_Product(aex, aextail, bey, beytail,
 
3593
                  axby7, axby[6], axby[5], axby[4],
 
3594
                  axby[3], axby[2], axby[1], axby[0]);
 
3595
  axby[7] = axby7;
 
3596
  negate = -aey;
 
3597
  negatetail = -aeytail;
 
3598
  Two_Two_Product(bex, bextail, negate, negatetail,
 
3599
                  bxay7, bxay[6], bxay[5], bxay[4],
 
3600
                  bxay[3], bxay[2], bxay[1], bxay[0]);
 
3601
  bxay[7] = bxay7;
 
3602
  ablen = fast_expansion_sum_zeroelim(8, axby, 8, bxay, ab);
 
3603
  Two_Two_Product(bex, bextail, cey, ceytail,
 
3604
                  bxcy7, bxcy[6], bxcy[5], bxcy[4],
 
3605
                  bxcy[3], bxcy[2], bxcy[1], bxcy[0]);
 
3606
  bxcy[7] = bxcy7;
 
3607
  negate = -bey;
 
3608
  negatetail = -beytail;
 
3609
  Two_Two_Product(cex, cextail, negate, negatetail,
 
3610
                  cxby7, cxby[6], cxby[5], cxby[4],
 
3611
                  cxby[3], cxby[2], cxby[1], cxby[0]);
 
3612
  cxby[7] = cxby7;
 
3613
  bclen = fast_expansion_sum_zeroelim(8, bxcy, 8, cxby, bc);
 
3614
  Two_Two_Product(cex, cextail, dey, deytail,
 
3615
                  cxdy7, cxdy[6], cxdy[5], cxdy[4],
 
3616
                  cxdy[3], cxdy[2], cxdy[1], cxdy[0]);
 
3617
  cxdy[7] = cxdy7;
 
3618
  negate = -cey;
 
3619
  negatetail = -ceytail;
 
3620
  Two_Two_Product(dex, dextail, negate, negatetail,
 
3621
                  dxcy7, dxcy[6], dxcy[5], dxcy[4],
 
3622
                  dxcy[3], dxcy[2], dxcy[1], dxcy[0]);
 
3623
  dxcy[7] = dxcy7;
 
3624
  cdlen = fast_expansion_sum_zeroelim(8, cxdy, 8, dxcy, cd);
 
3625
  Two_Two_Product(dex, dextail, aey, aeytail,
 
3626
                  dxay7, dxay[6], dxay[5], dxay[4],
 
3627
                  dxay[3], dxay[2], dxay[1], dxay[0]);
 
3628
  dxay[7] = dxay7;
 
3629
  negate = -dey;
 
3630
  negatetail = -deytail;
 
3631
  Two_Two_Product(aex, aextail, negate, negatetail,
 
3632
                  axdy7, axdy[6], axdy[5], axdy[4],
 
3633
                  axdy[3], axdy[2], axdy[1], axdy[0]);
 
3634
  axdy[7] = axdy7;
 
3635
  dalen = fast_expansion_sum_zeroelim(8, dxay, 8, axdy, da);
 
3636
  Two_Two_Product(aex, aextail, cey, ceytail,
 
3637
                  axcy7, axcy[6], axcy[5], axcy[4],
 
3638
                  axcy[3], axcy[2], axcy[1], axcy[0]);
 
3639
  axcy[7] = axcy7;
 
3640
  negate = -aey;
 
3641
  negatetail = -aeytail;
 
3642
  Two_Two_Product(cex, cextail, negate, negatetail,
 
3643
                  cxay7, cxay[6], cxay[5], cxay[4],
 
3644
                  cxay[3], cxay[2], cxay[1], cxay[0]);
 
3645
  cxay[7] = cxay7;
 
3646
  aclen = fast_expansion_sum_zeroelim(8, axcy, 8, cxay, ac);
 
3647
  Two_Two_Product(bex, bextail, dey, deytail,
 
3648
                  bxdy7, bxdy[6], bxdy[5], bxdy[4],
 
3649
                  bxdy[3], bxdy[2], bxdy[1], bxdy[0]);
 
3650
  bxdy[7] = bxdy7;
 
3651
  negate = -bey;
 
3652
  negatetail = -beytail;
 
3653
  Two_Two_Product(dex, dextail, negate, negatetail,
 
3654
                  dxby7, dxby[6], dxby[5], dxby[4],
 
3655
                  dxby[3], dxby[2], dxby[1], dxby[0]);
 
3656
  dxby[7] = dxby7;
 
3657
  bdlen = fast_expansion_sum_zeroelim(8, bxdy, 8, dxby, bd);
 
3658
 
 
3659
  temp32alen = scale_expansion_zeroelim(cdlen, cd, -bez, temp32a);
 
3660
  temp32blen = scale_expansion_zeroelim(cdlen, cd, -beztail, temp32b);
 
3661
  temp64alen = fast_expansion_sum_zeroelim(temp32alen, temp32a,
 
3662
                                           temp32blen, temp32b, temp64a);
 
3663
  temp32alen = scale_expansion_zeroelim(bdlen, bd, cez, temp32a);
 
3664
  temp32blen = scale_expansion_zeroelim(bdlen, bd, ceztail, temp32b);
 
3665
  temp64blen = fast_expansion_sum_zeroelim(temp32alen, temp32a,
 
3666
                                           temp32blen, temp32b, temp64b);
 
3667
  temp32alen = scale_expansion_zeroelim(bclen, bc, -dez, temp32a);
 
3668
  temp32blen = scale_expansion_zeroelim(bclen, bc, -deztail, temp32b);
 
3669
  temp64clen = fast_expansion_sum_zeroelim(temp32alen, temp32a,
 
3670
                                           temp32blen, temp32b, temp64c);
 
3671
  temp128len = fast_expansion_sum_zeroelim(temp64alen, temp64a,
 
3672
                                           temp64blen, temp64b, temp128);
 
3673
  temp192len = fast_expansion_sum_zeroelim(temp64clen, temp64c,
 
3674
                                           temp128len, temp128, temp192);
 
3675
  xlen = scale_expansion_zeroelim(temp192len, temp192, aex, detx);
 
3676
  xxlen = scale_expansion_zeroelim(xlen, detx, aex, detxx);
 
3677
  xtlen = scale_expansion_zeroelim(temp192len, temp192, aextail, detxt);
 
3678
  xxtlen = scale_expansion_zeroelim(xtlen, detxt, aex, detxxt);
 
3679
  for (i = 0; i < xxtlen; i++) {
 
3680
    detxxt[i] *= 2.0;
 
3681
  }
 
3682
  xtxtlen = scale_expansion_zeroelim(xtlen, detxt, aextail, detxtxt);
 
3683
  x1len = fast_expansion_sum_zeroelim(xxlen, detxx, xxtlen, detxxt, x1);
 
3684
  x2len = fast_expansion_sum_zeroelim(x1len, x1, xtxtlen, detxtxt, x2);
 
3685
  ylen = scale_expansion_zeroelim(temp192len, temp192, aey, dety);
 
3686
  yylen = scale_expansion_zeroelim(ylen, dety, aey, detyy);
 
3687
  ytlen = scale_expansion_zeroelim(temp192len, temp192, aeytail, detyt);
 
3688
  yytlen = scale_expansion_zeroelim(ytlen, detyt, aey, detyyt);
 
3689
  for (i = 0; i < yytlen; i++) {
 
3690
    detyyt[i] *= 2.0;
 
3691
  }
 
3692
  ytytlen = scale_expansion_zeroelim(ytlen, detyt, aeytail, detytyt);
 
3693
  y1len = fast_expansion_sum_zeroelim(yylen, detyy, yytlen, detyyt, y1);
 
3694
  y2len = fast_expansion_sum_zeroelim(y1len, y1, ytytlen, detytyt, y2);
 
3695
  zlen = scale_expansion_zeroelim(temp192len, temp192, aez, detz);
 
3696
  zzlen = scale_expansion_zeroelim(zlen, detz, aez, detzz);
 
3697
  ztlen = scale_expansion_zeroelim(temp192len, temp192, aeztail, detzt);
 
3698
  zztlen = scale_expansion_zeroelim(ztlen, detzt, aez, detzzt);
 
3699
  for (i = 0; i < zztlen; i++) {
 
3700
    detzzt[i] *= 2.0;
 
3701
  }
 
3702
  ztztlen = scale_expansion_zeroelim(ztlen, detzt, aeztail, detztzt);
 
3703
  z1len = fast_expansion_sum_zeroelim(zzlen, detzz, zztlen, detzzt, z1);
 
3704
  z2len = fast_expansion_sum_zeroelim(z1len, z1, ztztlen, detztzt, z2);
 
3705
  xylen = fast_expansion_sum_zeroelim(x2len, x2, y2len, y2, detxy);
 
3706
  alen = fast_expansion_sum_zeroelim(z2len, z2, xylen, detxy, adet);
 
3707
 
 
3708
  temp32alen = scale_expansion_zeroelim(dalen, da, cez, temp32a);
 
3709
  temp32blen = scale_expansion_zeroelim(dalen, da, ceztail, temp32b);
 
3710
  temp64alen = fast_expansion_sum_zeroelim(temp32alen, temp32a,
 
3711
                                           temp32blen, temp32b, temp64a);
 
3712
  temp32alen = scale_expansion_zeroelim(aclen, ac, dez, temp32a);
 
3713
  temp32blen = scale_expansion_zeroelim(aclen, ac, deztail, temp32b);
 
3714
  temp64blen = fast_expansion_sum_zeroelim(temp32alen, temp32a,
 
3715
                                           temp32blen, temp32b, temp64b);
 
3716
  temp32alen = scale_expansion_zeroelim(cdlen, cd, aez, temp32a);
 
3717
  temp32blen = scale_expansion_zeroelim(cdlen, cd, aeztail, temp32b);
 
3718
  temp64clen = fast_expansion_sum_zeroelim(temp32alen, temp32a,
 
3719
                                           temp32blen, temp32b, temp64c);
 
3720
  temp128len = fast_expansion_sum_zeroelim(temp64alen, temp64a,
 
3721
                                           temp64blen, temp64b, temp128);
 
3722
  temp192len = fast_expansion_sum_zeroelim(temp64clen, temp64c,
 
3723
                                           temp128len, temp128, temp192);
 
3724
  xlen = scale_expansion_zeroelim(temp192len, temp192, bex, detx);
 
3725
  xxlen = scale_expansion_zeroelim(xlen, detx, bex, detxx);
 
3726
  xtlen = scale_expansion_zeroelim(temp192len, temp192, bextail, detxt);
 
3727
  xxtlen = scale_expansion_zeroelim(xtlen, detxt, bex, detxxt);
 
3728
  for (i = 0; i < xxtlen; i++) {
 
3729
    detxxt[i] *= 2.0;
 
3730
  }
 
3731
  xtxtlen = scale_expansion_zeroelim(xtlen, detxt, bextail, detxtxt);
 
3732
  x1len = fast_expansion_sum_zeroelim(xxlen, detxx, xxtlen, detxxt, x1);
 
3733
  x2len = fast_expansion_sum_zeroelim(x1len, x1, xtxtlen, detxtxt, x2);
 
3734
  ylen = scale_expansion_zeroelim(temp192len, temp192, bey, dety);
 
3735
  yylen = scale_expansion_zeroelim(ylen, dety, bey, detyy);
 
3736
  ytlen = scale_expansion_zeroelim(temp192len, temp192, beytail, detyt);
 
3737
  yytlen = scale_expansion_zeroelim(ytlen, detyt, bey, detyyt);
 
3738
  for (i = 0; i < yytlen; i++) {
 
3739
    detyyt[i] *= 2.0;
 
3740
  }
 
3741
  ytytlen = scale_expansion_zeroelim(ytlen, detyt, beytail, detytyt);
 
3742
  y1len = fast_expansion_sum_zeroelim(yylen, detyy, yytlen, detyyt, y1);
 
3743
  y2len = fast_expansion_sum_zeroelim(y1len, y1, ytytlen, detytyt, y2);
 
3744
  zlen = scale_expansion_zeroelim(temp192len, temp192, bez, detz);
 
3745
  zzlen = scale_expansion_zeroelim(zlen, detz, bez, detzz);
 
3746
  ztlen = scale_expansion_zeroelim(temp192len, temp192, beztail, detzt);
 
3747
  zztlen = scale_expansion_zeroelim(ztlen, detzt, bez, detzzt);
 
3748
  for (i = 0; i < zztlen; i++) {
 
3749
    detzzt[i] *= 2.0;
 
3750
  }
 
3751
  ztztlen = scale_expansion_zeroelim(ztlen, detzt, beztail, detztzt);
 
3752
  z1len = fast_expansion_sum_zeroelim(zzlen, detzz, zztlen, detzzt, z1);
 
3753
  z2len = fast_expansion_sum_zeroelim(z1len, z1, ztztlen, detztzt, z2);
 
3754
  xylen = fast_expansion_sum_zeroelim(x2len, x2, y2len, y2, detxy);
 
3755
  blen = fast_expansion_sum_zeroelim(z2len, z2, xylen, detxy, bdet);
 
3756
 
 
3757
  temp32alen = scale_expansion_zeroelim(ablen, ab, -dez, temp32a);
 
3758
  temp32blen = scale_expansion_zeroelim(ablen, ab, -deztail, temp32b);
 
3759
  temp64alen = fast_expansion_sum_zeroelim(temp32alen, temp32a,
 
3760
                                           temp32blen, temp32b, temp64a);
 
3761
  temp32alen = scale_expansion_zeroelim(bdlen, bd, -aez, temp32a);
 
3762
  temp32blen = scale_expansion_zeroelim(bdlen, bd, -aeztail, temp32b);
 
3763
  temp64blen = fast_expansion_sum_zeroelim(temp32alen, temp32a,
 
3764
                                           temp32blen, temp32b, temp64b);
 
3765
  temp32alen = scale_expansion_zeroelim(dalen, da, -bez, temp32a);
 
3766
  temp32blen = scale_expansion_zeroelim(dalen, da, -beztail, temp32b);
 
3767
  temp64clen = fast_expansion_sum_zeroelim(temp32alen, temp32a,
 
3768
                                           temp32blen, temp32b, temp64c);
 
3769
  temp128len = fast_expansion_sum_zeroelim(temp64alen, temp64a,
 
3770
                                           temp64blen, temp64b, temp128);
 
3771
  temp192len = fast_expansion_sum_zeroelim(temp64clen, temp64c,
 
3772
                                           temp128len, temp128, temp192);
 
3773
  xlen = scale_expansion_zeroelim(temp192len, temp192, cex, detx);
 
3774
  xxlen = scale_expansion_zeroelim(xlen, detx, cex, detxx);
 
3775
  xtlen = scale_expansion_zeroelim(temp192len, temp192, cextail, detxt);
 
3776
  xxtlen = scale_expansion_zeroelim(xtlen, detxt, cex, detxxt);
 
3777
  for (i = 0; i < xxtlen; i++) {
 
3778
    detxxt[i] *= 2.0;
 
3779
  }
 
3780
  xtxtlen = scale_expansion_zeroelim(xtlen, detxt, cextail, detxtxt);
 
3781
  x1len = fast_expansion_sum_zeroelim(xxlen, detxx, xxtlen, detxxt, x1);
 
3782
  x2len = fast_expansion_sum_zeroelim(x1len, x1, xtxtlen, detxtxt, x2);
 
3783
  ylen = scale_expansion_zeroelim(temp192len, temp192, cey, dety);
 
3784
  yylen = scale_expansion_zeroelim(ylen, dety, cey, detyy);
 
3785
  ytlen = scale_expansion_zeroelim(temp192len, temp192, ceytail, detyt);
 
3786
  yytlen = scale_expansion_zeroelim(ytlen, detyt, cey, detyyt);
 
3787
  for (i = 0; i < yytlen; i++) {
 
3788
    detyyt[i] *= 2.0;
 
3789
  }
 
3790
  ytytlen = scale_expansion_zeroelim(ytlen, detyt, ceytail, detytyt);
 
3791
  y1len = fast_expansion_sum_zeroelim(yylen, detyy, yytlen, detyyt, y1);
 
3792
  y2len = fast_expansion_sum_zeroelim(y1len, y1, ytytlen, detytyt, y2);
 
3793
  zlen = scale_expansion_zeroelim(temp192len, temp192, cez, detz);
 
3794
  zzlen = scale_expansion_zeroelim(zlen, detz, cez, detzz);
 
3795
  ztlen = scale_expansion_zeroelim(temp192len, temp192, ceztail, detzt);
 
3796
  zztlen = scale_expansion_zeroelim(ztlen, detzt, cez, detzzt);
 
3797
  for (i = 0; i < zztlen; i++) {
 
3798
    detzzt[i] *= 2.0;
 
3799
  }
 
3800
  ztztlen = scale_expansion_zeroelim(ztlen, detzt, ceztail, detztzt);
 
3801
  z1len = fast_expansion_sum_zeroelim(zzlen, detzz, zztlen, detzzt, z1);
 
3802
  z2len = fast_expansion_sum_zeroelim(z1len, z1, ztztlen, detztzt, z2);
 
3803
  xylen = fast_expansion_sum_zeroelim(x2len, x2, y2len, y2, detxy);
 
3804
  clen = fast_expansion_sum_zeroelim(z2len, z2, xylen, detxy, cdet);
 
3805
 
 
3806
  temp32alen = scale_expansion_zeroelim(bclen, bc, aez, temp32a);
 
3807
  temp32blen = scale_expansion_zeroelim(bclen, bc, aeztail, temp32b);
 
3808
  temp64alen = fast_expansion_sum_zeroelim(temp32alen, temp32a,
 
3809
                                           temp32blen, temp32b, temp64a);
 
3810
  temp32alen = scale_expansion_zeroelim(aclen, ac, -bez, temp32a);
 
3811
  temp32blen = scale_expansion_zeroelim(aclen, ac, -beztail, temp32b);
 
3812
  temp64blen = fast_expansion_sum_zeroelim(temp32alen, temp32a,
 
3813
                                           temp32blen, temp32b, temp64b);
 
3814
  temp32alen = scale_expansion_zeroelim(ablen, ab, cez, temp32a);
 
3815
  temp32blen = scale_expansion_zeroelim(ablen, ab, ceztail, temp32b);
 
3816
  temp64clen = fast_expansion_sum_zeroelim(temp32alen, temp32a,
 
3817
                                           temp32blen, temp32b, temp64c);
 
3818
  temp128len = fast_expansion_sum_zeroelim(temp64alen, temp64a,
 
3819
                                           temp64blen, temp64b, temp128);
 
3820
  temp192len = fast_expansion_sum_zeroelim(temp64clen, temp64c,
 
3821
                                           temp128len, temp128, temp192);
 
3822
  xlen = scale_expansion_zeroelim(temp192len, temp192, dex, detx);
 
3823
  xxlen = scale_expansion_zeroelim(xlen, detx, dex, detxx);
 
3824
  xtlen = scale_expansion_zeroelim(temp192len, temp192, dextail, detxt);
 
3825
  xxtlen = scale_expansion_zeroelim(xtlen, detxt, dex, detxxt);
 
3826
  for (i = 0; i < xxtlen; i++) {
 
3827
    detxxt[i] *= 2.0;
 
3828
  }
 
3829
  xtxtlen = scale_expansion_zeroelim(xtlen, detxt, dextail, detxtxt);
 
3830
  x1len = fast_expansion_sum_zeroelim(xxlen, detxx, xxtlen, detxxt, x1);
 
3831
  x2len = fast_expansion_sum_zeroelim(x1len, x1, xtxtlen, detxtxt, x2);
 
3832
  ylen = scale_expansion_zeroelim(temp192len, temp192, dey, dety);
 
3833
  yylen = scale_expansion_zeroelim(ylen, dety, dey, detyy);
 
3834
  ytlen = scale_expansion_zeroelim(temp192len, temp192, deytail, detyt);
 
3835
  yytlen = scale_expansion_zeroelim(ytlen, detyt, dey, detyyt);
 
3836
  for (i = 0; i < yytlen; i++) {
 
3837
    detyyt[i] *= 2.0;
 
3838
  }
 
3839
  ytytlen = scale_expansion_zeroelim(ytlen, detyt, deytail, detytyt);
 
3840
  y1len = fast_expansion_sum_zeroelim(yylen, detyy, yytlen, detyyt, y1);
 
3841
  y2len = fast_expansion_sum_zeroelim(y1len, y1, ytytlen, detytyt, y2);
 
3842
  zlen = scale_expansion_zeroelim(temp192len, temp192, dez, detz);
 
3843
  zzlen = scale_expansion_zeroelim(zlen, detz, dez, detzz);
 
3844
  ztlen = scale_expansion_zeroelim(temp192len, temp192, deztail, detzt);
 
3845
  zztlen = scale_expansion_zeroelim(ztlen, detzt, dez, detzzt);
 
3846
  for (i = 0; i < zztlen; i++) {
 
3847
    detzzt[i] *= 2.0;
 
3848
  }
 
3849
  ztztlen = scale_expansion_zeroelim(ztlen, detzt, deztail, detztzt);
 
3850
  z1len = fast_expansion_sum_zeroelim(zzlen, detzz, zztlen, detzzt, z1);
 
3851
  z2len = fast_expansion_sum_zeroelim(z1len, z1, ztztlen, detztzt, z2);
 
3852
  xylen = fast_expansion_sum_zeroelim(x2len, x2, y2len, y2, detxy);
 
3853
  dlen = fast_expansion_sum_zeroelim(z2len, z2, xylen, detxy, ddet);
 
3854
 
 
3855
  ablen = fast_expansion_sum_zeroelim(alen, adet, blen, bdet, abdet);
 
3856
  cdlen = fast_expansion_sum_zeroelim(clen, cdet, dlen, ddet, cddet);
 
3857
  deterlen = fast_expansion_sum_zeroelim(ablen, abdet, cdlen, cddet, deter);
 
3858
 
 
3859
  return deter[deterlen - 1];
 
3860
}
 
3861
 
 
3862
REAL insphereadapt(REAL *pa, REAL *pb, REAL *pc, REAL *pd, REAL *pe,
 
3863
                   REAL permanent)
 
3864
{
 
3865
  INEXACT REAL aex, bex, cex, dex, aey, bey, cey, dey, aez, bez, cez, dez;
 
3866
  REAL det, errbound;
 
3867
 
 
3868
  INEXACT REAL aexbey1, bexaey1, bexcey1, cexbey1;
 
3869
  INEXACT REAL cexdey1, dexcey1, dexaey1, aexdey1;
 
3870
  INEXACT REAL aexcey1, cexaey1, bexdey1, dexbey1;
 
3871
  REAL aexbey0, bexaey0, bexcey0, cexbey0;
 
3872
  REAL cexdey0, dexcey0, dexaey0, aexdey0;
 
3873
  REAL aexcey0, cexaey0, bexdey0, dexbey0;
 
3874
  REAL ab[4], bc[4], cd[4], da[4], ac[4], bd[4];
 
3875
  INEXACT REAL ab3, bc3, cd3, da3, ac3, bd3;
 
3876
  REAL abeps, bceps, cdeps, daeps, aceps, bdeps;
 
3877
  REAL temp8a[8], temp8b[8], temp8c[8], temp16[16], temp24[24], temp48[48];
 
3878
  int temp8alen, temp8blen, temp8clen, temp16len, temp24len, temp48len;
 
3879
  REAL xdet[96], ydet[96], zdet[96], xydet[192];
 
3880
  int xlen, ylen, zlen, xylen;
 
3881
  REAL adet[288], bdet[288], cdet[288], ddet[288];
 
3882
  int alen, blen, clen, dlen;
 
3883
  REAL abdet[576], cddet[576];
 
3884
  int ablen, cdlen;
 
3885
  REAL fin1[1152];
 
3886
  int finlength;
 
3887
 
 
3888
  REAL aextail, bextail, cextail, dextail;
 
3889
  REAL aeytail, beytail, ceytail, deytail;
 
3890
  REAL aeztail, beztail, ceztail, deztail;
 
3891
 
 
3892
  INEXACT REAL bvirt;
 
3893
  REAL avirt, bround, around;
 
3894
  INEXACT REAL c;
 
3895
  INEXACT REAL abig;
 
3896
  REAL ahi, alo, bhi, blo;
 
3897
  REAL err1, err2, err3;
 
3898
  INEXACT REAL _i, _j;
 
3899
  REAL _0;
 
3900
 
 
3901
  aex = (REAL) (pa[0] - pe[0]);
 
3902
  bex = (REAL) (pb[0] - pe[0]);
 
3903
  cex = (REAL) (pc[0] - pe[0]);
 
3904
  dex = (REAL) (pd[0] - pe[0]);
 
3905
  aey = (REAL) (pa[1] - pe[1]);
 
3906
  bey = (REAL) (pb[1] - pe[1]);
 
3907
  cey = (REAL) (pc[1] - pe[1]);
 
3908
  dey = (REAL) (pd[1] - pe[1]);
 
3909
  aez = (REAL) (pa[2] - pe[2]);
 
3910
  bez = (REAL) (pb[2] - pe[2]);
 
3911
  cez = (REAL) (pc[2] - pe[2]);
 
3912
  dez = (REAL) (pd[2] - pe[2]);
 
3913
 
 
3914
  Two_Product(aex, bey, aexbey1, aexbey0);
 
3915
  Two_Product(bex, aey, bexaey1, bexaey0);
 
3916
  Two_Two_Diff(aexbey1, aexbey0, bexaey1, bexaey0, ab3, ab[2], ab[1], ab[0]);
 
3917
  ab[3] = ab3;
 
3918
 
 
3919
  Two_Product(bex, cey, bexcey1, bexcey0);
 
3920
  Two_Product(cex, bey, cexbey1, cexbey0);
 
3921
  Two_Two_Diff(bexcey1, bexcey0, cexbey1, cexbey0, bc3, bc[2], bc[1], bc[0]);
 
3922
  bc[3] = bc3;
 
3923
 
 
3924
  Two_Product(cex, dey, cexdey1, cexdey0);
 
3925
  Two_Product(dex, cey, dexcey1, dexcey0);
 
3926
  Two_Two_Diff(cexdey1, cexdey0, dexcey1, dexcey0, cd3, cd[2], cd[1], cd[0]);
 
3927
  cd[3] = cd3;
 
3928
 
 
3929
  Two_Product(dex, aey, dexaey1, dexaey0);
 
3930
  Two_Product(aex, dey, aexdey1, aexdey0);
 
3931
  Two_Two_Diff(dexaey1, dexaey0, aexdey1, aexdey0, da3, da[2], da[1], da[0]);
 
3932
  da[3] = da3;
 
3933
 
 
3934
  Two_Product(aex, cey, aexcey1, aexcey0);
 
3935
  Two_Product(cex, aey, cexaey1, cexaey0);
 
3936
  Two_Two_Diff(aexcey1, aexcey0, cexaey1, cexaey0, ac3, ac[2], ac[1], ac[0]);
 
3937
  ac[3] = ac3;
 
3938
 
 
3939
  Two_Product(bex, dey, bexdey1, bexdey0);
 
3940
  Two_Product(dex, bey, dexbey1, dexbey0);
 
3941
  Two_Two_Diff(bexdey1, bexdey0, dexbey1, dexbey0, bd3, bd[2], bd[1], bd[0]);
 
3942
  bd[3] = bd3;
 
3943
 
 
3944
  temp8alen = scale_expansion_zeroelim(4, cd, bez, temp8a);
 
3945
  temp8blen = scale_expansion_zeroelim(4, bd, -cez, temp8b);
 
3946
  temp8clen = scale_expansion_zeroelim(4, bc, dez, temp8c);
 
3947
  temp16len = fast_expansion_sum_zeroelim(temp8alen, temp8a,
 
3948
                                          temp8blen, temp8b, temp16);
 
3949
  temp24len = fast_expansion_sum_zeroelim(temp8clen, temp8c,
 
3950
                                          temp16len, temp16, temp24);
 
3951
  temp48len = scale_expansion_zeroelim(temp24len, temp24, aex, temp48);
 
3952
  xlen = scale_expansion_zeroelim(temp48len, temp48, -aex, xdet);
 
3953
  temp48len = scale_expansion_zeroelim(temp24len, temp24, aey, temp48);
 
3954
  ylen = scale_expansion_zeroelim(temp48len, temp48, -aey, ydet);
 
3955
  temp48len = scale_expansion_zeroelim(temp24len, temp24, aez, temp48);
 
3956
  zlen = scale_expansion_zeroelim(temp48len, temp48, -aez, zdet);
 
3957
  xylen = fast_expansion_sum_zeroelim(xlen, xdet, ylen, ydet, xydet);
 
3958
  alen = fast_expansion_sum_zeroelim(xylen, xydet, zlen, zdet, adet);
 
3959
 
 
3960
  temp8alen = scale_expansion_zeroelim(4, da, cez, temp8a);
 
3961
  temp8blen = scale_expansion_zeroelim(4, ac, dez, temp8b);
 
3962
  temp8clen = scale_expansion_zeroelim(4, cd, aez, temp8c);
 
3963
  temp16len = fast_expansion_sum_zeroelim(temp8alen, temp8a,
 
3964
                                          temp8blen, temp8b, temp16);
 
3965
  temp24len = fast_expansion_sum_zeroelim(temp8clen, temp8c,
 
3966
                                          temp16len, temp16, temp24);
 
3967
  temp48len = scale_expansion_zeroelim(temp24len, temp24, bex, temp48);
 
3968
  xlen = scale_expansion_zeroelim(temp48len, temp48, bex, xdet);
 
3969
  temp48len = scale_expansion_zeroelim(temp24len, temp24, bey, temp48);
 
3970
  ylen = scale_expansion_zeroelim(temp48len, temp48, bey, ydet);
 
3971
  temp48len = scale_expansion_zeroelim(temp24len, temp24, bez, temp48);
 
3972
  zlen = scale_expansion_zeroelim(temp48len, temp48, bez, zdet);
 
3973
  xylen = fast_expansion_sum_zeroelim(xlen, xdet, ylen, ydet, xydet);
 
3974
  blen = fast_expansion_sum_zeroelim(xylen, xydet, zlen, zdet, bdet);
 
3975
 
 
3976
  temp8alen = scale_expansion_zeroelim(4, ab, dez, temp8a);
 
3977
  temp8blen = scale_expansion_zeroelim(4, bd, aez, temp8b);
 
3978
  temp8clen = scale_expansion_zeroelim(4, da, bez, temp8c);
 
3979
  temp16len = fast_expansion_sum_zeroelim(temp8alen, temp8a,
 
3980
                                          temp8blen, temp8b, temp16);
 
3981
  temp24len = fast_expansion_sum_zeroelim(temp8clen, temp8c,
 
3982
                                          temp16len, temp16, temp24);
 
3983
  temp48len = scale_expansion_zeroelim(temp24len, temp24, cex, temp48);
 
3984
  xlen = scale_expansion_zeroelim(temp48len, temp48, -cex, xdet);
 
3985
  temp48len = scale_expansion_zeroelim(temp24len, temp24, cey, temp48);
 
3986
  ylen = scale_expansion_zeroelim(temp48len, temp48, -cey, ydet);
 
3987
  temp48len = scale_expansion_zeroelim(temp24len, temp24, cez, temp48);
 
3988
  zlen = scale_expansion_zeroelim(temp48len, temp48, -cez, zdet);
 
3989
  xylen = fast_expansion_sum_zeroelim(xlen, xdet, ylen, ydet, xydet);
 
3990
  clen = fast_expansion_sum_zeroelim(xylen, xydet, zlen, zdet, cdet);
 
3991
 
 
3992
  temp8alen = scale_expansion_zeroelim(4, bc, aez, temp8a);
 
3993
  temp8blen = scale_expansion_zeroelim(4, ac, -bez, temp8b);
 
3994
  temp8clen = scale_expansion_zeroelim(4, ab, cez, temp8c);
 
3995
  temp16len = fast_expansion_sum_zeroelim(temp8alen, temp8a,
 
3996
                                          temp8blen, temp8b, temp16);
 
3997
  temp24len = fast_expansion_sum_zeroelim(temp8clen, temp8c,
 
3998
                                          temp16len, temp16, temp24);
 
3999
  temp48len = scale_expansion_zeroelim(temp24len, temp24, dex, temp48);
 
4000
  xlen = scale_expansion_zeroelim(temp48len, temp48, dex, xdet);
 
4001
  temp48len = scale_expansion_zeroelim(temp24len, temp24, dey, temp48);
 
4002
  ylen = scale_expansion_zeroelim(temp48len, temp48, dey, ydet);
 
4003
  temp48len = scale_expansion_zeroelim(temp24len, temp24, dez, temp48);
 
4004
  zlen = scale_expansion_zeroelim(temp48len, temp48, dez, zdet);
 
4005
  xylen = fast_expansion_sum_zeroelim(xlen, xdet, ylen, ydet, xydet);
 
4006
  dlen = fast_expansion_sum_zeroelim(xylen, xydet, zlen, zdet, ddet);
 
4007
 
 
4008
  ablen = fast_expansion_sum_zeroelim(alen, adet, blen, bdet, abdet);
 
4009
  cdlen = fast_expansion_sum_zeroelim(clen, cdet, dlen, ddet, cddet);
 
4010
  finlength = fast_expansion_sum_zeroelim(ablen, abdet, cdlen, cddet, fin1);
 
4011
 
 
4012
  det = estimate(finlength, fin1);
 
4013
  errbound = isperrboundB * permanent;
 
4014
  if ((det >= errbound) || (-det >= errbound)) {
 
4015
    return det;
 
4016
  }
 
4017
 
 
4018
  Two_Diff_Tail(pa[0], pe[0], aex, aextail);
 
4019
  Two_Diff_Tail(pa[1], pe[1], aey, aeytail);
 
4020
  Two_Diff_Tail(pa[2], pe[2], aez, aeztail);
 
4021
  Two_Diff_Tail(pb[0], pe[0], bex, bextail);
 
4022
  Two_Diff_Tail(pb[1], pe[1], bey, beytail);
 
4023
  Two_Diff_Tail(pb[2], pe[2], bez, beztail);
 
4024
  Two_Diff_Tail(pc[0], pe[0], cex, cextail);
 
4025
  Two_Diff_Tail(pc[1], pe[1], cey, ceytail);
 
4026
  Two_Diff_Tail(pc[2], pe[2], cez, ceztail);
 
4027
  Two_Diff_Tail(pd[0], pe[0], dex, dextail);
 
4028
  Two_Diff_Tail(pd[1], pe[1], dey, deytail);
 
4029
  Two_Diff_Tail(pd[2], pe[2], dez, deztail);
 
4030
  if ((aextail == 0.0) && (aeytail == 0.0) && (aeztail == 0.0)
 
4031
      && (bextail == 0.0) && (beytail == 0.0) && (beztail == 0.0)
 
4032
      && (cextail == 0.0) && (ceytail == 0.0) && (ceztail == 0.0)
 
4033
      && (dextail == 0.0) && (deytail == 0.0) && (deztail == 0.0)) {
 
4034
    return det;
 
4035
  }
 
4036
 
 
4037
  errbound = isperrboundC * permanent + resulterrbound * Absolute(det);
 
4038
  abeps = (aex * beytail + bey * aextail)
 
4039
        - (aey * bextail + bex * aeytail);
 
4040
  bceps = (bex * ceytail + cey * bextail)
 
4041
        - (bey * cextail + cex * beytail);
 
4042
  cdeps = (cex * deytail + dey * cextail)
 
4043
        - (cey * dextail + dex * ceytail);
 
4044
  daeps = (dex * aeytail + aey * dextail)
 
4045
        - (dey * aextail + aex * deytail);
 
4046
  aceps = (aex * ceytail + cey * aextail)
 
4047
        - (aey * cextail + cex * aeytail);
 
4048
  bdeps = (bex * deytail + dey * bextail)
 
4049
        - (bey * dextail + dex * beytail);
 
4050
  det += (((bex * bex + bey * bey + bez * bez)
 
4051
           * ((cez * daeps + dez * aceps + aez * cdeps)
 
4052
              + (ceztail * da3 + deztail * ac3 + aeztail * cd3))
 
4053
           + (dex * dex + dey * dey + dez * dez)
 
4054
           * ((aez * bceps - bez * aceps + cez * abeps)
 
4055
              + (aeztail * bc3 - beztail * ac3 + ceztail * ab3)))
 
4056
          - ((aex * aex + aey * aey + aez * aez)
 
4057
           * ((bez * cdeps - cez * bdeps + dez * bceps)
 
4058
              + (beztail * cd3 - ceztail * bd3 + deztail * bc3))
 
4059
           + (cex * cex + cey * cey + cez * cez)
 
4060
           * ((dez * abeps + aez * bdeps + bez * daeps)
 
4061
              + (deztail * ab3 + aeztail * bd3 + beztail * da3))))
 
4062
       + 2.0 * (((bex * bextail + bey * beytail + bez * beztail)
 
4063
                 * (cez * da3 + dez * ac3 + aez * cd3)
 
4064
                 + (dex * dextail + dey * deytail + dez * deztail)
 
4065
                 * (aez * bc3 - bez * ac3 + cez * ab3))
 
4066
                - ((aex * aextail + aey * aeytail + aez * aeztail)
 
4067
                 * (bez * cd3 - cez * bd3 + dez * bc3)
 
4068
                 + (cex * cextail + cey * ceytail + cez * ceztail)
 
4069
                 * (dez * ab3 + aez * bd3 + bez * da3)));
 
4070
  if ((det >= errbound) || (-det >= errbound)) {
 
4071
    return det;
 
4072
  }
 
4073
 
 
4074
  return insphereexact(pa, pb, pc, pd, pe);
 
4075
}
 
4076
 
 
4077
REAL insphere(REAL *pa, REAL *pb, REAL *pc, REAL *pd, REAL *pe)
 
4078
{
 
4079
  REAL aex, bex, cex, dex;
 
4080
  REAL aey, bey, cey, dey;
 
4081
  REAL aez, bez, cez, dez;
 
4082
  REAL aexbey, bexaey, bexcey, cexbey, cexdey, dexcey, dexaey, aexdey;
 
4083
  REAL aexcey, cexaey, bexdey, dexbey;
 
4084
  REAL alift, blift, clift, dlift;
 
4085
  REAL ab, bc, cd, da, ac, bd;
 
4086
  REAL abc, bcd, cda, dab;
 
4087
  REAL aezplus, bezplus, cezplus, dezplus;
 
4088
  REAL aexbeyplus, bexaeyplus, bexceyplus, cexbeyplus;
 
4089
  REAL cexdeyplus, dexceyplus, dexaeyplus, aexdeyplus;
 
4090
  REAL aexceyplus, cexaeyplus, bexdeyplus, dexbeyplus;
 
4091
  REAL det;
 
4092
  REAL permanent, errbound;
 
4093
 
 
4094
  aex = pa[0] - pe[0];
 
4095
  bex = pb[0] - pe[0];
 
4096
  cex = pc[0] - pe[0];
 
4097
  dex = pd[0] - pe[0];
 
4098
  aey = pa[1] - pe[1];
 
4099
  bey = pb[1] - pe[1];
 
4100
  cey = pc[1] - pe[1];
 
4101
  dey = pd[1] - pe[1];
 
4102
  aez = pa[2] - pe[2];
 
4103
  bez = pb[2] - pe[2];
 
4104
  cez = pc[2] - pe[2];
 
4105
  dez = pd[2] - pe[2];
 
4106
 
 
4107
  aexbey = aex * bey;
 
4108
  bexaey = bex * aey;
 
4109
  ab = aexbey - bexaey;
 
4110
  bexcey = bex * cey;
 
4111
  cexbey = cex * bey;
 
4112
  bc = bexcey - cexbey;
 
4113
  cexdey = cex * dey;
 
4114
  dexcey = dex * cey;
 
4115
  cd = cexdey - dexcey;
 
4116
  dexaey = dex * aey;
 
4117
  aexdey = aex * dey;
 
4118
  da = dexaey - aexdey;
 
4119
 
 
4120
  aexcey = aex * cey;
 
4121
  cexaey = cex * aey;
 
4122
  ac = aexcey - cexaey;
 
4123
  bexdey = bex * dey;
 
4124
  dexbey = dex * bey;
 
4125
  bd = bexdey - dexbey;
 
4126
 
 
4127
  abc = aez * bc - bez * ac + cez * ab;
 
4128
  bcd = bez * cd - cez * bd + dez * bc;
 
4129
  cda = cez * da + dez * ac + aez * cd;
 
4130
  dab = dez * ab + aez * bd + bez * da;
 
4131
 
 
4132
  alift = aex * aex + aey * aey + aez * aez;
 
4133
  blift = bex * bex + bey * bey + bez * bez;
 
4134
  clift = cex * cex + cey * cey + cez * cez;
 
4135
  dlift = dex * dex + dey * dey + dez * dez;
 
4136
 
 
4137
  det = (dlift * abc - clift * dab) + (blift * cda - alift * bcd);
 
4138
 
 
4139
  aezplus = Absolute(aez);
 
4140
  bezplus = Absolute(bez);
 
4141
  cezplus = Absolute(cez);
 
4142
  dezplus = Absolute(dez);
 
4143
  aexbeyplus = Absolute(aexbey);
 
4144
  bexaeyplus = Absolute(bexaey);
 
4145
  bexceyplus = Absolute(bexcey);
 
4146
  cexbeyplus = Absolute(cexbey);
 
4147
  cexdeyplus = Absolute(cexdey);
 
4148
  dexceyplus = Absolute(dexcey);
 
4149
  dexaeyplus = Absolute(dexaey);
 
4150
  aexdeyplus = Absolute(aexdey);
 
4151
  aexceyplus = Absolute(aexcey);
 
4152
  cexaeyplus = Absolute(cexaey);
 
4153
  bexdeyplus = Absolute(bexdey);
 
4154
  dexbeyplus = Absolute(dexbey);
 
4155
  permanent = ((cexdeyplus + dexceyplus) * bezplus
 
4156
               + (dexbeyplus + bexdeyplus) * cezplus
 
4157
               + (bexceyplus + cexbeyplus) * dezplus)
 
4158
            * alift
 
4159
            + ((dexaeyplus + aexdeyplus) * cezplus
 
4160
               + (aexceyplus + cexaeyplus) * dezplus
 
4161
               + (cexdeyplus + dexceyplus) * aezplus)
 
4162
            * blift
 
4163
            + ((aexbeyplus + bexaeyplus) * dezplus
 
4164
               + (bexdeyplus + dexbeyplus) * aezplus
 
4165
               + (dexaeyplus + aexdeyplus) * bezplus)
 
4166
            * clift
 
4167
            + ((bexceyplus + cexbeyplus) * aezplus
 
4168
               + (cexaeyplus + aexceyplus) * bezplus
 
4169
               + (aexbeyplus + bexaeyplus) * cezplus)
 
4170
            * dlift;
 
4171
  errbound = isperrboundA * permanent;
 
4172
  if ((det > errbound) || (-det > errbound)) {
 
4173
    return det;
 
4174
  }
 
4175
 
 
4176
  return insphereadapt(pa, pb, pc, pd, pe, permanent);
 
4177
}
 
4178
 
 
4179
} // namespace robustPredicates