math_private.h 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370
  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. * from: @(#)fdlibm.h 5.1 93/09/24
  13. * $FreeBSD: src/lib/msun/src/math_private.h,v 1.34 2011/10/21 06:27:56 das Exp $
  14. */
  15. #ifndef _MATH_PRIVATE_H_
  16. #define _MATH_PRIVATE_H_
  17. #include <openlibm_complex.h>
  18. #include "cdefs-compat.h"
  19. #include "types-compat.h"
  20. #include "fpmath.h"
  21. #include <stdint.h>
  22. #include "math_private_openbsd.h"
  23. /*
  24. * The original fdlibm code used statements like:
  25. * n0 = ((*(int*)&one)>>29)^1; * index of high word *
  26. * ix0 = *(n0+(int*)&x); * high word of x *
  27. * ix1 = *((1-n0)+(int*)&x); * low word of x *
  28. * to dig two 32 bit words out of the 64 bit IEEE floating point
  29. * value. That is non-ANSI, and, moreover, the gcc instruction
  30. * scheduler gets it wrong. We instead use the following macros.
  31. * Unlike the original code, we determine the endianness at compile
  32. * time, not at run time; I don't see much benefit to selecting
  33. * endianness at run time.
  34. */
  35. /*
  36. * A union which permits us to convert between a double and two 32 bit
  37. * ints.
  38. */
  39. #if __FLOAT_WORD_ORDER__ == __ORDER_BIG_ENDIAN__
  40. typedef union
  41. {
  42. double value;
  43. struct
  44. {
  45. u_int32_t msw;
  46. u_int32_t lsw;
  47. } parts;
  48. struct
  49. {
  50. u_int64_t w;
  51. } xparts;
  52. } ieee_double_shape_type;
  53. #endif
  54. #if __FLOAT_WORD_ORDER__ == __ORDER_LITTLE_ENDIAN__
  55. typedef union
  56. {
  57. double value;
  58. struct
  59. {
  60. u_int32_t lsw;
  61. u_int32_t msw;
  62. } parts;
  63. struct
  64. {
  65. u_int64_t w;
  66. } xparts;
  67. } ieee_double_shape_type;
  68. #endif
  69. /* Get two 32 bit ints from a double. */
  70. #define EXTRACT_WORDS(ix0,ix1,d) \
  71. do { \
  72. ieee_double_shape_type ew_u; \
  73. ew_u.value = (d); \
  74. (ix0) = ew_u.parts.msw; \
  75. (ix1) = ew_u.parts.lsw; \
  76. } while (0)
  77. /* Get a 64-bit int from a double. */
  78. #define EXTRACT_WORD64(ix,d) \
  79. do { \
  80. ieee_double_shape_type ew_u; \
  81. ew_u.value = (d); \
  82. (ix) = ew_u.xparts.w; \
  83. } while (0)
  84. /* Get the more significant 32 bit int from a double. */
  85. #define GET_HIGH_WORD(i,d) \
  86. do { \
  87. ieee_double_shape_type gh_u; \
  88. gh_u.value = (d); \
  89. (i) = gh_u.parts.msw; \
  90. } while (0)
  91. /* Get the less significant 32 bit int from a double. */
  92. #define GET_LOW_WORD(i,d) \
  93. do { \
  94. ieee_double_shape_type gl_u; \
  95. gl_u.value = (d); \
  96. (i) = gl_u.parts.lsw; \
  97. } while (0)
  98. /* Set a double from two 32 bit ints. */
  99. #define INSERT_WORDS(d,ix0,ix1) \
  100. do { \
  101. ieee_double_shape_type iw_u; \
  102. iw_u.parts.msw = (ix0); \
  103. iw_u.parts.lsw = (ix1); \
  104. (d) = iw_u.value; \
  105. } while (0)
  106. /* Set a double from a 64-bit int. */
  107. #define INSERT_WORD64(d,ix) \
  108. do { \
  109. ieee_double_shape_type iw_u; \
  110. iw_u.xparts.w = (ix); \
  111. (d) = iw_u.value; \
  112. } while (0)
  113. /* Set the more significant 32 bits of a double from an int. */
  114. #define SET_HIGH_WORD(d,v) \
  115. do { \
  116. ieee_double_shape_type sh_u; \
  117. sh_u.value = (d); \
  118. sh_u.parts.msw = (v); \
  119. (d) = sh_u.value; \
  120. } while (0)
  121. /* Set the less significant 32 bits of a double from an int. */
  122. #define SET_LOW_WORD(d,v) \
  123. do { \
  124. ieee_double_shape_type sl_u; \
  125. sl_u.value = (d); \
  126. sl_u.parts.lsw = (v); \
  127. (d) = sl_u.value; \
  128. } while (0)
  129. /*
  130. * A union which permits us to convert between a float and a 32 bit
  131. * int.
  132. */
  133. typedef union
  134. {
  135. float value;
  136. /* FIXME: Assumes 32 bit int. */
  137. unsigned int word;
  138. } ieee_float_shape_type;
  139. /* Get a 32 bit int from a float. */
  140. #define GET_FLOAT_WORD(i,d) \
  141. do { \
  142. ieee_float_shape_type gf_u; \
  143. gf_u.value = (d); \
  144. (i) = gf_u.word; \
  145. } while (0)
  146. /* Set a float from a 32 bit int. */
  147. #define SET_FLOAT_WORD(d,i) \
  148. do { \
  149. ieee_float_shape_type sf_u; \
  150. sf_u.word = (i); \
  151. (d) = sf_u.value; \
  152. } while (0)
  153. /* Get expsign as a 16 bit int from a long double. */
  154. #define GET_LDBL_EXPSIGN(i,d) \
  155. do { \
  156. union IEEEl2bits ge_u; \
  157. ge_u.e = (d); \
  158. (i) = ge_u.xbits.expsign; \
  159. } while (0)
  160. /* Set expsign of a long double from a 16 bit int. */
  161. #define SET_LDBL_EXPSIGN(d,v) \
  162. do { \
  163. union IEEEl2bits se_u; \
  164. se_u.e = (d); \
  165. se_u.xbits.expsign = (v); \
  166. (d) = se_u.e; \
  167. } while (0)
  168. //VBS
  169. #define STRICT_ASSIGN(type, lval, rval) ((lval) = (rval))
  170. /* VBS
  171. #ifdef FLT_EVAL_METHOD
  172. // Attempt to get strict C99 semantics for assignment with non-C99 compilers.
  173. #if FLT_EVAL_METHOD == 0 || __GNUC__ == 0
  174. #define STRICT_ASSIGN(type, lval, rval) ((lval) = (rval))
  175. #else
  176. #define STRICT_ASSIGN(type, lval, rval) do { \
  177. volatile type __lval; \
  178. \
  179. if (sizeof(type) >= sizeof(double)) \
  180. (lval) = (rval); \
  181. else { \
  182. __lval = (rval); \
  183. (lval) = __lval; \
  184. } \
  185. } while (0)
  186. #endif
  187. #endif
  188. */
  189. /*
  190. * Common routine to process the arguments to nan(), nanf(), and nanl().
  191. */
  192. void __scan_nan(u_int32_t *__words, int __num_words, const char *__s);
  193. #ifdef __GNUCLIKE_ASM
  194. /* Asm versions of some functions. */
  195. #ifdef __amd64__
  196. static __inline int
  197. irint(double x)
  198. {
  199. int n;
  200. __asm__("cvtsd2si %1,%0" : "=r" (n) : "x" (x));
  201. return (n);
  202. }
  203. #define HAVE_EFFICIENT_IRINT
  204. #endif
  205. #ifdef __i386__
  206. static __inline int
  207. irint(double x)
  208. {
  209. int n;
  210. __asm__("fistl %0" : "=m" (n) : "t" (x));
  211. return (n);
  212. }
  213. #define HAVE_EFFICIENT_IRINT
  214. #endif
  215. #endif /* __GNUCLIKE_ASM */
  216. /*
  217. * ieee style elementary functions
  218. *
  219. * We rename functions here to improve other sources' diffability
  220. * against fdlibm.
  221. */
  222. #define __ieee754_sqrt sqrt
  223. #define __ieee754_acos acos
  224. #define __ieee754_acosh acosh
  225. #define __ieee754_log log
  226. #define __ieee754_log2 log2
  227. #define __ieee754_atanh atanh
  228. #define __ieee754_asin asin
  229. #define __ieee754_atan2 atan2
  230. #define __ieee754_exp exp
  231. #define __ieee754_cosh cosh
  232. #define __ieee754_fmod fmod
  233. #define __ieee754_pow pow
  234. #define __ieee754_lgamma lgamma
  235. #define __ieee754_lgamma_r lgamma_r
  236. #define __ieee754_log10 log10
  237. #define __ieee754_sinh sinh
  238. #define __ieee754_hypot hypot
  239. #define __ieee754_j0 j0
  240. #define __ieee754_j1 j1
  241. #define __ieee754_y0 y0
  242. #define __ieee754_y1 y1
  243. #define __ieee754_jn jn
  244. #define __ieee754_yn yn
  245. #define __ieee754_remainder remainder
  246. #define __ieee754_sqrtf sqrtf
  247. #define __ieee754_acosf acosf
  248. #define __ieee754_acoshf acoshf
  249. #define __ieee754_logf logf
  250. #define __ieee754_atanhf atanhf
  251. #define __ieee754_asinf asinf
  252. #define __ieee754_atan2f atan2f
  253. #define __ieee754_expf expf
  254. #define __ieee754_coshf coshf
  255. #define __ieee754_fmodf fmodf
  256. #define __ieee754_powf powf
  257. #define __ieee754_lgammaf lgammaf
  258. #define __ieee754_lgammaf_r lgammaf_r
  259. #define __ieee754_log10f log10f
  260. #define __ieee754_log2f log2f
  261. #define __ieee754_sinhf sinhf
  262. #define __ieee754_hypotf hypotf
  263. #define __ieee754_j0f j0f
  264. #define __ieee754_j1f j1f
  265. #define __ieee754_y0f y0f
  266. #define __ieee754_y1f y1f
  267. #define __ieee754_jnf jnf
  268. #define __ieee754_ynf ynf
  269. #define __ieee754_remainderf remainderf
  270. /* fdlibm kernel function */
  271. int __kernel_rem_pio2(double*,double*,int,int,int);
  272. /* double precision kernel functions */
  273. #ifdef INLINE_REM_PIO2
  274. __inline
  275. #endif
  276. int __ieee754_rem_pio2(double,double*);
  277. double __kernel_sin(double,double,int);
  278. double __kernel_cos(double,double);
  279. double __kernel_tan(double,double,int);
  280. double __ldexp_exp(double,int);
  281. double complex __ldexp_cexp(double complex,int);
  282. /* float precision kernel functions */
  283. #ifdef INLINE_REM_PIO2F
  284. __inline
  285. #endif
  286. int __ieee754_rem_pio2f(float,double*);
  287. #ifdef INLINE_KERNEL_SINDF
  288. __inline
  289. #endif
  290. float __kernel_sindf(double);
  291. #ifdef INLINE_KERNEL_COSDF
  292. __inline
  293. #endif
  294. float __kernel_cosdf(double);
  295. #ifdef INLINE_KERNEL_TANDF
  296. __inline
  297. #endif
  298. float __kernel_tandf(double,int);
  299. float __ldexp_expf(float,int);
  300. float complex __ldexp_cexpf(float complex,int);
  301. /* long double precision kernel functions */
  302. long double __kernel_sinl(long double, long double, int);
  303. long double __kernel_cosl(long double, long double);
  304. long double __kernel_tanl(long double, long double, int);
  305. #undef OLM_DLLEXPORT
  306. #ifdef _WIN32
  307. # ifdef IMPORT_EXPORTS
  308. # define OLM_DLLEXPORT __declspec(dllimport)
  309. # else
  310. # define OLM_DLLEXPORT __declspec(dllexport)
  311. # endif
  312. #else
  313. #define OLM_DLLEXPORT __attribute__ ((visibility("default")))
  314. #endif
  315. #endif /* !_MATH_PRIVATE_H_ */