Imt.Base C++ API V4.1.1.0
Loading...
Searching...
No Matches
math_private.h
Go to the documentation of this file.
1/*
2 * ====================================================
3 * Copyright (C) 1993 by Sun Microsystems, Inc. All rights reserved.
4 *
5 * Developed at SunPro, a Sun Microsystems, Inc. business.
6 * Permission to use, copy, modify, and distribute this
7 * software is freely granted, provided that this notice
8 * is preserved.
9 * ====================================================
10 */
11
12/*
13 * from: @(#)fdlibm.h 5.1 93/09/24
14 * $FreeBSD: src/lib/msun/src/math_private.h,v 1.34 2011/10/21 06:27:56 das Exp $
15 */
16
17#ifndef _MATH_PRIVATE_H_
18#define _MATH_PRIVATE_H_
19
20#include "../inc/openlibm_defs.h"
21#include "../inc/cdefs-compat.h"
22#include "../inc/types-compat.h"
23#include "fpmath.h"
24#include <stdint.h>
26
27/*
28 * The original fdlibm code used statements like:
29 * n0 = ((*(int*)&one)>>29)^1; * index of high word *
30 * ix0 = *(n0+(int*)&x); * high word of x *
31 * ix1 = *((1-n0)+(int*)&x); * low word of x *
32 * to dig two 32 bit words out of the 64 bit IEEE floating point
33 * value. That is non-ANSI, and, moreover, the gcc instruction
34 * scheduler gets it wrong. We instead use the following macros.
35 * Unlike the original code, we determine the endianness at compile
36 * time, not at run time; I don't see much benefit to selecting
37 * endianness at run time.
38 */
39
40/*
41 * A union which permits us to convert between a double and two 32 bit
42 * ints.
43 */
44
45#if __FLOAT_WORD_ORDER__ == __ORDER_BIG_ENDIAN__
46
47typedef union
48{
49 double value;
50 struct
51 {
54 } parts;
55 struct
56 {
58 } xparts;
60
61#endif
62
63#if __FLOAT_WORD_ORDER__ == __ORDER_LITTLE_ENDIAN__
64
65typedef union
66{
67 double value;
68 struct
69 {
70 u_int32_t lsw;
71 u_int32_t msw;
72 } parts;
73 struct
74 {
75 u_int64_t w;
76 } xparts;
78
79#endif
80
81/* Get two 32 bit ints from a double. */
82
83#define EXTRACT_WORDS(ix0,ix1,d) \
84do { \
85 ieee_double_shape_type ew_u; \
86 ew_u.value = (d); \
87 (ix0) = ew_u.parts.msw; \
88 (ix1) = ew_u.parts.lsw; \
89} while (0)
90
91/* Get a 64-bit int from a double. */
92#define EXTRACT_WORD64(ix,d) \
93do { \
94 ieee_double_shape_type ew_u; \
95 ew_u.value = (d); \
96 (ix) = ew_u.xparts.w; \
97} while (0)
98
99/* Get the more significant 32 bit int from a double. */
100
101#define GET_HIGH_WORD(i,d) \
102do { \
103 ieee_double_shape_type gh_u; \
104 gh_u.value = (d); \
105 (i) = gh_u.parts.msw; \
106} while (0)
107
108/* Get the less significant 32 bit int from a double. */
109
110#define GET_LOW_WORD(i,d) \
111do { \
112 ieee_double_shape_type gl_u; \
113 gl_u.value = (d); \
114 (i) = gl_u.parts.lsw; \
115} while (0)
116
117/* Set a double from two 32 bit ints. */
118
119#define INSERT_WORDS(d,ix0,ix1) \
120do { \
121 ieee_double_shape_type iw_u; \
122 iw_u.parts.msw = (ix0); \
123 iw_u.parts.lsw = (ix1); \
124 (d) = iw_u.value; \
125} while (0)
126
127/* Set a double from a 64-bit int. */
128#define INSERT_WORD64(d,ix) \
129do { \
130 ieee_double_shape_type iw_u; \
131 iw_u.xparts.w = (ix); \
132 (d) = iw_u.value; \
133} while (0)
134
135/* Set the more significant 32 bits of a double from an int. */
136
137#define SET_HIGH_WORD(d,v) \
138do { \
139 ieee_double_shape_type sh_u; \
140 sh_u.value = (d); \
141 sh_u.parts.msw = (v); \
142 (d) = sh_u.value; \
143} while (0)
144
145/* Set the less significant 32 bits of a double from an int. */
146
147#define SET_LOW_WORD(d,v) \
148do { \
149 ieee_double_shape_type sl_u; \
150 sl_u.value = (d); \
151 sl_u.parts.lsw = (v); \
152 (d) = sl_u.value; \
153} while (0)
154
155/*
156 * A union which permits us to convert between a float and a 32 bit
157 * int.
158 */
159
160typedef union
161{
162 float value;
163 /* FIXME: Assumes 32 bit int. */
164 unsigned int word;
166
167/* Get a 32 bit int from a float. */
168
169#define GET_FLOAT_WORD(i,d) \
170do { \
171 ieee_float_shape_type gf_u; \
172 gf_u.value = (d); \
173 (i) = gf_u.word; \
174} while (0)
175
176/* Set a float from a 32 bit int. */
177
178#define SET_FLOAT_WORD(d,i) \
179do { \
180 ieee_float_shape_type sf_u; \
181 sf_u.word = (i); \
182 (d) = sf_u.value; \
183} while (0)
184
185/* Get expsign as a 16 bit int from a long double. */
186
187#define GET_LDBL_EXPSIGN(i,d) \
188do { \
189 union IEEEl2bits ge_u; \
190 ge_u.e = (d); \
191 (i) = ge_u.xbits.expsign; \
192} while (0)
193
194/* Set expsign of a long double from a 16 bit int. */
195
196#define SET_LDBL_EXPSIGN(d,v) \
197do { \
198 union IEEEl2bits se_u; \
199 se_u.e = (d); \
200 se_u.xbits.expsign = (v); \
201 (d) = se_u.e; \
202} while (0)
203
204
205#ifndef __FreeBSD__
206#define STRICT_ASSIGN(type, lval, rval) ((lval) = (rval))
207#else
208#ifdef FLT_EVAL_METHOD
209// Attempt to get strict C99 semantics for assignment with non-C99 compilers.
210#if FLT_EVAL_METHOD == 0 || __GNUC__ == 0
211#define STRICT_ASSIGN(type, lval, rval) ((lval) = (rval))
212#else
213#define STRICT_ASSIGN(type, lval, rval) do { \
214 volatile type __lval; \
215 \
216 if (sizeof(type) >= sizeof(long double)) \
217 (lval) = (rval); \
218 else { \
219 __lval = (rval); \
220 (lval) = __lval; \
221 } \
222} while (0)
223#endif
224#endif
225#endif
226
227/*
228 * Common routine to process the arguments to nan(), nanf(), and nanl().
229 */
230void __scan_nan(u_int32_t *__words, int __num_words, const char *__s);
231
232/*
233 * Mix 1 or 2 NaNs. First add 0 to each arg. This normally just turns
234 * signaling NaNs into quiet NaNs by setting a quiet bit. We do this
235 * because we want to never return a signaling NaN, and also because we
236 * don't want the quiet bit to affect the result. Then mix the converted
237 * args using addition. The result is typically the arg whose mantissa
238 * bits (considered as in integer) are largest.
239 *
240 * Technical complications: the result in bits might depend on the precision
241 * and/or on compiler optimizations, especially when different register sets
242 * are used for different precisions. Try to make the result not depend on
243 * at least the precision by always doing the main mixing step in long double
244 * precision. Try to reduce dependencies on optimizations by adding the
245 * the 0's in different precisions (unless everything is in long double
246 * precision).
247 */
248#define nan_mix(x, y) (((x) + 0.0L) + ((y) + 0))
249
250#ifdef __GNUCLIKE_ASM
251
252/* Asm versions of some functions. */
253
254#ifdef __amd64__
255static __inline int
256irint(double x)
257{
258 int n;
259
260 __asm__("cvtsd2si %1,%0" : "=r" (n) : "x" (x));
261 return (n);
262}
263#define HAVE_EFFICIENT_IRINT
264#endif
265
266#ifdef __i386__
267static __inline int
268irint(double x)
269{
270 int n;
271
272 __asm__("fistl %0" : "=m" (n) : "t" (x));
273 return (n);
274}
275#define HAVE_EFFICIENT_IRINT
276#endif
277
278#endif /* __GNUCLIKE_ASM */
279
280/*
281 * ieee style elementary functions
282 *
283 * We rename functions here to improve other sources' diffability
284 * against fdlibm.
285 */
286#define __ieee754_sqrt c_sqrt
287#define __ieee754_acos c_acos
288#define __ieee754_acosh c_acosh
289#define __ieee754_log c_log
290#define __ieee754_log2 c_log2
291#define __ieee754_atanh c_atanh
292#define __ieee754_asin c_asin
293#define __ieee754_atan2 c_atan2
294#define __ieee754_exp c_exp
295#define __ieee754_cosh c_cosh
296#define __ieee754_fmod c_fmod
297#define __ieee754_pow c_pow
298#define __ieee754_lgamma c_lgamma
299#define __ieee754_lgamma_r c_lgamma_r
300#define __ieee754_log10 c_log10
301#define __ieee754_sinh c_sinh
302#define __ieee754_hypot c_hypot
303#define __ieee754_j0 c_j0
304#define __ieee754_j1 c_j1
305#define __ieee754_y0 c_y0
306#define __ieee754_y1 c_y1
307#define __ieee754_jn c_jn
308#define __ieee754_yn c_yn
309#define __ieee754_remainder c_remainder
310#define __ieee754_sqrtf c_sqrtf
311#define __ieee754_acosf c_acosf
312#define __ieee754_acoshf c_acoshf
313#define __ieee754_logf c_logf
314#define __ieee754_atanhf c_atanhf
315#define __ieee754_asinf c_asinf
316#define __ieee754_atan2f c_atan2f
317#define __ieee754_expf c_expf
318#define __ieee754_coshf c_coshf
319#define __ieee754_fmodf c_fmodf
320#define __ieee754_powf c_powf
321#define __ieee754_lgammaf c_lgammaf
322#define __ieee754_lgammaf_r c_lgammaf_r
323#define __ieee754_log10f c_log10f
324#define __ieee754_log2f c_log2f
325#define __ieee754_sinhf c_sinhf
326#define __ieee754_hypotf c_hypotf
327#define __ieee754_j0f c_j0f
328#define __ieee754_j1f c_j1f
329#define __ieee754_y0f c_y0f
330#define __ieee754_y1f c_y1f
331#define __ieee754_jnf c_jnf
332#define __ieee754_ynf c_ynf
333#define __ieee754_remainderf c_remainderf
334
335/* fdlibm kernel function */
336int __kernel_rem_pio2(double*,double*,int,int,int);
337
338/* double precision kernel functions */
339int __ieee754_rem_pio2(double,double*);
340double __kernel_sin(double,double,int);
341double __kernel_cos(double,double);
342double __kernel_tan(double,double,int);
343double __ldexp_exp(double,int);
344/* float precision kernel functions */
345int __ieee754_rem_pio2f(float,double*);
346#ifdef INLINE_KERNEL_SINDF
348#endif
349float __kernel_sindf(double);
350#ifdef INLINE_KERNEL_COSDF
352#endif
353float __kernel_cosdf(double);
354#ifdef INLINE_KERNEL_TANDF
356#endif
357float __kernel_tandf(double,int);
358float __ldexp_expf(float,int);
359
360/* long double precision kernel functions */
361long double __kernel_sinl(long double, long double, int);
362long double __kernel_cosl(long double, long double);
363long double __kernel_tanl(long double, long double, int);
364
365#endif /* !_MATH_PRIVATE_H_ */
#define __inline
Definition bsd_cdefs.h:73
long double __kernel_tanl(long double, long double, int)
long double __kernel_sinl(long double, long double, int)
double __kernel_sin(double, double, int)
float __ldexp_expf(float, int)
int __ieee754_rem_pio2f(float, double *)
void __scan_nan(u_int32_t *__words, int __num_words, const char *__s)
double __ldexp_exp(double, int)
double __kernel_cos(double, double)
long double __kernel_cosl(long double, long double)
int __kernel_rem_pio2(double *, double *, int, int, int)
double __kernel_tan(double, double, int)
float __kernel_tandf(double, int)
float __kernel_sindf(double)
float __kernel_cosdf(double)
int __ieee754_rem_pio2(double, double *)
uint32_t u_int32_t
Definition types-compat.h:8
uint64_t u_int64_t
Definition types-compat.h:9