rework langinfo code for ABI compat and for use by time code
[musl] / src / math / rintf.c
1 /* origin: FreeBSD /usr/src/lib/msun/src/s_rintf.c */
2 /*
3  * Conversion to float by Ian Lance Taylor, Cygnus Support, ian@cygnus.com.
4  */
5 /*
6  * ====================================================
7  * Copyright (C) 1993 by Sun Microsystems, Inc. All rights reserved.
8  *
9  * Developed at SunPro, a Sun Microsystems, Inc. business.
10  * Permission to use, copy, modify, and distribute this
11  * software is freely granted, provided that this notice
12  * is preserved.
13  * ====================================================
14  */
15
16 #include "libm.h"
17
18 static const float
19 TWO23[2] = {
20   8.3886080000e+06, /* 0x4b000000 */
21  -8.3886080000e+06, /* 0xcb000000 */
22 };
23
24 float rintf(float x)
25 {
26         int32_t i0,j0,sx;
27         float w,t;
28
29         GET_FLOAT_WORD(i0, x);
30         sx = (i0>>31) & 1;
31         j0 = ((i0>>23)&0xff) - 0x7f;
32         if (j0 < 23) {
33                 if (j0 < 0) {
34                         if ((i0&0x7fffffff) == 0)
35                                 return x;
36                         STRICT_ASSIGN(float, w, TWO23[sx] + x);
37                         t = w - TWO23[sx];
38                         GET_FLOAT_WORD(i0, t);
39                         SET_FLOAT_WORD(t, (i0&0x7fffffff)|(sx<<31));
40                         return t;
41                 }
42                 STRICT_ASSIGN(float, w, TWO23[sx] + x);
43                 return w - TWO23[sx];
44         }
45         if (j0 == 0x80)
46                 return x+x;  /* inf or NaN */
47         return x;            /* x is integral */
48 }