annotate libgcc/fp-bit.c @ 118:fd00160c1b76

ifdef TARGET_64BIT
author mir3636
date Tue, 27 Feb 2018 15:01:35 +0900
parents 04ced10e8804
children 84e7813d76e9
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
111
kono
parents:
diff changeset
1 /* This is a software floating point library which can be used
kono
parents:
diff changeset
2 for targets without hardware floating point.
kono
parents:
diff changeset
3 Copyright (C) 1994-2017 Free Software Foundation, Inc.
kono
parents:
diff changeset
4
kono
parents:
diff changeset
5 This file is part of GCC.
kono
parents:
diff changeset
6
kono
parents:
diff changeset
7 GCC is free software; you can redistribute it and/or modify it under
kono
parents:
diff changeset
8 the terms of the GNU General Public License as published by the Free
kono
parents:
diff changeset
9 Software Foundation; either version 3, or (at your option) any later
kono
parents:
diff changeset
10 version.
kono
parents:
diff changeset
11
kono
parents:
diff changeset
12 GCC is distributed in the hope that it will be useful, but WITHOUT ANY
kono
parents:
diff changeset
13 WARRANTY; without even the implied warranty of MERCHANTABILITY or
kono
parents:
diff changeset
14 FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
kono
parents:
diff changeset
15 for more details.
kono
parents:
diff changeset
16
kono
parents:
diff changeset
17 Under Section 7 of GPL version 3, you are granted additional
kono
parents:
diff changeset
18 permissions described in the GCC Runtime Library Exception, version
kono
parents:
diff changeset
19 3.1, as published by the Free Software Foundation.
kono
parents:
diff changeset
20
kono
parents:
diff changeset
21 You should have received a copy of the GNU General Public License and
kono
parents:
diff changeset
22 a copy of the GCC Runtime Library Exception along with this program;
kono
parents:
diff changeset
23 see the files COPYING3 and COPYING.RUNTIME respectively. If not, see
kono
parents:
diff changeset
24 <http://www.gnu.org/licenses/>. */
kono
parents:
diff changeset
25
kono
parents:
diff changeset
26 /* This implements IEEE 754 format arithmetic, but does not provide a
kono
parents:
diff changeset
27 mechanism for setting the rounding mode, or for generating or handling
kono
parents:
diff changeset
28 exceptions.
kono
parents:
diff changeset
29
kono
parents:
diff changeset
30 The original code by Steve Chamberlain, hacked by Mark Eichin and Jim
kono
parents:
diff changeset
31 Wilson, all of Cygnus Support. */
kono
parents:
diff changeset
32
kono
parents:
diff changeset
33 /* The intended way to use this file is to make two copies, add `#define FLOAT'
kono
parents:
diff changeset
34 to one copy, then compile both copies and add them to libgcc.a. */
kono
parents:
diff changeset
35
kono
parents:
diff changeset
36 #include "tconfig.h"
kono
parents:
diff changeset
37 #include "coretypes.h"
kono
parents:
diff changeset
38 #include "tm.h"
kono
parents:
diff changeset
39 #include "libgcc_tm.h"
kono
parents:
diff changeset
40 #include "fp-bit.h"
kono
parents:
diff changeset
41
kono
parents:
diff changeset
42 /* The following macros can be defined to change the behavior of this file:
kono
parents:
diff changeset
43 FLOAT: Implement a `float', aka SFmode, fp library. If this is not
kono
parents:
diff changeset
44 defined, then this file implements a `double', aka DFmode, fp library.
kono
parents:
diff changeset
45 FLOAT_ONLY: Used with FLOAT, to implement a `float' only library, i.e.
kono
parents:
diff changeset
46 don't include float->double conversion which requires the double library.
kono
parents:
diff changeset
47 This is useful only for machines which can't support doubles, e.g. some
kono
parents:
diff changeset
48 8-bit processors.
kono
parents:
diff changeset
49 CMPtype: Specify the type that floating point compares should return.
kono
parents:
diff changeset
50 This defaults to SItype, aka int.
kono
parents:
diff changeset
51 _DEBUG_BITFLOAT: This makes debugging the code a little easier, by adding
kono
parents:
diff changeset
52 two integers to the FLO_union_type.
kono
parents:
diff changeset
53 NO_DENORMALS: Disable handling of denormals.
kono
parents:
diff changeset
54 NO_NANS: Disable nan and infinity handling
kono
parents:
diff changeset
55 SMALL_MACHINE: Useful when operations on QIs and HIs are faster
kono
parents:
diff changeset
56 than on an SI */
kono
parents:
diff changeset
57
kono
parents:
diff changeset
58 /* We don't currently support extended floats (long doubles) on machines
kono
parents:
diff changeset
59 without hardware to deal with them.
kono
parents:
diff changeset
60
kono
parents:
diff changeset
61 These stubs are just to keep the linker from complaining about unresolved
kono
parents:
diff changeset
62 references which can be pulled in from libio & libstdc++, even if the
kono
parents:
diff changeset
63 user isn't using long doubles. However, they may generate an unresolved
kono
parents:
diff changeset
64 external to abort if abort is not used by the function, and the stubs
kono
parents:
diff changeset
65 are referenced from within libc, since libgcc goes before and after the
kono
parents:
diff changeset
66 system library. */
kono
parents:
diff changeset
67
kono
parents:
diff changeset
68 #ifdef DECLARE_LIBRARY_RENAMES
kono
parents:
diff changeset
69 DECLARE_LIBRARY_RENAMES
kono
parents:
diff changeset
70 #endif
kono
parents:
diff changeset
71
kono
parents:
diff changeset
72 #ifdef EXTENDED_FLOAT_STUBS
kono
parents:
diff changeset
73 extern void abort (void);
kono
parents:
diff changeset
74 void __extendsfxf2 (void) { abort(); }
kono
parents:
diff changeset
75 void __extenddfxf2 (void) { abort(); }
kono
parents:
diff changeset
76 void __truncxfdf2 (void) { abort(); }
kono
parents:
diff changeset
77 void __truncxfsf2 (void) { abort(); }
kono
parents:
diff changeset
78 void __fixxfsi (void) { abort(); }
kono
parents:
diff changeset
79 void __floatsixf (void) { abort(); }
kono
parents:
diff changeset
80 void __addxf3 (void) { abort(); }
kono
parents:
diff changeset
81 void __subxf3 (void) { abort(); }
kono
parents:
diff changeset
82 void __mulxf3 (void) { abort(); }
kono
parents:
diff changeset
83 void __divxf3 (void) { abort(); }
kono
parents:
diff changeset
84 void __negxf2 (void) { abort(); }
kono
parents:
diff changeset
85 void __eqxf2 (void) { abort(); }
kono
parents:
diff changeset
86 void __nexf2 (void) { abort(); }
kono
parents:
diff changeset
87 void __gtxf2 (void) { abort(); }
kono
parents:
diff changeset
88 void __gexf2 (void) { abort(); }
kono
parents:
diff changeset
89 void __lexf2 (void) { abort(); }
kono
parents:
diff changeset
90 void __ltxf2 (void) { abort(); }
kono
parents:
diff changeset
91
kono
parents:
diff changeset
92 void __extendsftf2 (void) { abort(); }
kono
parents:
diff changeset
93 void __extenddftf2 (void) { abort(); }
kono
parents:
diff changeset
94 void __trunctfdf2 (void) { abort(); }
kono
parents:
diff changeset
95 void __trunctfsf2 (void) { abort(); }
kono
parents:
diff changeset
96 void __fixtfsi (void) { abort(); }
kono
parents:
diff changeset
97 void __floatsitf (void) { abort(); }
kono
parents:
diff changeset
98 void __addtf3 (void) { abort(); }
kono
parents:
diff changeset
99 void __subtf3 (void) { abort(); }
kono
parents:
diff changeset
100 void __multf3 (void) { abort(); }
kono
parents:
diff changeset
101 void __divtf3 (void) { abort(); }
kono
parents:
diff changeset
102 void __negtf2 (void) { abort(); }
kono
parents:
diff changeset
103 void __eqtf2 (void) { abort(); }
kono
parents:
diff changeset
104 void __netf2 (void) { abort(); }
kono
parents:
diff changeset
105 void __gttf2 (void) { abort(); }
kono
parents:
diff changeset
106 void __getf2 (void) { abort(); }
kono
parents:
diff changeset
107 void __letf2 (void) { abort(); }
kono
parents:
diff changeset
108 void __lttf2 (void) { abort(); }
kono
parents:
diff changeset
109 #else /* !EXTENDED_FLOAT_STUBS, rest of file */
kono
parents:
diff changeset
110
kono
parents:
diff changeset
111 /* IEEE "special" number predicates */
kono
parents:
diff changeset
112
kono
parents:
diff changeset
113 #ifdef NO_NANS
kono
parents:
diff changeset
114
kono
parents:
diff changeset
115 #define nan() 0
kono
parents:
diff changeset
116 #define isnan(x) 0
kono
parents:
diff changeset
117 #define isinf(x) 0
kono
parents:
diff changeset
118 #else
kono
parents:
diff changeset
119
kono
parents:
diff changeset
120 #if defined L_thenan_sf
kono
parents:
diff changeset
121 const fp_number_type __thenan_sf = { CLASS_SNAN, 0, 0, {(fractype) 0} };
kono
parents:
diff changeset
122 #elif defined L_thenan_df
kono
parents:
diff changeset
123 const fp_number_type __thenan_df = { CLASS_SNAN, 0, 0, {(fractype) 0} };
kono
parents:
diff changeset
124 #elif defined L_thenan_tf
kono
parents:
diff changeset
125 const fp_number_type __thenan_tf = { CLASS_SNAN, 0, 0, {(fractype) 0} };
kono
parents:
diff changeset
126 #elif defined TFLOAT
kono
parents:
diff changeset
127 extern const fp_number_type __thenan_tf;
kono
parents:
diff changeset
128 #elif defined FLOAT
kono
parents:
diff changeset
129 extern const fp_number_type __thenan_sf;
kono
parents:
diff changeset
130 #else
kono
parents:
diff changeset
131 extern const fp_number_type __thenan_df;
kono
parents:
diff changeset
132 #endif
kono
parents:
diff changeset
133
kono
parents:
diff changeset
134 INLINE
kono
parents:
diff changeset
135 static const fp_number_type *
kono
parents:
diff changeset
136 makenan (void)
kono
parents:
diff changeset
137 {
kono
parents:
diff changeset
138 #ifdef TFLOAT
kono
parents:
diff changeset
139 return & __thenan_tf;
kono
parents:
diff changeset
140 #elif defined FLOAT
kono
parents:
diff changeset
141 return & __thenan_sf;
kono
parents:
diff changeset
142 #else
kono
parents:
diff changeset
143 return & __thenan_df;
kono
parents:
diff changeset
144 #endif
kono
parents:
diff changeset
145 }
kono
parents:
diff changeset
146
kono
parents:
diff changeset
147 INLINE
kono
parents:
diff changeset
148 static int
kono
parents:
diff changeset
149 isnan (const fp_number_type *x)
kono
parents:
diff changeset
150 {
kono
parents:
diff changeset
151 return __builtin_expect (x->class == CLASS_SNAN || x->class == CLASS_QNAN,
kono
parents:
diff changeset
152 0);
kono
parents:
diff changeset
153 }
kono
parents:
diff changeset
154
kono
parents:
diff changeset
155 INLINE
kono
parents:
diff changeset
156 static int
kono
parents:
diff changeset
157 isinf (const fp_number_type * x)
kono
parents:
diff changeset
158 {
kono
parents:
diff changeset
159 return __builtin_expect (x->class == CLASS_INFINITY, 0);
kono
parents:
diff changeset
160 }
kono
parents:
diff changeset
161
kono
parents:
diff changeset
162 #endif /* NO_NANS */
kono
parents:
diff changeset
163
kono
parents:
diff changeset
164 INLINE
kono
parents:
diff changeset
165 static int
kono
parents:
diff changeset
166 iszero (const fp_number_type * x)
kono
parents:
diff changeset
167 {
kono
parents:
diff changeset
168 return x->class == CLASS_ZERO;
kono
parents:
diff changeset
169 }
kono
parents:
diff changeset
170
kono
parents:
diff changeset
171 INLINE
kono
parents:
diff changeset
172 static void
kono
parents:
diff changeset
173 flip_sign ( fp_number_type * x)
kono
parents:
diff changeset
174 {
kono
parents:
diff changeset
175 x->sign = !x->sign;
kono
parents:
diff changeset
176 }
kono
parents:
diff changeset
177
kono
parents:
diff changeset
178 /* Count leading zeroes in N. */
kono
parents:
diff changeset
179 INLINE
kono
parents:
diff changeset
180 static int
kono
parents:
diff changeset
181 clzusi (USItype n)
kono
parents:
diff changeset
182 {
kono
parents:
diff changeset
183 extern int __clzsi2 (USItype);
kono
parents:
diff changeset
184 if (sizeof (USItype) == sizeof (unsigned int))
kono
parents:
diff changeset
185 return __builtin_clz (n);
kono
parents:
diff changeset
186 else if (sizeof (USItype) == sizeof (unsigned long))
kono
parents:
diff changeset
187 return __builtin_clzl (n);
kono
parents:
diff changeset
188 else if (sizeof (USItype) == sizeof (unsigned long long))
kono
parents:
diff changeset
189 return __builtin_clzll (n);
kono
parents:
diff changeset
190 else
kono
parents:
diff changeset
191 return __clzsi2 (n);
kono
parents:
diff changeset
192 }
kono
parents:
diff changeset
193
kono
parents:
diff changeset
194 extern FLO_type pack_d (const fp_number_type * );
kono
parents:
diff changeset
195
kono
parents:
diff changeset
196 #if defined(L_pack_df) || defined(L_pack_sf) || defined(L_pack_tf)
kono
parents:
diff changeset
197 FLO_type
kono
parents:
diff changeset
198 pack_d (const fp_number_type *src)
kono
parents:
diff changeset
199 {
kono
parents:
diff changeset
200 FLO_union_type dst;
kono
parents:
diff changeset
201 fractype fraction = src->fraction.ll; /* wasn't unsigned before? */
kono
parents:
diff changeset
202 int sign = src->sign;
kono
parents:
diff changeset
203 int exp = 0;
kono
parents:
diff changeset
204
kono
parents:
diff changeset
205 if (isnan (src))
kono
parents:
diff changeset
206 {
kono
parents:
diff changeset
207 exp = EXPMAX;
kono
parents:
diff changeset
208 /* Restore the NaN's payload. */
kono
parents:
diff changeset
209 fraction >>= NGARDS;
kono
parents:
diff changeset
210 fraction &= QUIET_NAN - 1;
kono
parents:
diff changeset
211 if (src->class == CLASS_QNAN || 1)
kono
parents:
diff changeset
212 {
kono
parents:
diff changeset
213 #ifdef QUIET_NAN_NEGATED
kono
parents:
diff changeset
214 /* The quiet/signaling bit remains unset. */
kono
parents:
diff changeset
215 /* Make sure the fraction has a non-zero value. */
kono
parents:
diff changeset
216 if (fraction == 0)
kono
parents:
diff changeset
217 fraction |= QUIET_NAN - 1;
kono
parents:
diff changeset
218 #else
kono
parents:
diff changeset
219 /* Set the quiet/signaling bit. */
kono
parents:
diff changeset
220 fraction |= QUIET_NAN;
kono
parents:
diff changeset
221 #endif
kono
parents:
diff changeset
222 }
kono
parents:
diff changeset
223 }
kono
parents:
diff changeset
224 else if (isinf (src))
kono
parents:
diff changeset
225 {
kono
parents:
diff changeset
226 exp = EXPMAX;
kono
parents:
diff changeset
227 fraction = 0;
kono
parents:
diff changeset
228 }
kono
parents:
diff changeset
229 else if (iszero (src))
kono
parents:
diff changeset
230 {
kono
parents:
diff changeset
231 exp = 0;
kono
parents:
diff changeset
232 fraction = 0;
kono
parents:
diff changeset
233 }
kono
parents:
diff changeset
234 else if (fraction == 0)
kono
parents:
diff changeset
235 {
kono
parents:
diff changeset
236 exp = 0;
kono
parents:
diff changeset
237 }
kono
parents:
diff changeset
238 else
kono
parents:
diff changeset
239 {
kono
parents:
diff changeset
240 if (__builtin_expect (src->normal_exp < NORMAL_EXPMIN, 0))
kono
parents:
diff changeset
241 {
kono
parents:
diff changeset
242 #ifdef NO_DENORMALS
kono
parents:
diff changeset
243 /* Go straight to a zero representation if denormals are not
kono
parents:
diff changeset
244 supported. The denormal handling would be harmless but
kono
parents:
diff changeset
245 isn't unnecessary. */
kono
parents:
diff changeset
246 exp = 0;
kono
parents:
diff changeset
247 fraction = 0;
kono
parents:
diff changeset
248 #else /* NO_DENORMALS */
kono
parents:
diff changeset
249 /* This number's exponent is too low to fit into the bits
kono
parents:
diff changeset
250 available in the number, so we'll store 0 in the exponent and
kono
parents:
diff changeset
251 shift the fraction to the right to make up for it. */
kono
parents:
diff changeset
252
kono
parents:
diff changeset
253 int shift = NORMAL_EXPMIN - src->normal_exp;
kono
parents:
diff changeset
254
kono
parents:
diff changeset
255 exp = 0;
kono
parents:
diff changeset
256
kono
parents:
diff changeset
257 if (shift > FRAC_NBITS - NGARDS)
kono
parents:
diff changeset
258 {
kono
parents:
diff changeset
259 /* No point shifting, since it's more that 64 out. */
kono
parents:
diff changeset
260 fraction = 0;
kono
parents:
diff changeset
261 }
kono
parents:
diff changeset
262 else
kono
parents:
diff changeset
263 {
kono
parents:
diff changeset
264 int lowbit = (fraction & (((fractype)1 << shift) - 1)) ? 1 : 0;
kono
parents:
diff changeset
265 fraction = (fraction >> shift) | lowbit;
kono
parents:
diff changeset
266 }
kono
parents:
diff changeset
267 if ((fraction & GARDMASK) == GARDMSB)
kono
parents:
diff changeset
268 {
kono
parents:
diff changeset
269 if ((fraction & (1 << NGARDS)))
kono
parents:
diff changeset
270 fraction += GARDROUND + 1;
kono
parents:
diff changeset
271 }
kono
parents:
diff changeset
272 else
kono
parents:
diff changeset
273 {
kono
parents:
diff changeset
274 /* Add to the guards to round up. */
kono
parents:
diff changeset
275 fraction += GARDROUND;
kono
parents:
diff changeset
276 }
kono
parents:
diff changeset
277 /* Perhaps the rounding means we now need to change the
kono
parents:
diff changeset
278 exponent, because the fraction is no longer denormal. */
kono
parents:
diff changeset
279 if (fraction >= IMPLICIT_1)
kono
parents:
diff changeset
280 {
kono
parents:
diff changeset
281 exp += 1;
kono
parents:
diff changeset
282 }
kono
parents:
diff changeset
283 fraction >>= NGARDS;
kono
parents:
diff changeset
284 #endif /* NO_DENORMALS */
kono
parents:
diff changeset
285 }
kono
parents:
diff changeset
286 else if (__builtin_expect (src->normal_exp > EXPBIAS, 0))
kono
parents:
diff changeset
287 {
kono
parents:
diff changeset
288 exp = EXPMAX;
kono
parents:
diff changeset
289 fraction = 0;
kono
parents:
diff changeset
290 }
kono
parents:
diff changeset
291 else
kono
parents:
diff changeset
292 {
kono
parents:
diff changeset
293 exp = src->normal_exp + EXPBIAS;
kono
parents:
diff changeset
294 /* IF the gard bits are the all zero, but the first, then we're
kono
parents:
diff changeset
295 half way between two numbers, choose the one which makes the
kono
parents:
diff changeset
296 lsb of the answer 0. */
kono
parents:
diff changeset
297 if ((fraction & GARDMASK) == GARDMSB)
kono
parents:
diff changeset
298 {
kono
parents:
diff changeset
299 if (fraction & (1 << NGARDS))
kono
parents:
diff changeset
300 fraction += GARDROUND + 1;
kono
parents:
diff changeset
301 }
kono
parents:
diff changeset
302 else
kono
parents:
diff changeset
303 {
kono
parents:
diff changeset
304 /* Add a one to the guards to round up */
kono
parents:
diff changeset
305 fraction += GARDROUND;
kono
parents:
diff changeset
306 }
kono
parents:
diff changeset
307 if (fraction >= IMPLICIT_2)
kono
parents:
diff changeset
308 {
kono
parents:
diff changeset
309 fraction >>= 1;
kono
parents:
diff changeset
310 exp += 1;
kono
parents:
diff changeset
311 }
kono
parents:
diff changeset
312 fraction >>= NGARDS;
kono
parents:
diff changeset
313 }
kono
parents:
diff changeset
314 }
kono
parents:
diff changeset
315
kono
parents:
diff changeset
316 /* We previously used bitfields to store the number, but this doesn't
kono
parents:
diff changeset
317 handle little/big endian systems conveniently, so use shifts and
kono
parents:
diff changeset
318 masks */
kono
parents:
diff changeset
319 #ifdef FLOAT_BIT_ORDER_MISMATCH
kono
parents:
diff changeset
320 dst.bits.fraction = fraction;
kono
parents:
diff changeset
321 dst.bits.exp = exp;
kono
parents:
diff changeset
322 dst.bits.sign = sign;
kono
parents:
diff changeset
323 #else
kono
parents:
diff changeset
324 # if defined TFLOAT && defined HALFFRACBITS
kono
parents:
diff changeset
325 {
kono
parents:
diff changeset
326 halffractype high, low, unity;
kono
parents:
diff changeset
327 int lowsign, lowexp;
kono
parents:
diff changeset
328
kono
parents:
diff changeset
329 unity = (halffractype) 1 << HALFFRACBITS;
kono
parents:
diff changeset
330
kono
parents:
diff changeset
331 /* Set HIGH to the high double's significand, masking out the implicit 1.
kono
parents:
diff changeset
332 Set LOW to the low double's full significand. */
kono
parents:
diff changeset
333 high = (fraction >> (FRACBITS - HALFFRACBITS)) & (unity - 1);
kono
parents:
diff changeset
334 low = fraction & (unity * 2 - 1);
kono
parents:
diff changeset
335
kono
parents:
diff changeset
336 /* Get the initial sign and exponent of the low double. */
kono
parents:
diff changeset
337 lowexp = exp - HALFFRACBITS - 1;
kono
parents:
diff changeset
338 lowsign = sign;
kono
parents:
diff changeset
339
kono
parents:
diff changeset
340 /* HIGH should be rounded like a normal double, making |LOW| <=
kono
parents:
diff changeset
341 0.5 ULP of HIGH. Assume round-to-nearest. */
kono
parents:
diff changeset
342 if (exp < EXPMAX)
kono
parents:
diff changeset
343 if (low > unity || (low == unity && (high & 1) == 1))
kono
parents:
diff changeset
344 {
kono
parents:
diff changeset
345 /* Round HIGH up and adjust LOW to match. */
kono
parents:
diff changeset
346 high++;
kono
parents:
diff changeset
347 if (high == unity)
kono
parents:
diff changeset
348 {
kono
parents:
diff changeset
349 /* May make it infinite, but that's OK. */
kono
parents:
diff changeset
350 high = 0;
kono
parents:
diff changeset
351 exp++;
kono
parents:
diff changeset
352 }
kono
parents:
diff changeset
353 low = unity * 2 - low;
kono
parents:
diff changeset
354 lowsign ^= 1;
kono
parents:
diff changeset
355 }
kono
parents:
diff changeset
356
kono
parents:
diff changeset
357 high |= (halffractype) exp << HALFFRACBITS;
kono
parents:
diff changeset
358 high |= (halffractype) sign << (HALFFRACBITS + EXPBITS);
kono
parents:
diff changeset
359
kono
parents:
diff changeset
360 if (exp == EXPMAX || exp == 0 || low == 0)
kono
parents:
diff changeset
361 low = 0;
kono
parents:
diff changeset
362 else
kono
parents:
diff changeset
363 {
kono
parents:
diff changeset
364 while (lowexp > 0 && low < unity)
kono
parents:
diff changeset
365 {
kono
parents:
diff changeset
366 low <<= 1;
kono
parents:
diff changeset
367 lowexp--;
kono
parents:
diff changeset
368 }
kono
parents:
diff changeset
369
kono
parents:
diff changeset
370 if (lowexp <= 0)
kono
parents:
diff changeset
371 {
kono
parents:
diff changeset
372 halffractype roundmsb, round;
kono
parents:
diff changeset
373 int shift;
kono
parents:
diff changeset
374
kono
parents:
diff changeset
375 shift = 1 - lowexp;
kono
parents:
diff changeset
376 roundmsb = (1 << (shift - 1));
kono
parents:
diff changeset
377 round = low & ((roundmsb << 1) - 1);
kono
parents:
diff changeset
378
kono
parents:
diff changeset
379 low >>= shift;
kono
parents:
diff changeset
380 lowexp = 0;
kono
parents:
diff changeset
381
kono
parents:
diff changeset
382 if (round > roundmsb || (round == roundmsb && (low & 1) == 1))
kono
parents:
diff changeset
383 {
kono
parents:
diff changeset
384 low++;
kono
parents:
diff changeset
385 if (low == unity)
kono
parents:
diff changeset
386 /* LOW rounds up to the smallest normal number. */
kono
parents:
diff changeset
387 lowexp++;
kono
parents:
diff changeset
388 }
kono
parents:
diff changeset
389 }
kono
parents:
diff changeset
390
kono
parents:
diff changeset
391 low &= unity - 1;
kono
parents:
diff changeset
392 low |= (halffractype) lowexp << HALFFRACBITS;
kono
parents:
diff changeset
393 low |= (halffractype) lowsign << (HALFFRACBITS + EXPBITS);
kono
parents:
diff changeset
394 }
kono
parents:
diff changeset
395 dst.value_raw = ((fractype) high << HALFSHIFT) | low;
kono
parents:
diff changeset
396 }
kono
parents:
diff changeset
397 # else
kono
parents:
diff changeset
398 dst.value_raw = fraction & ((((fractype)1) << FRACBITS) - (fractype)1);
kono
parents:
diff changeset
399 dst.value_raw |= ((fractype) (exp & ((1 << EXPBITS) - 1))) << FRACBITS;
kono
parents:
diff changeset
400 dst.value_raw |= ((fractype) (sign & 1)) << (FRACBITS | EXPBITS);
kono
parents:
diff changeset
401 # endif
kono
parents:
diff changeset
402 #endif
kono
parents:
diff changeset
403
kono
parents:
diff changeset
404 #if defined(FLOAT_WORD_ORDER_MISMATCH) && !defined(FLOAT)
kono
parents:
diff changeset
405 #ifdef TFLOAT
kono
parents:
diff changeset
406 {
kono
parents:
diff changeset
407 qrtrfractype tmp1 = dst.words[0];
kono
parents:
diff changeset
408 qrtrfractype tmp2 = dst.words[1];
kono
parents:
diff changeset
409 dst.words[0] = dst.words[3];
kono
parents:
diff changeset
410 dst.words[1] = dst.words[2];
kono
parents:
diff changeset
411 dst.words[2] = tmp2;
kono
parents:
diff changeset
412 dst.words[3] = tmp1;
kono
parents:
diff changeset
413 }
kono
parents:
diff changeset
414 #else
kono
parents:
diff changeset
415 {
kono
parents:
diff changeset
416 halffractype tmp = dst.words[0];
kono
parents:
diff changeset
417 dst.words[0] = dst.words[1];
kono
parents:
diff changeset
418 dst.words[1] = tmp;
kono
parents:
diff changeset
419 }
kono
parents:
diff changeset
420 #endif
kono
parents:
diff changeset
421 #endif
kono
parents:
diff changeset
422
kono
parents:
diff changeset
423 return dst.value;
kono
parents:
diff changeset
424 }
kono
parents:
diff changeset
425 #endif
kono
parents:
diff changeset
426
kono
parents:
diff changeset
427 #if defined(L_unpack_df) || defined(L_unpack_sf) || defined(L_unpack_tf)
kono
parents:
diff changeset
428 void
kono
parents:
diff changeset
429 unpack_d (FLO_union_type * src, fp_number_type * dst)
kono
parents:
diff changeset
430 {
kono
parents:
diff changeset
431 /* We previously used bitfields to store the number, but this doesn't
kono
parents:
diff changeset
432 handle little/big endian systems conveniently, so use shifts and
kono
parents:
diff changeset
433 masks */
kono
parents:
diff changeset
434 fractype fraction;
kono
parents:
diff changeset
435 int exp;
kono
parents:
diff changeset
436 int sign;
kono
parents:
diff changeset
437
kono
parents:
diff changeset
438 #if defined(FLOAT_WORD_ORDER_MISMATCH) && !defined(FLOAT)
kono
parents:
diff changeset
439 FLO_union_type swapped;
kono
parents:
diff changeset
440
kono
parents:
diff changeset
441 #ifdef TFLOAT
kono
parents:
diff changeset
442 swapped.words[0] = src->words[3];
kono
parents:
diff changeset
443 swapped.words[1] = src->words[2];
kono
parents:
diff changeset
444 swapped.words[2] = src->words[1];
kono
parents:
diff changeset
445 swapped.words[3] = src->words[0];
kono
parents:
diff changeset
446 #else
kono
parents:
diff changeset
447 swapped.words[0] = src->words[1];
kono
parents:
diff changeset
448 swapped.words[1] = src->words[0];
kono
parents:
diff changeset
449 #endif
kono
parents:
diff changeset
450 src = &swapped;
kono
parents:
diff changeset
451 #endif
kono
parents:
diff changeset
452
kono
parents:
diff changeset
453 #ifdef FLOAT_BIT_ORDER_MISMATCH
kono
parents:
diff changeset
454 fraction = src->bits.fraction;
kono
parents:
diff changeset
455 exp = src->bits.exp;
kono
parents:
diff changeset
456 sign = src->bits.sign;
kono
parents:
diff changeset
457 #else
kono
parents:
diff changeset
458 # if defined TFLOAT && defined HALFFRACBITS
kono
parents:
diff changeset
459 {
kono
parents:
diff changeset
460 halffractype high, low;
kono
parents:
diff changeset
461
kono
parents:
diff changeset
462 high = src->value_raw >> HALFSHIFT;
kono
parents:
diff changeset
463 low = src->value_raw & (((fractype)1 << HALFSHIFT) - 1);
kono
parents:
diff changeset
464
kono
parents:
diff changeset
465 fraction = high & ((((fractype)1) << HALFFRACBITS) - 1);
kono
parents:
diff changeset
466 fraction <<= FRACBITS - HALFFRACBITS;
kono
parents:
diff changeset
467 exp = ((int)(high >> HALFFRACBITS)) & ((1 << EXPBITS) - 1);
kono
parents:
diff changeset
468 sign = ((int)(high >> (((HALFFRACBITS + EXPBITS))))) & 1;
kono
parents:
diff changeset
469
kono
parents:
diff changeset
470 if (exp != EXPMAX && exp != 0 && low != 0)
kono
parents:
diff changeset
471 {
kono
parents:
diff changeset
472 int lowexp = ((int)(low >> HALFFRACBITS)) & ((1 << EXPBITS) - 1);
kono
parents:
diff changeset
473 int lowsign = ((int)(low >> (((HALFFRACBITS + EXPBITS))))) & 1;
kono
parents:
diff changeset
474 int shift;
kono
parents:
diff changeset
475 fractype xlow;
kono
parents:
diff changeset
476
kono
parents:
diff changeset
477 xlow = low & ((((fractype)1) << HALFFRACBITS) - 1);
kono
parents:
diff changeset
478 if (lowexp)
kono
parents:
diff changeset
479 xlow |= (((halffractype)1) << HALFFRACBITS);
kono
parents:
diff changeset
480 else
kono
parents:
diff changeset
481 lowexp = 1;
kono
parents:
diff changeset
482 shift = (FRACBITS - HALFFRACBITS) - (exp - lowexp);
kono
parents:
diff changeset
483 if (shift > 0)
kono
parents:
diff changeset
484 xlow <<= shift;
kono
parents:
diff changeset
485 else if (shift < 0)
kono
parents:
diff changeset
486 xlow >>= -shift;
kono
parents:
diff changeset
487 if (sign == lowsign)
kono
parents:
diff changeset
488 fraction += xlow;
kono
parents:
diff changeset
489 else if (fraction >= xlow)
kono
parents:
diff changeset
490 fraction -= xlow;
kono
parents:
diff changeset
491 else
kono
parents:
diff changeset
492 {
kono
parents:
diff changeset
493 /* The high part is a power of two but the full number is lower.
kono
parents:
diff changeset
494 This code will leave the implicit 1 in FRACTION, but we'd
kono
parents:
diff changeset
495 have added that below anyway. */
kono
parents:
diff changeset
496 fraction = (((fractype) 1 << FRACBITS) - xlow) << 1;
kono
parents:
diff changeset
497 exp--;
kono
parents:
diff changeset
498 }
kono
parents:
diff changeset
499 }
kono
parents:
diff changeset
500 }
kono
parents:
diff changeset
501 # else
kono
parents:
diff changeset
502 fraction = src->value_raw & ((((fractype)1) << FRACBITS) - 1);
kono
parents:
diff changeset
503 exp = ((int)(src->value_raw >> FRACBITS)) & ((1 << EXPBITS) - 1);
kono
parents:
diff changeset
504 sign = ((int)(src->value_raw >> (FRACBITS + EXPBITS))) & 1;
kono
parents:
diff changeset
505 # endif
kono
parents:
diff changeset
506 #endif
kono
parents:
diff changeset
507
kono
parents:
diff changeset
508 dst->sign = sign;
kono
parents:
diff changeset
509 if (exp == 0)
kono
parents:
diff changeset
510 {
kono
parents:
diff changeset
511 /* Hmm. Looks like 0 */
kono
parents:
diff changeset
512 if (fraction == 0
kono
parents:
diff changeset
513 #ifdef NO_DENORMALS
kono
parents:
diff changeset
514 || 1
kono
parents:
diff changeset
515 #endif
kono
parents:
diff changeset
516 )
kono
parents:
diff changeset
517 {
kono
parents:
diff changeset
518 /* tastes like zero */
kono
parents:
diff changeset
519 dst->class = CLASS_ZERO;
kono
parents:
diff changeset
520 }
kono
parents:
diff changeset
521 else
kono
parents:
diff changeset
522 {
kono
parents:
diff changeset
523 /* Zero exponent with nonzero fraction - it's denormalized,
kono
parents:
diff changeset
524 so there isn't a leading implicit one - we'll shift it so
kono
parents:
diff changeset
525 it gets one. */
kono
parents:
diff changeset
526 dst->normal_exp = exp - EXPBIAS + 1;
kono
parents:
diff changeset
527 fraction <<= NGARDS;
kono
parents:
diff changeset
528
kono
parents:
diff changeset
529 dst->class = CLASS_NUMBER;
kono
parents:
diff changeset
530 #if 1
kono
parents:
diff changeset
531 while (fraction < IMPLICIT_1)
kono
parents:
diff changeset
532 {
kono
parents:
diff changeset
533 fraction <<= 1;
kono
parents:
diff changeset
534 dst->normal_exp--;
kono
parents:
diff changeset
535 }
kono
parents:
diff changeset
536 #endif
kono
parents:
diff changeset
537 dst->fraction.ll = fraction;
kono
parents:
diff changeset
538 }
kono
parents:
diff changeset
539 }
kono
parents:
diff changeset
540 else if (__builtin_expect (exp == EXPMAX, 0))
kono
parents:
diff changeset
541 {
kono
parents:
diff changeset
542 /* Huge exponent*/
kono
parents:
diff changeset
543 if (fraction == 0)
kono
parents:
diff changeset
544 {
kono
parents:
diff changeset
545 /* Attached to a zero fraction - means infinity */
kono
parents:
diff changeset
546 dst->class = CLASS_INFINITY;
kono
parents:
diff changeset
547 }
kono
parents:
diff changeset
548 else
kono
parents:
diff changeset
549 {
kono
parents:
diff changeset
550 /* Nonzero fraction, means nan */
kono
parents:
diff changeset
551 #ifdef QUIET_NAN_NEGATED
kono
parents:
diff changeset
552 if ((fraction & QUIET_NAN) == 0)
kono
parents:
diff changeset
553 #else
kono
parents:
diff changeset
554 if (fraction & QUIET_NAN)
kono
parents:
diff changeset
555 #endif
kono
parents:
diff changeset
556 {
kono
parents:
diff changeset
557 dst->class = CLASS_QNAN;
kono
parents:
diff changeset
558 }
kono
parents:
diff changeset
559 else
kono
parents:
diff changeset
560 {
kono
parents:
diff changeset
561 dst->class = CLASS_SNAN;
kono
parents:
diff changeset
562 }
kono
parents:
diff changeset
563 /* Now that we know which kind of NaN we got, discard the
kono
parents:
diff changeset
564 quiet/signaling bit, but do preserve the NaN payload. */
kono
parents:
diff changeset
565 fraction &= ~QUIET_NAN;
kono
parents:
diff changeset
566 dst->fraction.ll = fraction << NGARDS;
kono
parents:
diff changeset
567 }
kono
parents:
diff changeset
568 }
kono
parents:
diff changeset
569 else
kono
parents:
diff changeset
570 {
kono
parents:
diff changeset
571 /* Nothing strange about this number */
kono
parents:
diff changeset
572 dst->normal_exp = exp - EXPBIAS;
kono
parents:
diff changeset
573 dst->class = CLASS_NUMBER;
kono
parents:
diff changeset
574 dst->fraction.ll = (fraction << NGARDS) | IMPLICIT_1;
kono
parents:
diff changeset
575 }
kono
parents:
diff changeset
576 }
kono
parents:
diff changeset
577 #endif /* L_unpack_df || L_unpack_sf */
kono
parents:
diff changeset
578
kono
parents:
diff changeset
579 #if defined(L_addsub_sf) || defined(L_addsub_df) || defined(L_addsub_tf)
kono
parents:
diff changeset
580 static const fp_number_type *
kono
parents:
diff changeset
581 _fpadd_parts (fp_number_type * a,
kono
parents:
diff changeset
582 fp_number_type * b,
kono
parents:
diff changeset
583 fp_number_type * tmp)
kono
parents:
diff changeset
584 {
kono
parents:
diff changeset
585 intfrac tfraction;
kono
parents:
diff changeset
586
kono
parents:
diff changeset
587 /* Put commonly used fields in local variables. */
kono
parents:
diff changeset
588 int a_normal_exp;
kono
parents:
diff changeset
589 int b_normal_exp;
kono
parents:
diff changeset
590 fractype a_fraction;
kono
parents:
diff changeset
591 fractype b_fraction;
kono
parents:
diff changeset
592
kono
parents:
diff changeset
593 if (isnan (a))
kono
parents:
diff changeset
594 {
kono
parents:
diff changeset
595 return a;
kono
parents:
diff changeset
596 }
kono
parents:
diff changeset
597 if (isnan (b))
kono
parents:
diff changeset
598 {
kono
parents:
diff changeset
599 return b;
kono
parents:
diff changeset
600 }
kono
parents:
diff changeset
601 if (isinf (a))
kono
parents:
diff changeset
602 {
kono
parents:
diff changeset
603 /* Adding infinities with opposite signs yields a NaN. */
kono
parents:
diff changeset
604 if (isinf (b) && a->sign != b->sign)
kono
parents:
diff changeset
605 return makenan ();
kono
parents:
diff changeset
606 return a;
kono
parents:
diff changeset
607 }
kono
parents:
diff changeset
608 if (isinf (b))
kono
parents:
diff changeset
609 {
kono
parents:
diff changeset
610 return b;
kono
parents:
diff changeset
611 }
kono
parents:
diff changeset
612 if (iszero (b))
kono
parents:
diff changeset
613 {
kono
parents:
diff changeset
614 if (iszero (a))
kono
parents:
diff changeset
615 {
kono
parents:
diff changeset
616 *tmp = *a;
kono
parents:
diff changeset
617 tmp->sign = a->sign & b->sign;
kono
parents:
diff changeset
618 return tmp;
kono
parents:
diff changeset
619 }
kono
parents:
diff changeset
620 return a;
kono
parents:
diff changeset
621 }
kono
parents:
diff changeset
622 if (iszero (a))
kono
parents:
diff changeset
623 {
kono
parents:
diff changeset
624 return b;
kono
parents:
diff changeset
625 }
kono
parents:
diff changeset
626
kono
parents:
diff changeset
627 /* Got two numbers. shift the smaller and increment the exponent till
kono
parents:
diff changeset
628 they're the same */
kono
parents:
diff changeset
629 {
kono
parents:
diff changeset
630 int diff;
kono
parents:
diff changeset
631 int sdiff;
kono
parents:
diff changeset
632
kono
parents:
diff changeset
633 a_normal_exp = a->normal_exp;
kono
parents:
diff changeset
634 b_normal_exp = b->normal_exp;
kono
parents:
diff changeset
635 a_fraction = a->fraction.ll;
kono
parents:
diff changeset
636 b_fraction = b->fraction.ll;
kono
parents:
diff changeset
637
kono
parents:
diff changeset
638 diff = a_normal_exp - b_normal_exp;
kono
parents:
diff changeset
639 sdiff = diff;
kono
parents:
diff changeset
640
kono
parents:
diff changeset
641 if (diff < 0)
kono
parents:
diff changeset
642 diff = -diff;
kono
parents:
diff changeset
643 if (diff < FRAC_NBITS)
kono
parents:
diff changeset
644 {
kono
parents:
diff changeset
645 if (sdiff > 0)
kono
parents:
diff changeset
646 {
kono
parents:
diff changeset
647 b_normal_exp += diff;
kono
parents:
diff changeset
648 LSHIFT (b_fraction, diff);
kono
parents:
diff changeset
649 }
kono
parents:
diff changeset
650 else if (sdiff < 0)
kono
parents:
diff changeset
651 {
kono
parents:
diff changeset
652 a_normal_exp += diff;
kono
parents:
diff changeset
653 LSHIFT (a_fraction, diff);
kono
parents:
diff changeset
654 }
kono
parents:
diff changeset
655 }
kono
parents:
diff changeset
656 else
kono
parents:
diff changeset
657 {
kono
parents:
diff changeset
658 /* Somethings's up.. choose the biggest */
kono
parents:
diff changeset
659 if (a_normal_exp > b_normal_exp)
kono
parents:
diff changeset
660 {
kono
parents:
diff changeset
661 b_normal_exp = a_normal_exp;
kono
parents:
diff changeset
662 b_fraction = 0;
kono
parents:
diff changeset
663 }
kono
parents:
diff changeset
664 else
kono
parents:
diff changeset
665 {
kono
parents:
diff changeset
666 a_normal_exp = b_normal_exp;
kono
parents:
diff changeset
667 a_fraction = 0;
kono
parents:
diff changeset
668 }
kono
parents:
diff changeset
669 }
kono
parents:
diff changeset
670 }
kono
parents:
diff changeset
671
kono
parents:
diff changeset
672 if (a->sign != b->sign)
kono
parents:
diff changeset
673 {
kono
parents:
diff changeset
674 if (a->sign)
kono
parents:
diff changeset
675 {
kono
parents:
diff changeset
676 tfraction = -a_fraction + b_fraction;
kono
parents:
diff changeset
677 }
kono
parents:
diff changeset
678 else
kono
parents:
diff changeset
679 {
kono
parents:
diff changeset
680 tfraction = a_fraction - b_fraction;
kono
parents:
diff changeset
681 }
kono
parents:
diff changeset
682 if (tfraction >= 0)
kono
parents:
diff changeset
683 {
kono
parents:
diff changeset
684 tmp->sign = 0;
kono
parents:
diff changeset
685 tmp->normal_exp = a_normal_exp;
kono
parents:
diff changeset
686 tmp->fraction.ll = tfraction;
kono
parents:
diff changeset
687 }
kono
parents:
diff changeset
688 else
kono
parents:
diff changeset
689 {
kono
parents:
diff changeset
690 tmp->sign = 1;
kono
parents:
diff changeset
691 tmp->normal_exp = a_normal_exp;
kono
parents:
diff changeset
692 tmp->fraction.ll = -tfraction;
kono
parents:
diff changeset
693 }
kono
parents:
diff changeset
694 /* and renormalize it */
kono
parents:
diff changeset
695
kono
parents:
diff changeset
696 while (tmp->fraction.ll < IMPLICIT_1 && tmp->fraction.ll)
kono
parents:
diff changeset
697 {
kono
parents:
diff changeset
698 tmp->fraction.ll <<= 1;
kono
parents:
diff changeset
699 tmp->normal_exp--;
kono
parents:
diff changeset
700 }
kono
parents:
diff changeset
701 }
kono
parents:
diff changeset
702 else
kono
parents:
diff changeset
703 {
kono
parents:
diff changeset
704 tmp->sign = a->sign;
kono
parents:
diff changeset
705 tmp->normal_exp = a_normal_exp;
kono
parents:
diff changeset
706 tmp->fraction.ll = a_fraction + b_fraction;
kono
parents:
diff changeset
707 }
kono
parents:
diff changeset
708 tmp->class = CLASS_NUMBER;
kono
parents:
diff changeset
709 /* Now the fraction is added, we have to shift down to renormalize the
kono
parents:
diff changeset
710 number */
kono
parents:
diff changeset
711
kono
parents:
diff changeset
712 if (tmp->fraction.ll >= IMPLICIT_2)
kono
parents:
diff changeset
713 {
kono
parents:
diff changeset
714 LSHIFT (tmp->fraction.ll, 1);
kono
parents:
diff changeset
715 tmp->normal_exp++;
kono
parents:
diff changeset
716 }
kono
parents:
diff changeset
717 return tmp;
kono
parents:
diff changeset
718 }
kono
parents:
diff changeset
719
kono
parents:
diff changeset
720 FLO_type
kono
parents:
diff changeset
721 add (FLO_type arg_a, FLO_type arg_b)
kono
parents:
diff changeset
722 {
kono
parents:
diff changeset
723 fp_number_type a;
kono
parents:
diff changeset
724 fp_number_type b;
kono
parents:
diff changeset
725 fp_number_type tmp;
kono
parents:
diff changeset
726 const fp_number_type *res;
kono
parents:
diff changeset
727 FLO_union_type au, bu;
kono
parents:
diff changeset
728
kono
parents:
diff changeset
729 au.value = arg_a;
kono
parents:
diff changeset
730 bu.value = arg_b;
kono
parents:
diff changeset
731
kono
parents:
diff changeset
732 unpack_d (&au, &a);
kono
parents:
diff changeset
733 unpack_d (&bu, &b);
kono
parents:
diff changeset
734
kono
parents:
diff changeset
735 res = _fpadd_parts (&a, &b, &tmp);
kono
parents:
diff changeset
736
kono
parents:
diff changeset
737 return pack_d (res);
kono
parents:
diff changeset
738 }
kono
parents:
diff changeset
739
kono
parents:
diff changeset
740 FLO_type
kono
parents:
diff changeset
741 sub (FLO_type arg_a, FLO_type arg_b)
kono
parents:
diff changeset
742 {
kono
parents:
diff changeset
743 fp_number_type a;
kono
parents:
diff changeset
744 fp_number_type b;
kono
parents:
diff changeset
745 fp_number_type tmp;
kono
parents:
diff changeset
746 const fp_number_type *res;
kono
parents:
diff changeset
747 FLO_union_type au, bu;
kono
parents:
diff changeset
748
kono
parents:
diff changeset
749 au.value = arg_a;
kono
parents:
diff changeset
750 bu.value = arg_b;
kono
parents:
diff changeset
751
kono
parents:
diff changeset
752 unpack_d (&au, &a);
kono
parents:
diff changeset
753 unpack_d (&bu, &b);
kono
parents:
diff changeset
754
kono
parents:
diff changeset
755 b.sign ^= 1;
kono
parents:
diff changeset
756
kono
parents:
diff changeset
757 res = _fpadd_parts (&a, &b, &tmp);
kono
parents:
diff changeset
758
kono
parents:
diff changeset
759 return pack_d (res);
kono
parents:
diff changeset
760 }
kono
parents:
diff changeset
761 #endif /* L_addsub_sf || L_addsub_df */
kono
parents:
diff changeset
762
kono
parents:
diff changeset
763 #if defined(L_mul_sf) || defined(L_mul_df) || defined(L_mul_tf)
kono
parents:
diff changeset
764 static inline __attribute__ ((__always_inline__)) const fp_number_type *
kono
parents:
diff changeset
765 _fpmul_parts ( fp_number_type * a,
kono
parents:
diff changeset
766 fp_number_type * b,
kono
parents:
diff changeset
767 fp_number_type * tmp)
kono
parents:
diff changeset
768 {
kono
parents:
diff changeset
769 fractype low = 0;
kono
parents:
diff changeset
770 fractype high = 0;
kono
parents:
diff changeset
771
kono
parents:
diff changeset
772 if (isnan (a))
kono
parents:
diff changeset
773 {
kono
parents:
diff changeset
774 a->sign = a->sign != b->sign;
kono
parents:
diff changeset
775 return a;
kono
parents:
diff changeset
776 }
kono
parents:
diff changeset
777 if (isnan (b))
kono
parents:
diff changeset
778 {
kono
parents:
diff changeset
779 b->sign = a->sign != b->sign;
kono
parents:
diff changeset
780 return b;
kono
parents:
diff changeset
781 }
kono
parents:
diff changeset
782 if (isinf (a))
kono
parents:
diff changeset
783 {
kono
parents:
diff changeset
784 if (iszero (b))
kono
parents:
diff changeset
785 return makenan ();
kono
parents:
diff changeset
786 a->sign = a->sign != b->sign;
kono
parents:
diff changeset
787 return a;
kono
parents:
diff changeset
788 }
kono
parents:
diff changeset
789 if (isinf (b))
kono
parents:
diff changeset
790 {
kono
parents:
diff changeset
791 if (iszero (a))
kono
parents:
diff changeset
792 {
kono
parents:
diff changeset
793 return makenan ();
kono
parents:
diff changeset
794 }
kono
parents:
diff changeset
795 b->sign = a->sign != b->sign;
kono
parents:
diff changeset
796 return b;
kono
parents:
diff changeset
797 }
kono
parents:
diff changeset
798 if (iszero (a))
kono
parents:
diff changeset
799 {
kono
parents:
diff changeset
800 a->sign = a->sign != b->sign;
kono
parents:
diff changeset
801 return a;
kono
parents:
diff changeset
802 }
kono
parents:
diff changeset
803 if (iszero (b))
kono
parents:
diff changeset
804 {
kono
parents:
diff changeset
805 b->sign = a->sign != b->sign;
kono
parents:
diff changeset
806 return b;
kono
parents:
diff changeset
807 }
kono
parents:
diff changeset
808
kono
parents:
diff changeset
809 /* Calculate the mantissa by multiplying both numbers to get a
kono
parents:
diff changeset
810 twice-as-wide number. */
kono
parents:
diff changeset
811 {
kono
parents:
diff changeset
812 #if defined(NO_DI_MODE) || defined(TFLOAT)
kono
parents:
diff changeset
813 {
kono
parents:
diff changeset
814 fractype x = a->fraction.ll;
kono
parents:
diff changeset
815 fractype ylow = b->fraction.ll;
kono
parents:
diff changeset
816 fractype yhigh = 0;
kono
parents:
diff changeset
817 int bit;
kono
parents:
diff changeset
818
kono
parents:
diff changeset
819 /* ??? This does multiplies one bit at a time. Optimize. */
kono
parents:
diff changeset
820 for (bit = 0; bit < FRAC_NBITS; bit++)
kono
parents:
diff changeset
821 {
kono
parents:
diff changeset
822 int carry;
kono
parents:
diff changeset
823
kono
parents:
diff changeset
824 if (x & 1)
kono
parents:
diff changeset
825 {
kono
parents:
diff changeset
826 carry = (low += ylow) < ylow;
kono
parents:
diff changeset
827 high += yhigh + carry;
kono
parents:
diff changeset
828 }
kono
parents:
diff changeset
829 yhigh <<= 1;
kono
parents:
diff changeset
830 if (ylow & FRACHIGH)
kono
parents:
diff changeset
831 {
kono
parents:
diff changeset
832 yhigh |= 1;
kono
parents:
diff changeset
833 }
kono
parents:
diff changeset
834 ylow <<= 1;
kono
parents:
diff changeset
835 x >>= 1;
kono
parents:
diff changeset
836 }
kono
parents:
diff changeset
837 }
kono
parents:
diff changeset
838 #elif defined(FLOAT)
kono
parents:
diff changeset
839 /* Multiplying two USIs to get a UDI, we're safe. */
kono
parents:
diff changeset
840 {
kono
parents:
diff changeset
841 UDItype answer = (UDItype)a->fraction.ll * (UDItype)b->fraction.ll;
kono
parents:
diff changeset
842
kono
parents:
diff changeset
843 high = answer >> BITS_PER_SI;
kono
parents:
diff changeset
844 low = answer;
kono
parents:
diff changeset
845 }
kono
parents:
diff changeset
846 #else
kono
parents:
diff changeset
847 /* fractype is DImode, but we need the result to be twice as wide.
kono
parents:
diff changeset
848 Assuming a widening multiply from DImode to TImode is not
kono
parents:
diff changeset
849 available, build one by hand. */
kono
parents:
diff changeset
850 {
kono
parents:
diff changeset
851 USItype nl = a->fraction.ll;
kono
parents:
diff changeset
852 USItype nh = a->fraction.ll >> BITS_PER_SI;
kono
parents:
diff changeset
853 USItype ml = b->fraction.ll;
kono
parents:
diff changeset
854 USItype mh = b->fraction.ll >> BITS_PER_SI;
kono
parents:
diff changeset
855 UDItype pp_ll = (UDItype) ml * nl;
kono
parents:
diff changeset
856 UDItype pp_hl = (UDItype) mh * nl;
kono
parents:
diff changeset
857 UDItype pp_lh = (UDItype) ml * nh;
kono
parents:
diff changeset
858 UDItype pp_hh = (UDItype) mh * nh;
kono
parents:
diff changeset
859 UDItype res2 = 0;
kono
parents:
diff changeset
860 UDItype res0 = 0;
kono
parents:
diff changeset
861 UDItype ps_hh__ = pp_hl + pp_lh;
kono
parents:
diff changeset
862 if (ps_hh__ < pp_hl)
kono
parents:
diff changeset
863 res2 += (UDItype)1 << BITS_PER_SI;
kono
parents:
diff changeset
864 pp_hl = (UDItype)(USItype)ps_hh__ << BITS_PER_SI;
kono
parents:
diff changeset
865 res0 = pp_ll + pp_hl;
kono
parents:
diff changeset
866 if (res0 < pp_ll)
kono
parents:
diff changeset
867 res2++;
kono
parents:
diff changeset
868 res2 += (ps_hh__ >> BITS_PER_SI) + pp_hh;
kono
parents:
diff changeset
869 high = res2;
kono
parents:
diff changeset
870 low = res0;
kono
parents:
diff changeset
871 }
kono
parents:
diff changeset
872 #endif
kono
parents:
diff changeset
873 }
kono
parents:
diff changeset
874
kono
parents:
diff changeset
875 tmp->normal_exp = a->normal_exp + b->normal_exp
kono
parents:
diff changeset
876 + FRAC_NBITS - (FRACBITS + NGARDS);
kono
parents:
diff changeset
877 tmp->sign = a->sign != b->sign;
kono
parents:
diff changeset
878 while (high >= IMPLICIT_2)
kono
parents:
diff changeset
879 {
kono
parents:
diff changeset
880 tmp->normal_exp++;
kono
parents:
diff changeset
881 if (high & 1)
kono
parents:
diff changeset
882 {
kono
parents:
diff changeset
883 low >>= 1;
kono
parents:
diff changeset
884 low |= FRACHIGH;
kono
parents:
diff changeset
885 }
kono
parents:
diff changeset
886 high >>= 1;
kono
parents:
diff changeset
887 }
kono
parents:
diff changeset
888 while (high < IMPLICIT_1)
kono
parents:
diff changeset
889 {
kono
parents:
diff changeset
890 tmp->normal_exp--;
kono
parents:
diff changeset
891
kono
parents:
diff changeset
892 high <<= 1;
kono
parents:
diff changeset
893 if (low & FRACHIGH)
kono
parents:
diff changeset
894 high |= 1;
kono
parents:
diff changeset
895 low <<= 1;
kono
parents:
diff changeset
896 }
kono
parents:
diff changeset
897
kono
parents:
diff changeset
898 if ((high & GARDMASK) == GARDMSB)
kono
parents:
diff changeset
899 {
kono
parents:
diff changeset
900 if (high & (1 << NGARDS))
kono
parents:
diff changeset
901 {
kono
parents:
diff changeset
902 /* Because we're half way, we would round to even by adding
kono
parents:
diff changeset
903 GARDROUND + 1, except that's also done in the packing
kono
parents:
diff changeset
904 function, and rounding twice will lose precision and cause
kono
parents:
diff changeset
905 the result to be too far off. Example: 32-bit floats with
kono
parents:
diff changeset
906 bit patterns 0xfff * 0x3f800400 ~= 0xfff (less than 0.5ulp
kono
parents:
diff changeset
907 off), not 0x1000 (more than 0.5ulp off). */
kono
parents:
diff changeset
908 }
kono
parents:
diff changeset
909 else if (low)
kono
parents:
diff changeset
910 {
kono
parents:
diff changeset
911 /* We're a further than half way by a small amount corresponding
kono
parents:
diff changeset
912 to the bits set in "low". Knowing that, we round here and
kono
parents:
diff changeset
913 not in pack_d, because there we don't have "low" available
kono
parents:
diff changeset
914 anymore. */
kono
parents:
diff changeset
915 high += GARDROUND + 1;
kono
parents:
diff changeset
916
kono
parents:
diff changeset
917 /* Avoid further rounding in pack_d. */
kono
parents:
diff changeset
918 high &= ~(fractype) GARDMASK;
kono
parents:
diff changeset
919 }
kono
parents:
diff changeset
920 }
kono
parents:
diff changeset
921 tmp->fraction.ll = high;
kono
parents:
diff changeset
922 tmp->class = CLASS_NUMBER;
kono
parents:
diff changeset
923 return tmp;
kono
parents:
diff changeset
924 }
kono
parents:
diff changeset
925
kono
parents:
diff changeset
926 FLO_type
kono
parents:
diff changeset
927 multiply (FLO_type arg_a, FLO_type arg_b)
kono
parents:
diff changeset
928 {
kono
parents:
diff changeset
929 fp_number_type a;
kono
parents:
diff changeset
930 fp_number_type b;
kono
parents:
diff changeset
931 fp_number_type tmp;
kono
parents:
diff changeset
932 const fp_number_type *res;
kono
parents:
diff changeset
933 FLO_union_type au, bu;
kono
parents:
diff changeset
934
kono
parents:
diff changeset
935 au.value = arg_a;
kono
parents:
diff changeset
936 bu.value = arg_b;
kono
parents:
diff changeset
937
kono
parents:
diff changeset
938 unpack_d (&au, &a);
kono
parents:
diff changeset
939 unpack_d (&bu, &b);
kono
parents:
diff changeset
940
kono
parents:
diff changeset
941 res = _fpmul_parts (&a, &b, &tmp);
kono
parents:
diff changeset
942
kono
parents:
diff changeset
943 return pack_d (res);
kono
parents:
diff changeset
944 }
kono
parents:
diff changeset
945 #endif /* L_mul_sf || L_mul_df || L_mul_tf */
kono
parents:
diff changeset
946
kono
parents:
diff changeset
947 #if defined(L_div_sf) || defined(L_div_df) || defined(L_div_tf)
kono
parents:
diff changeset
948 static inline __attribute__ ((__always_inline__)) const fp_number_type *
kono
parents:
diff changeset
949 _fpdiv_parts (fp_number_type * a,
kono
parents:
diff changeset
950 fp_number_type * b)
kono
parents:
diff changeset
951 {
kono
parents:
diff changeset
952 fractype bit;
kono
parents:
diff changeset
953 fractype numerator;
kono
parents:
diff changeset
954 fractype denominator;
kono
parents:
diff changeset
955 fractype quotient;
kono
parents:
diff changeset
956
kono
parents:
diff changeset
957 if (isnan (a))
kono
parents:
diff changeset
958 {
kono
parents:
diff changeset
959 return a;
kono
parents:
diff changeset
960 }
kono
parents:
diff changeset
961 if (isnan (b))
kono
parents:
diff changeset
962 {
kono
parents:
diff changeset
963 return b;
kono
parents:
diff changeset
964 }
kono
parents:
diff changeset
965
kono
parents:
diff changeset
966 a->sign = a->sign ^ b->sign;
kono
parents:
diff changeset
967
kono
parents:
diff changeset
968 if (isinf (a) || iszero (a))
kono
parents:
diff changeset
969 {
kono
parents:
diff changeset
970 if (a->class == b->class)
kono
parents:
diff changeset
971 return makenan ();
kono
parents:
diff changeset
972 return a;
kono
parents:
diff changeset
973 }
kono
parents:
diff changeset
974
kono
parents:
diff changeset
975 if (isinf (b))
kono
parents:
diff changeset
976 {
kono
parents:
diff changeset
977 a->fraction.ll = 0;
kono
parents:
diff changeset
978 a->normal_exp = 0;
kono
parents:
diff changeset
979 return a;
kono
parents:
diff changeset
980 }
kono
parents:
diff changeset
981 if (iszero (b))
kono
parents:
diff changeset
982 {
kono
parents:
diff changeset
983 a->class = CLASS_INFINITY;
kono
parents:
diff changeset
984 return a;
kono
parents:
diff changeset
985 }
kono
parents:
diff changeset
986
kono
parents:
diff changeset
987 /* Calculate the mantissa by multiplying both 64bit numbers to get a
kono
parents:
diff changeset
988 128 bit number */
kono
parents:
diff changeset
989 {
kono
parents:
diff changeset
990 /* quotient =
kono
parents:
diff changeset
991 ( numerator / denominator) * 2^(numerator exponent - denominator exponent)
kono
parents:
diff changeset
992 */
kono
parents:
diff changeset
993
kono
parents:
diff changeset
994 a->normal_exp = a->normal_exp - b->normal_exp;
kono
parents:
diff changeset
995 numerator = a->fraction.ll;
kono
parents:
diff changeset
996 denominator = b->fraction.ll;
kono
parents:
diff changeset
997
kono
parents:
diff changeset
998 if (numerator < denominator)
kono
parents:
diff changeset
999 {
kono
parents:
diff changeset
1000 /* Fraction will be less than 1.0 */
kono
parents:
diff changeset
1001 numerator *= 2;
kono
parents:
diff changeset
1002 a->normal_exp--;
kono
parents:
diff changeset
1003 }
kono
parents:
diff changeset
1004 bit = IMPLICIT_1;
kono
parents:
diff changeset
1005 quotient = 0;
kono
parents:
diff changeset
1006 /* ??? Does divide one bit at a time. Optimize. */
kono
parents:
diff changeset
1007 while (bit)
kono
parents:
diff changeset
1008 {
kono
parents:
diff changeset
1009 if (numerator >= denominator)
kono
parents:
diff changeset
1010 {
kono
parents:
diff changeset
1011 quotient |= bit;
kono
parents:
diff changeset
1012 numerator -= denominator;
kono
parents:
diff changeset
1013 }
kono
parents:
diff changeset
1014 bit >>= 1;
kono
parents:
diff changeset
1015 numerator *= 2;
kono
parents:
diff changeset
1016 }
kono
parents:
diff changeset
1017
kono
parents:
diff changeset
1018 if ((quotient & GARDMASK) == GARDMSB)
kono
parents:
diff changeset
1019 {
kono
parents:
diff changeset
1020 if (quotient & (1 << NGARDS))
kono
parents:
diff changeset
1021 {
kono
parents:
diff changeset
1022 /* Because we're half way, we would round to even by adding
kono
parents:
diff changeset
1023 GARDROUND + 1, except that's also done in the packing
kono
parents:
diff changeset
1024 function, and rounding twice will lose precision and cause
kono
parents:
diff changeset
1025 the result to be too far off. */
kono
parents:
diff changeset
1026 }
kono
parents:
diff changeset
1027 else if (numerator)
kono
parents:
diff changeset
1028 {
kono
parents:
diff changeset
1029 /* We're a further than half way by the small amount
kono
parents:
diff changeset
1030 corresponding to the bits set in "numerator". Knowing
kono
parents:
diff changeset
1031 that, we round here and not in pack_d, because there we
kono
parents:
diff changeset
1032 don't have "numerator" available anymore. */
kono
parents:
diff changeset
1033 quotient += GARDROUND + 1;
kono
parents:
diff changeset
1034
kono
parents:
diff changeset
1035 /* Avoid further rounding in pack_d. */
kono
parents:
diff changeset
1036 quotient &= ~(fractype) GARDMASK;
kono
parents:
diff changeset
1037 }
kono
parents:
diff changeset
1038 }
kono
parents:
diff changeset
1039
kono
parents:
diff changeset
1040 a->fraction.ll = quotient;
kono
parents:
diff changeset
1041 return (a);
kono
parents:
diff changeset
1042 }
kono
parents:
diff changeset
1043 }
kono
parents:
diff changeset
1044
kono
parents:
diff changeset
1045 FLO_type
kono
parents:
diff changeset
1046 divide (FLO_type arg_a, FLO_type arg_b)
kono
parents:
diff changeset
1047 {
kono
parents:
diff changeset
1048 fp_number_type a;
kono
parents:
diff changeset
1049 fp_number_type b;
kono
parents:
diff changeset
1050 const fp_number_type *res;
kono
parents:
diff changeset
1051 FLO_union_type au, bu;
kono
parents:
diff changeset
1052
kono
parents:
diff changeset
1053 au.value = arg_a;
kono
parents:
diff changeset
1054 bu.value = arg_b;
kono
parents:
diff changeset
1055
kono
parents:
diff changeset
1056 unpack_d (&au, &a);
kono
parents:
diff changeset
1057 unpack_d (&bu, &b);
kono
parents:
diff changeset
1058
kono
parents:
diff changeset
1059 res = _fpdiv_parts (&a, &b);
kono
parents:
diff changeset
1060
kono
parents:
diff changeset
1061 return pack_d (res);
kono
parents:
diff changeset
1062 }
kono
parents:
diff changeset
1063 #endif /* L_div_sf || L_div_df */
kono
parents:
diff changeset
1064
kono
parents:
diff changeset
1065 #if defined(L_fpcmp_parts_sf) || defined(L_fpcmp_parts_df) \
kono
parents:
diff changeset
1066 || defined(L_fpcmp_parts_tf)
kono
parents:
diff changeset
1067 /* according to the demo, fpcmp returns a comparison with 0... thus
kono
parents:
diff changeset
1068 a<b -> -1
kono
parents:
diff changeset
1069 a==b -> 0
kono
parents:
diff changeset
1070 a>b -> +1
kono
parents:
diff changeset
1071 */
kono
parents:
diff changeset
1072
kono
parents:
diff changeset
1073 int
kono
parents:
diff changeset
1074 __fpcmp_parts (fp_number_type * a, fp_number_type * b)
kono
parents:
diff changeset
1075 {
kono
parents:
diff changeset
1076 #if 0
kono
parents:
diff changeset
1077 /* either nan -> unordered. Must be checked outside of this routine. */
kono
parents:
diff changeset
1078 if (isnan (a) && isnan (b))
kono
parents:
diff changeset
1079 {
kono
parents:
diff changeset
1080 return 1; /* still unordered! */
kono
parents:
diff changeset
1081 }
kono
parents:
diff changeset
1082 #endif
kono
parents:
diff changeset
1083
kono
parents:
diff changeset
1084 if (isnan (a) || isnan (b))
kono
parents:
diff changeset
1085 {
kono
parents:
diff changeset
1086 return 1; /* how to indicate unordered compare? */
kono
parents:
diff changeset
1087 }
kono
parents:
diff changeset
1088 if (isinf (a) && isinf (b))
kono
parents:
diff changeset
1089 {
kono
parents:
diff changeset
1090 /* +inf > -inf, but +inf != +inf */
kono
parents:
diff changeset
1091 /* b \a| +inf(0)| -inf(1)
kono
parents:
diff changeset
1092 ______\+--------+--------
kono
parents:
diff changeset
1093 +inf(0)| a==b(0)| a<b(-1)
kono
parents:
diff changeset
1094 -------+--------+--------
kono
parents:
diff changeset
1095 -inf(1)| a>b(1) | a==b(0)
kono
parents:
diff changeset
1096 -------+--------+--------
kono
parents:
diff changeset
1097 So since unordered must be nonzero, just line up the columns...
kono
parents:
diff changeset
1098 */
kono
parents:
diff changeset
1099 return b->sign - a->sign;
kono
parents:
diff changeset
1100 }
kono
parents:
diff changeset
1101 /* but not both... */
kono
parents:
diff changeset
1102 if (isinf (a))
kono
parents:
diff changeset
1103 {
kono
parents:
diff changeset
1104 return a->sign ? -1 : 1;
kono
parents:
diff changeset
1105 }
kono
parents:
diff changeset
1106 if (isinf (b))
kono
parents:
diff changeset
1107 {
kono
parents:
diff changeset
1108 return b->sign ? 1 : -1;
kono
parents:
diff changeset
1109 }
kono
parents:
diff changeset
1110 if (iszero (a) && iszero (b))
kono
parents:
diff changeset
1111 {
kono
parents:
diff changeset
1112 return 0;
kono
parents:
diff changeset
1113 }
kono
parents:
diff changeset
1114 if (iszero (a))
kono
parents:
diff changeset
1115 {
kono
parents:
diff changeset
1116 return b->sign ? 1 : -1;
kono
parents:
diff changeset
1117 }
kono
parents:
diff changeset
1118 if (iszero (b))
kono
parents:
diff changeset
1119 {
kono
parents:
diff changeset
1120 return a->sign ? -1 : 1;
kono
parents:
diff changeset
1121 }
kono
parents:
diff changeset
1122 /* now both are "normal". */
kono
parents:
diff changeset
1123 if (a->sign != b->sign)
kono
parents:
diff changeset
1124 {
kono
parents:
diff changeset
1125 /* opposite signs */
kono
parents:
diff changeset
1126 return a->sign ? -1 : 1;
kono
parents:
diff changeset
1127 }
kono
parents:
diff changeset
1128 /* same sign; exponents? */
kono
parents:
diff changeset
1129 if (a->normal_exp > b->normal_exp)
kono
parents:
diff changeset
1130 {
kono
parents:
diff changeset
1131 return a->sign ? -1 : 1;
kono
parents:
diff changeset
1132 }
kono
parents:
diff changeset
1133 if (a->normal_exp < b->normal_exp)
kono
parents:
diff changeset
1134 {
kono
parents:
diff changeset
1135 return a->sign ? 1 : -1;
kono
parents:
diff changeset
1136 }
kono
parents:
diff changeset
1137 /* same exponents; check size. */
kono
parents:
diff changeset
1138 if (a->fraction.ll > b->fraction.ll)
kono
parents:
diff changeset
1139 {
kono
parents:
diff changeset
1140 return a->sign ? -1 : 1;
kono
parents:
diff changeset
1141 }
kono
parents:
diff changeset
1142 if (a->fraction.ll < b->fraction.ll)
kono
parents:
diff changeset
1143 {
kono
parents:
diff changeset
1144 return a->sign ? 1 : -1;
kono
parents:
diff changeset
1145 }
kono
parents:
diff changeset
1146 /* after all that, they're equal. */
kono
parents:
diff changeset
1147 return 0;
kono
parents:
diff changeset
1148 }
kono
parents:
diff changeset
1149 #endif
kono
parents:
diff changeset
1150
kono
parents:
diff changeset
1151 #if defined(L_compare_sf) || defined(L_compare_df) || defined(L_compoare_tf)
kono
parents:
diff changeset
1152 CMPtype
kono
parents:
diff changeset
1153 compare (FLO_type arg_a, FLO_type arg_b)
kono
parents:
diff changeset
1154 {
kono
parents:
diff changeset
1155 fp_number_type a;
kono
parents:
diff changeset
1156 fp_number_type b;
kono
parents:
diff changeset
1157 FLO_union_type au, bu;
kono
parents:
diff changeset
1158
kono
parents:
diff changeset
1159 au.value = arg_a;
kono
parents:
diff changeset
1160 bu.value = arg_b;
kono
parents:
diff changeset
1161
kono
parents:
diff changeset
1162 unpack_d (&au, &a);
kono
parents:
diff changeset
1163 unpack_d (&bu, &b);
kono
parents:
diff changeset
1164
kono
parents:
diff changeset
1165 return __fpcmp_parts (&a, &b);
kono
parents:
diff changeset
1166 }
kono
parents:
diff changeset
1167 #endif /* L_compare_sf || L_compare_df */
kono
parents:
diff changeset
1168
kono
parents:
diff changeset
1169 /* These should be optimized for their specific tasks someday. */
kono
parents:
diff changeset
1170
kono
parents:
diff changeset
1171 #if defined(L_eq_sf) || defined(L_eq_df) || defined(L_eq_tf)
kono
parents:
diff changeset
1172 CMPtype
kono
parents:
diff changeset
1173 _eq_f2 (FLO_type arg_a, FLO_type arg_b)
kono
parents:
diff changeset
1174 {
kono
parents:
diff changeset
1175 fp_number_type a;
kono
parents:
diff changeset
1176 fp_number_type b;
kono
parents:
diff changeset
1177 FLO_union_type au, bu;
kono
parents:
diff changeset
1178
kono
parents:
diff changeset
1179 au.value = arg_a;
kono
parents:
diff changeset
1180 bu.value = arg_b;
kono
parents:
diff changeset
1181
kono
parents:
diff changeset
1182 unpack_d (&au, &a);
kono
parents:
diff changeset
1183 unpack_d (&bu, &b);
kono
parents:
diff changeset
1184
kono
parents:
diff changeset
1185 if (isnan (&a) || isnan (&b))
kono
parents:
diff changeset
1186 return 1; /* false, truth == 0 */
kono
parents:
diff changeset
1187
kono
parents:
diff changeset
1188 return __fpcmp_parts (&a, &b) ;
kono
parents:
diff changeset
1189 }
kono
parents:
diff changeset
1190 #endif /* L_eq_sf || L_eq_df */
kono
parents:
diff changeset
1191
kono
parents:
diff changeset
1192 #if defined(L_ne_sf) || defined(L_ne_df) || defined(L_ne_tf)
kono
parents:
diff changeset
1193 CMPtype
kono
parents:
diff changeset
1194 _ne_f2 (FLO_type arg_a, FLO_type arg_b)
kono
parents:
diff changeset
1195 {
kono
parents:
diff changeset
1196 fp_number_type a;
kono
parents:
diff changeset
1197 fp_number_type b;
kono
parents:
diff changeset
1198 FLO_union_type au, bu;
kono
parents:
diff changeset
1199
kono
parents:
diff changeset
1200 au.value = arg_a;
kono
parents:
diff changeset
1201 bu.value = arg_b;
kono
parents:
diff changeset
1202
kono
parents:
diff changeset
1203 unpack_d (&au, &a);
kono
parents:
diff changeset
1204 unpack_d (&bu, &b);
kono
parents:
diff changeset
1205
kono
parents:
diff changeset
1206 if (isnan (&a) || isnan (&b))
kono
parents:
diff changeset
1207 return 1; /* true, truth != 0 */
kono
parents:
diff changeset
1208
kono
parents:
diff changeset
1209 return __fpcmp_parts (&a, &b) ;
kono
parents:
diff changeset
1210 }
kono
parents:
diff changeset
1211 #endif /* L_ne_sf || L_ne_df */
kono
parents:
diff changeset
1212
kono
parents:
diff changeset
1213 #if defined(L_gt_sf) || defined(L_gt_df) || defined(L_gt_tf)
kono
parents:
diff changeset
1214 CMPtype
kono
parents:
diff changeset
1215 _gt_f2 (FLO_type arg_a, FLO_type arg_b)
kono
parents:
diff changeset
1216 {
kono
parents:
diff changeset
1217 fp_number_type a;
kono
parents:
diff changeset
1218 fp_number_type b;
kono
parents:
diff changeset
1219 FLO_union_type au, bu;
kono
parents:
diff changeset
1220
kono
parents:
diff changeset
1221 au.value = arg_a;
kono
parents:
diff changeset
1222 bu.value = arg_b;
kono
parents:
diff changeset
1223
kono
parents:
diff changeset
1224 unpack_d (&au, &a);
kono
parents:
diff changeset
1225 unpack_d (&bu, &b);
kono
parents:
diff changeset
1226
kono
parents:
diff changeset
1227 if (isnan (&a) || isnan (&b))
kono
parents:
diff changeset
1228 return -1; /* false, truth > 0 */
kono
parents:
diff changeset
1229
kono
parents:
diff changeset
1230 return __fpcmp_parts (&a, &b);
kono
parents:
diff changeset
1231 }
kono
parents:
diff changeset
1232 #endif /* L_gt_sf || L_gt_df */
kono
parents:
diff changeset
1233
kono
parents:
diff changeset
1234 #if defined(L_ge_sf) || defined(L_ge_df) || defined(L_ge_tf)
kono
parents:
diff changeset
1235 CMPtype
kono
parents:
diff changeset
1236 _ge_f2 (FLO_type arg_a, FLO_type arg_b)
kono
parents:
diff changeset
1237 {
kono
parents:
diff changeset
1238 fp_number_type a;
kono
parents:
diff changeset
1239 fp_number_type b;
kono
parents:
diff changeset
1240 FLO_union_type au, bu;
kono
parents:
diff changeset
1241
kono
parents:
diff changeset
1242 au.value = arg_a;
kono
parents:
diff changeset
1243 bu.value = arg_b;
kono
parents:
diff changeset
1244
kono
parents:
diff changeset
1245 unpack_d (&au, &a);
kono
parents:
diff changeset
1246 unpack_d (&bu, &b);
kono
parents:
diff changeset
1247
kono
parents:
diff changeset
1248 if (isnan (&a) || isnan (&b))
kono
parents:
diff changeset
1249 return -1; /* false, truth >= 0 */
kono
parents:
diff changeset
1250 return __fpcmp_parts (&a, &b) ;
kono
parents:
diff changeset
1251 }
kono
parents:
diff changeset
1252 #endif /* L_ge_sf || L_ge_df */
kono
parents:
diff changeset
1253
kono
parents:
diff changeset
1254 #if defined(L_lt_sf) || defined(L_lt_df) || defined(L_lt_tf)
kono
parents:
diff changeset
1255 CMPtype
kono
parents:
diff changeset
1256 _lt_f2 (FLO_type arg_a, FLO_type arg_b)
kono
parents:
diff changeset
1257 {
kono
parents:
diff changeset
1258 fp_number_type a;
kono
parents:
diff changeset
1259 fp_number_type b;
kono
parents:
diff changeset
1260 FLO_union_type au, bu;
kono
parents:
diff changeset
1261
kono
parents:
diff changeset
1262 au.value = arg_a;
kono
parents:
diff changeset
1263 bu.value = arg_b;
kono
parents:
diff changeset
1264
kono
parents:
diff changeset
1265 unpack_d (&au, &a);
kono
parents:
diff changeset
1266 unpack_d (&bu, &b);
kono
parents:
diff changeset
1267
kono
parents:
diff changeset
1268 if (isnan (&a) || isnan (&b))
kono
parents:
diff changeset
1269 return 1; /* false, truth < 0 */
kono
parents:
diff changeset
1270
kono
parents:
diff changeset
1271 return __fpcmp_parts (&a, &b);
kono
parents:
diff changeset
1272 }
kono
parents:
diff changeset
1273 #endif /* L_lt_sf || L_lt_df */
kono
parents:
diff changeset
1274
kono
parents:
diff changeset
1275 #if defined(L_le_sf) || defined(L_le_df) || defined(L_le_tf)
kono
parents:
diff changeset
1276 CMPtype
kono
parents:
diff changeset
1277 _le_f2 (FLO_type arg_a, FLO_type arg_b)
kono
parents:
diff changeset
1278 {
kono
parents:
diff changeset
1279 fp_number_type a;
kono
parents:
diff changeset
1280 fp_number_type b;
kono
parents:
diff changeset
1281 FLO_union_type au, bu;
kono
parents:
diff changeset
1282
kono
parents:
diff changeset
1283 au.value = arg_a;
kono
parents:
diff changeset
1284 bu.value = arg_b;
kono
parents:
diff changeset
1285
kono
parents:
diff changeset
1286 unpack_d (&au, &a);
kono
parents:
diff changeset
1287 unpack_d (&bu, &b);
kono
parents:
diff changeset
1288
kono
parents:
diff changeset
1289 if (isnan (&a) || isnan (&b))
kono
parents:
diff changeset
1290 return 1; /* false, truth <= 0 */
kono
parents:
diff changeset
1291
kono
parents:
diff changeset
1292 return __fpcmp_parts (&a, &b) ;
kono
parents:
diff changeset
1293 }
kono
parents:
diff changeset
1294 #endif /* L_le_sf || L_le_df */
kono
parents:
diff changeset
1295
kono
parents:
diff changeset
1296 #if defined(L_unord_sf) || defined(L_unord_df) || defined(L_unord_tf)
kono
parents:
diff changeset
1297 CMPtype
kono
parents:
diff changeset
1298 _unord_f2 (FLO_type arg_a, FLO_type arg_b)
kono
parents:
diff changeset
1299 {
kono
parents:
diff changeset
1300 fp_number_type a;
kono
parents:
diff changeset
1301 fp_number_type b;
kono
parents:
diff changeset
1302 FLO_union_type au, bu;
kono
parents:
diff changeset
1303
kono
parents:
diff changeset
1304 au.value = arg_a;
kono
parents:
diff changeset
1305 bu.value = arg_b;
kono
parents:
diff changeset
1306
kono
parents:
diff changeset
1307 unpack_d (&au, &a);
kono
parents:
diff changeset
1308 unpack_d (&bu, &b);
kono
parents:
diff changeset
1309
kono
parents:
diff changeset
1310 return (isnan (&a) || isnan (&b));
kono
parents:
diff changeset
1311 }
kono
parents:
diff changeset
1312 #endif /* L_unord_sf || L_unord_df */
kono
parents:
diff changeset
1313
kono
parents:
diff changeset
1314 #if defined(L_si_to_sf) || defined(L_si_to_df) || defined(L_si_to_tf)
kono
parents:
diff changeset
1315 FLO_type
kono
parents:
diff changeset
1316 si_to_float (SItype arg_a)
kono
parents:
diff changeset
1317 {
kono
parents:
diff changeset
1318 fp_number_type in;
kono
parents:
diff changeset
1319
kono
parents:
diff changeset
1320 in.class = CLASS_NUMBER;
kono
parents:
diff changeset
1321 in.sign = arg_a < 0;
kono
parents:
diff changeset
1322 if (!arg_a)
kono
parents:
diff changeset
1323 {
kono
parents:
diff changeset
1324 in.class = CLASS_ZERO;
kono
parents:
diff changeset
1325 }
kono
parents:
diff changeset
1326 else
kono
parents:
diff changeset
1327 {
kono
parents:
diff changeset
1328 USItype uarg;
kono
parents:
diff changeset
1329 int shift;
kono
parents:
diff changeset
1330 in.normal_exp = FRACBITS + NGARDS;
kono
parents:
diff changeset
1331 if (in.sign)
kono
parents:
diff changeset
1332 {
kono
parents:
diff changeset
1333 /* Special case for minint, since there is no +ve integer
kono
parents:
diff changeset
1334 representation for it */
kono
parents:
diff changeset
1335 if (arg_a == (- MAX_SI_INT - 1))
kono
parents:
diff changeset
1336 {
kono
parents:
diff changeset
1337 return (FLO_type)(- MAX_SI_INT - 1);
kono
parents:
diff changeset
1338 }
kono
parents:
diff changeset
1339 uarg = (-arg_a);
kono
parents:
diff changeset
1340 }
kono
parents:
diff changeset
1341 else
kono
parents:
diff changeset
1342 uarg = arg_a;
kono
parents:
diff changeset
1343
kono
parents:
diff changeset
1344 in.fraction.ll = uarg;
kono
parents:
diff changeset
1345 shift = clzusi (uarg) - (BITS_PER_SI - 1 - FRACBITS - NGARDS);
kono
parents:
diff changeset
1346 if (shift > 0)
kono
parents:
diff changeset
1347 {
kono
parents:
diff changeset
1348 in.fraction.ll <<= shift;
kono
parents:
diff changeset
1349 in.normal_exp -= shift;
kono
parents:
diff changeset
1350 }
kono
parents:
diff changeset
1351 }
kono
parents:
diff changeset
1352 return pack_d (&in);
kono
parents:
diff changeset
1353 }
kono
parents:
diff changeset
1354 #endif /* L_si_to_sf || L_si_to_df */
kono
parents:
diff changeset
1355
kono
parents:
diff changeset
1356 #if defined(L_usi_to_sf) || defined(L_usi_to_df) || defined(L_usi_to_tf)
kono
parents:
diff changeset
1357 FLO_type
kono
parents:
diff changeset
1358 usi_to_float (USItype arg_a)
kono
parents:
diff changeset
1359 {
kono
parents:
diff changeset
1360 fp_number_type in;
kono
parents:
diff changeset
1361
kono
parents:
diff changeset
1362 in.sign = 0;
kono
parents:
diff changeset
1363 if (!arg_a)
kono
parents:
diff changeset
1364 {
kono
parents:
diff changeset
1365 in.class = CLASS_ZERO;
kono
parents:
diff changeset
1366 }
kono
parents:
diff changeset
1367 else
kono
parents:
diff changeset
1368 {
kono
parents:
diff changeset
1369 int shift;
kono
parents:
diff changeset
1370 in.class = CLASS_NUMBER;
kono
parents:
diff changeset
1371 in.normal_exp = FRACBITS + NGARDS;
kono
parents:
diff changeset
1372 in.fraction.ll = arg_a;
kono
parents:
diff changeset
1373
kono
parents:
diff changeset
1374 shift = clzusi (arg_a) - (BITS_PER_SI - 1 - FRACBITS - NGARDS);
kono
parents:
diff changeset
1375 if (shift < 0)
kono
parents:
diff changeset
1376 {
kono
parents:
diff changeset
1377 fractype guard = in.fraction.ll & (((fractype)1 << -shift) - 1);
kono
parents:
diff changeset
1378 in.fraction.ll >>= -shift;
kono
parents:
diff changeset
1379 in.fraction.ll |= (guard != 0);
kono
parents:
diff changeset
1380 in.normal_exp -= shift;
kono
parents:
diff changeset
1381 }
kono
parents:
diff changeset
1382 else if (shift > 0)
kono
parents:
diff changeset
1383 {
kono
parents:
diff changeset
1384 in.fraction.ll <<= shift;
kono
parents:
diff changeset
1385 in.normal_exp -= shift;
kono
parents:
diff changeset
1386 }
kono
parents:
diff changeset
1387 }
kono
parents:
diff changeset
1388 return pack_d (&in);
kono
parents:
diff changeset
1389 }
kono
parents:
diff changeset
1390 #endif
kono
parents:
diff changeset
1391
kono
parents:
diff changeset
1392 #if defined(L_sf_to_si) || defined(L_df_to_si) || defined(L_tf_to_si)
kono
parents:
diff changeset
1393 SItype
kono
parents:
diff changeset
1394 float_to_si (FLO_type arg_a)
kono
parents:
diff changeset
1395 {
kono
parents:
diff changeset
1396 fp_number_type a;
kono
parents:
diff changeset
1397 SItype tmp;
kono
parents:
diff changeset
1398 FLO_union_type au;
kono
parents:
diff changeset
1399
kono
parents:
diff changeset
1400 au.value = arg_a;
kono
parents:
diff changeset
1401 unpack_d (&au, &a);
kono
parents:
diff changeset
1402
kono
parents:
diff changeset
1403 if (iszero (&a))
kono
parents:
diff changeset
1404 return 0;
kono
parents:
diff changeset
1405 if (isnan (&a))
kono
parents:
diff changeset
1406 return 0;
kono
parents:
diff changeset
1407 /* get reasonable MAX_SI_INT... */
kono
parents:
diff changeset
1408 if (isinf (&a))
kono
parents:
diff changeset
1409 return a.sign ? (-MAX_SI_INT)-1 : MAX_SI_INT;
kono
parents:
diff changeset
1410 /* it is a number, but a small one */
kono
parents:
diff changeset
1411 if (a.normal_exp < 0)
kono
parents:
diff changeset
1412 return 0;
kono
parents:
diff changeset
1413 if (a.normal_exp > BITS_PER_SI - 2)
kono
parents:
diff changeset
1414 return a.sign ? (-MAX_SI_INT)-1 : MAX_SI_INT;
kono
parents:
diff changeset
1415 tmp = a.fraction.ll >> ((FRACBITS + NGARDS) - a.normal_exp);
kono
parents:
diff changeset
1416 return a.sign ? (-tmp) : (tmp);
kono
parents:
diff changeset
1417 }
kono
parents:
diff changeset
1418 #endif /* L_sf_to_si || L_df_to_si */
kono
parents:
diff changeset
1419
kono
parents:
diff changeset
1420 #if defined(L_tf_to_usi)
kono
parents:
diff changeset
1421 USItype
kono
parents:
diff changeset
1422 float_to_usi (FLO_type arg_a)
kono
parents:
diff changeset
1423 {
kono
parents:
diff changeset
1424 fp_number_type a;
kono
parents:
diff changeset
1425 FLO_union_type au;
kono
parents:
diff changeset
1426
kono
parents:
diff changeset
1427 au.value = arg_a;
kono
parents:
diff changeset
1428 unpack_d (&au, &a);
kono
parents:
diff changeset
1429
kono
parents:
diff changeset
1430 if (iszero (&a))
kono
parents:
diff changeset
1431 return 0;
kono
parents:
diff changeset
1432 if (isnan (&a))
kono
parents:
diff changeset
1433 return 0;
kono
parents:
diff changeset
1434 /* it is a negative number */
kono
parents:
diff changeset
1435 if (a.sign)
kono
parents:
diff changeset
1436 return 0;
kono
parents:
diff changeset
1437 /* get reasonable MAX_USI_INT... */
kono
parents:
diff changeset
1438 if (isinf (&a))
kono
parents:
diff changeset
1439 return MAX_USI_INT;
kono
parents:
diff changeset
1440 /* it is a number, but a small one */
kono
parents:
diff changeset
1441 if (a.normal_exp < 0)
kono
parents:
diff changeset
1442 return 0;
kono
parents:
diff changeset
1443 if (a.normal_exp > BITS_PER_SI - 1)
kono
parents:
diff changeset
1444 return MAX_USI_INT;
kono
parents:
diff changeset
1445 else if (a.normal_exp > (FRACBITS + NGARDS))
kono
parents:
diff changeset
1446 return a.fraction.ll << (a.normal_exp - (FRACBITS + NGARDS));
kono
parents:
diff changeset
1447 else
kono
parents:
diff changeset
1448 return a.fraction.ll >> ((FRACBITS + NGARDS) - a.normal_exp);
kono
parents:
diff changeset
1449 }
kono
parents:
diff changeset
1450 #endif /* L_tf_to_usi */
kono
parents:
diff changeset
1451
kono
parents:
diff changeset
1452 #if defined(L_negate_sf) || defined(L_negate_df) || defined(L_negate_tf)
kono
parents:
diff changeset
1453 FLO_type
kono
parents:
diff changeset
1454 negate (FLO_type arg_a)
kono
parents:
diff changeset
1455 {
kono
parents:
diff changeset
1456 fp_number_type a;
kono
parents:
diff changeset
1457 FLO_union_type au;
kono
parents:
diff changeset
1458
kono
parents:
diff changeset
1459 au.value = arg_a;
kono
parents:
diff changeset
1460 unpack_d (&au, &a);
kono
parents:
diff changeset
1461
kono
parents:
diff changeset
1462 flip_sign (&a);
kono
parents:
diff changeset
1463 return pack_d (&a);
kono
parents:
diff changeset
1464 }
kono
parents:
diff changeset
1465 #endif /* L_negate_sf || L_negate_df */
kono
parents:
diff changeset
1466
kono
parents:
diff changeset
1467 #ifdef FLOAT
kono
parents:
diff changeset
1468
kono
parents:
diff changeset
1469 #if defined(L_make_sf)
kono
parents:
diff changeset
1470 SFtype
kono
parents:
diff changeset
1471 __make_fp(fp_class_type class,
kono
parents:
diff changeset
1472 unsigned int sign,
kono
parents:
diff changeset
1473 int exp,
kono
parents:
diff changeset
1474 USItype frac)
kono
parents:
diff changeset
1475 {
kono
parents:
diff changeset
1476 fp_number_type in;
kono
parents:
diff changeset
1477
kono
parents:
diff changeset
1478 in.class = class;
kono
parents:
diff changeset
1479 in.sign = sign;
kono
parents:
diff changeset
1480 in.normal_exp = exp;
kono
parents:
diff changeset
1481 in.fraction.ll = frac;
kono
parents:
diff changeset
1482 return pack_d (&in);
kono
parents:
diff changeset
1483 }
kono
parents:
diff changeset
1484 #endif /* L_make_sf */
kono
parents:
diff changeset
1485
kono
parents:
diff changeset
1486 #ifndef FLOAT_ONLY
kono
parents:
diff changeset
1487
kono
parents:
diff changeset
1488 /* This enables one to build an fp library that supports float but not double.
kono
parents:
diff changeset
1489 Otherwise, we would get an undefined reference to __make_dp.
kono
parents:
diff changeset
1490 This is needed for some 8-bit ports that can't handle well values that
kono
parents:
diff changeset
1491 are 8-bytes in size, so we just don't support double for them at all. */
kono
parents:
diff changeset
1492
kono
parents:
diff changeset
1493 #if defined(L_sf_to_df)
kono
parents:
diff changeset
1494 DFtype
kono
parents:
diff changeset
1495 sf_to_df (SFtype arg_a)
kono
parents:
diff changeset
1496 {
kono
parents:
diff changeset
1497 fp_number_type in;
kono
parents:
diff changeset
1498 FLO_union_type au;
kono
parents:
diff changeset
1499
kono
parents:
diff changeset
1500 au.value = arg_a;
kono
parents:
diff changeset
1501 unpack_d (&au, &in);
kono
parents:
diff changeset
1502
kono
parents:
diff changeset
1503 return __make_dp (in.class, in.sign, in.normal_exp,
kono
parents:
diff changeset
1504 ((UDItype) in.fraction.ll) << F_D_BITOFF);
kono
parents:
diff changeset
1505 }
kono
parents:
diff changeset
1506 #endif /* L_sf_to_df */
kono
parents:
diff changeset
1507
kono
parents:
diff changeset
1508 #if defined(L_sf_to_tf) && defined(TMODES)
kono
parents:
diff changeset
1509 TFtype
kono
parents:
diff changeset
1510 sf_to_tf (SFtype arg_a)
kono
parents:
diff changeset
1511 {
kono
parents:
diff changeset
1512 fp_number_type in;
kono
parents:
diff changeset
1513 FLO_union_type au;
kono
parents:
diff changeset
1514
kono
parents:
diff changeset
1515 au.value = arg_a;
kono
parents:
diff changeset
1516 unpack_d (&au, &in);
kono
parents:
diff changeset
1517
kono
parents:
diff changeset
1518 return __make_tp (in.class, in.sign, in.normal_exp,
kono
parents:
diff changeset
1519 ((UTItype) in.fraction.ll) << F_T_BITOFF);
kono
parents:
diff changeset
1520 }
kono
parents:
diff changeset
1521 #endif /* L_sf_to_df */
kono
parents:
diff changeset
1522
kono
parents:
diff changeset
1523 #endif /* ! FLOAT_ONLY */
kono
parents:
diff changeset
1524 #endif /* FLOAT */
kono
parents:
diff changeset
1525
kono
parents:
diff changeset
1526 #ifndef FLOAT
kono
parents:
diff changeset
1527
kono
parents:
diff changeset
1528 extern SFtype __make_fp (fp_class_type, unsigned int, int, USItype);
kono
parents:
diff changeset
1529
kono
parents:
diff changeset
1530 #if defined(L_make_df)
kono
parents:
diff changeset
1531 DFtype
kono
parents:
diff changeset
1532 __make_dp (fp_class_type class, unsigned int sign, int exp, UDItype frac)
kono
parents:
diff changeset
1533 {
kono
parents:
diff changeset
1534 fp_number_type in;
kono
parents:
diff changeset
1535
kono
parents:
diff changeset
1536 in.class = class;
kono
parents:
diff changeset
1537 in.sign = sign;
kono
parents:
diff changeset
1538 in.normal_exp = exp;
kono
parents:
diff changeset
1539 in.fraction.ll = frac;
kono
parents:
diff changeset
1540 return pack_d (&in);
kono
parents:
diff changeset
1541 }
kono
parents:
diff changeset
1542 #endif /* L_make_df */
kono
parents:
diff changeset
1543
kono
parents:
diff changeset
1544 #if defined(L_df_to_sf)
kono
parents:
diff changeset
1545 SFtype
kono
parents:
diff changeset
1546 df_to_sf (DFtype arg_a)
kono
parents:
diff changeset
1547 {
kono
parents:
diff changeset
1548 fp_number_type in;
kono
parents:
diff changeset
1549 USItype sffrac;
kono
parents:
diff changeset
1550 FLO_union_type au;
kono
parents:
diff changeset
1551
kono
parents:
diff changeset
1552 au.value = arg_a;
kono
parents:
diff changeset
1553 unpack_d (&au, &in);
kono
parents:
diff changeset
1554
kono
parents:
diff changeset
1555 sffrac = in.fraction.ll >> F_D_BITOFF;
kono
parents:
diff changeset
1556
kono
parents:
diff changeset
1557 /* We set the lowest guard bit in SFFRAC if we discarded any non
kono
parents:
diff changeset
1558 zero bits. */
kono
parents:
diff changeset
1559 if ((in.fraction.ll & (((USItype) 1 << F_D_BITOFF) - 1)) != 0)
kono
parents:
diff changeset
1560 sffrac |= 1;
kono
parents:
diff changeset
1561
kono
parents:
diff changeset
1562 return __make_fp (in.class, in.sign, in.normal_exp, sffrac);
kono
parents:
diff changeset
1563 }
kono
parents:
diff changeset
1564 #endif /* L_df_to_sf */
kono
parents:
diff changeset
1565
kono
parents:
diff changeset
1566 #if defined(L_df_to_tf) && defined(TMODES) \
kono
parents:
diff changeset
1567 && !defined(FLOAT) && !defined(TFLOAT)
kono
parents:
diff changeset
1568 TFtype
kono
parents:
diff changeset
1569 df_to_tf (DFtype arg_a)
kono
parents:
diff changeset
1570 {
kono
parents:
diff changeset
1571 fp_number_type in;
kono
parents:
diff changeset
1572 FLO_union_type au;
kono
parents:
diff changeset
1573
kono
parents:
diff changeset
1574 au.value = arg_a;
kono
parents:
diff changeset
1575 unpack_d (&au, &in);
kono
parents:
diff changeset
1576
kono
parents:
diff changeset
1577 return __make_tp (in.class, in.sign, in.normal_exp,
kono
parents:
diff changeset
1578 ((UTItype) in.fraction.ll) << D_T_BITOFF);
kono
parents:
diff changeset
1579 }
kono
parents:
diff changeset
1580 #endif /* L_sf_to_df */
kono
parents:
diff changeset
1581
kono
parents:
diff changeset
1582 #ifdef TFLOAT
kono
parents:
diff changeset
1583 #if defined(L_make_tf)
kono
parents:
diff changeset
1584 TFtype
kono
parents:
diff changeset
1585 __make_tp(fp_class_type class,
kono
parents:
diff changeset
1586 unsigned int sign,
kono
parents:
diff changeset
1587 int exp,
kono
parents:
diff changeset
1588 UTItype frac)
kono
parents:
diff changeset
1589 {
kono
parents:
diff changeset
1590 fp_number_type in;
kono
parents:
diff changeset
1591
kono
parents:
diff changeset
1592 in.class = class;
kono
parents:
diff changeset
1593 in.sign = sign;
kono
parents:
diff changeset
1594 in.normal_exp = exp;
kono
parents:
diff changeset
1595 in.fraction.ll = frac;
kono
parents:
diff changeset
1596 return pack_d (&in);
kono
parents:
diff changeset
1597 }
kono
parents:
diff changeset
1598 #endif /* L_make_tf */
kono
parents:
diff changeset
1599
kono
parents:
diff changeset
1600 #if defined(L_tf_to_df)
kono
parents:
diff changeset
1601 DFtype
kono
parents:
diff changeset
1602 tf_to_df (TFtype arg_a)
kono
parents:
diff changeset
1603 {
kono
parents:
diff changeset
1604 fp_number_type in;
kono
parents:
diff changeset
1605 UDItype sffrac;
kono
parents:
diff changeset
1606 FLO_union_type au;
kono
parents:
diff changeset
1607
kono
parents:
diff changeset
1608 au.value = arg_a;
kono
parents:
diff changeset
1609 unpack_d (&au, &in);
kono
parents:
diff changeset
1610
kono
parents:
diff changeset
1611 sffrac = in.fraction.ll >> D_T_BITOFF;
kono
parents:
diff changeset
1612
kono
parents:
diff changeset
1613 /* We set the lowest guard bit in SFFRAC if we discarded any non
kono
parents:
diff changeset
1614 zero bits. */
kono
parents:
diff changeset
1615 if ((in.fraction.ll & (((UTItype) 1 << D_T_BITOFF) - 1)) != 0)
kono
parents:
diff changeset
1616 sffrac |= 1;
kono
parents:
diff changeset
1617
kono
parents:
diff changeset
1618 return __make_dp (in.class, in.sign, in.normal_exp, sffrac);
kono
parents:
diff changeset
1619 }
kono
parents:
diff changeset
1620 #endif /* L_tf_to_df */
kono
parents:
diff changeset
1621
kono
parents:
diff changeset
1622 #if defined(L_tf_to_sf)
kono
parents:
diff changeset
1623 SFtype
kono
parents:
diff changeset
1624 tf_to_sf (TFtype arg_a)
kono
parents:
diff changeset
1625 {
kono
parents:
diff changeset
1626 fp_number_type in;
kono
parents:
diff changeset
1627 USItype sffrac;
kono
parents:
diff changeset
1628 FLO_union_type au;
kono
parents:
diff changeset
1629
kono
parents:
diff changeset
1630 au.value = arg_a;
kono
parents:
diff changeset
1631 unpack_d (&au, &in);
kono
parents:
diff changeset
1632
kono
parents:
diff changeset
1633 sffrac = in.fraction.ll >> F_T_BITOFF;
kono
parents:
diff changeset
1634
kono
parents:
diff changeset
1635 /* We set the lowest guard bit in SFFRAC if we discarded any non
kono
parents:
diff changeset
1636 zero bits. */
kono
parents:
diff changeset
1637 if ((in.fraction.ll & (((UTItype) 1 << F_T_BITOFF) - 1)) != 0)
kono
parents:
diff changeset
1638 sffrac |= 1;
kono
parents:
diff changeset
1639
kono
parents:
diff changeset
1640 return __make_fp (in.class, in.sign, in.normal_exp, sffrac);
kono
parents:
diff changeset
1641 }
kono
parents:
diff changeset
1642 #endif /* L_tf_to_sf */
kono
parents:
diff changeset
1643 #endif /* TFLOAT */
kono
parents:
diff changeset
1644
kono
parents:
diff changeset
1645 #endif /* ! FLOAT */
kono
parents:
diff changeset
1646 #endif /* !EXTENDED_FLOAT_STUBS */