]>
git.saurik.com Git - apple/javascriptcore.git/blob - wtf/MathExtras.h
6da125aa10fbf8a7892d17b44ef08c5c6d9b7ddf
2 * Copyright (C) 2006, 2007, 2008, 2009, 2010 Apple Inc. All rights reserved.
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions
7 * 1. Redistributions of source code must retain the above copyright
8 * notice, this list of conditions and the following disclaimer.
9 * 2. Redistributions in binary form must reproduce the above copyright
10 * notice, this list of conditions and the following disclaimer in the
11 * documentation and/or other materials provided with the distribution.
13 * THIS SOFTWARE IS PROVIDED BY APPLE COMPUTER, INC. ``AS IS'' AND ANY
14 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE COMPUTER, INC. OR
17 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
18 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
19 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
20 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
21 * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
23 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26 #ifndef WTF_MathExtras_h
27 #define WTF_MathExtras_h
40 #include <sys/types.h>
41 #include <machine/ieee.h>
52 const double piDouble
= 3.14159265358979323846;
53 const float piFloat
= 3.14159265358979323846f
;
55 const double piDouble
= M_PI
;
56 const float piFloat
= static_cast<float>(M_PI
);
60 const double piOverTwoDouble
= 1.57079632679489661923;
61 const float piOverTwoFloat
= 1.57079632679489661923f
;
63 const double piOverTwoDouble
= M_PI_2
;
64 const float piOverTwoFloat
= static_cast<float>(M_PI_2
);
68 const double piOverFourDouble
= 0.785398163397448309616;
69 const float piOverFourFloat
= 0.785398163397448309616f
;
71 const double piOverFourDouble
= M_PI_4
;
72 const float piOverFourFloat
= static_cast<float>(M_PI_4
);
77 // Work around a bug in the Mac OS X libc where ceil(-0.1) return +0.
78 inline double wtf_ceil(double x
) { return copysign(ceil(x
), x
); }
80 #define ceil(x) wtf_ceil(x)
87 inline bool isfinite(double x
) { return finite(x
) && !isnand(x
); }
90 inline bool isinf(double x
) { return !finite(x
) && !isnand(x
); }
93 inline bool signbit(double x
) { return copysign(1.0, x
) < 0; }
101 inline bool isfinite(double x
) { return finite(x
); }
104 inline bool signbit(double x
) { struct ieee_double
*p
= (struct ieee_double
*)&x
; return p
->dbl_sign
; }
109 #if COMPILER(MSVC) || (COMPILER(RVCT) && !(RVCT_VERSION_AT_LEAST(3, 0, 0, 0)))
111 // We must not do 'num + 0.5' or 'num - 0.5' because they can cause precision loss.
112 static double round(double num
)
114 double integer
= ceil(num
);
116 return integer
- num
> 0.5 ? integer
- 1.0 : integer
;
117 return integer
- num
>= 0.5 ? integer
- 1.0 : integer
;
119 static float roundf(float num
)
121 float integer
= ceilf(num
);
123 return integer
- num
> 0.5f
? integer
- 1.0f
: integer
;
124 return integer
- num
>= 0.5f
? integer
- 1.0f
: integer
;
126 inline long long llround(double num
) { return static_cast<long long>(round(num
)); }
127 inline long long llroundf(float num
) { return static_cast<long long>(roundf(num
)); }
128 inline long lround(double num
) { return static_cast<long>(round(num
)); }
129 inline long lroundf(float num
) { return static_cast<long>(roundf(num
)); }
130 inline double trunc(double num
) { return num
> 0 ? floor(num
) : ceil(num
); }
135 // The 64bit version of abs() is already defined in stdlib.h which comes with VC10
136 #if COMPILER(MSVC9_OR_LOWER)
137 inline long long abs(long long num
) { return _abs64(num
); }
140 inline bool isinf(double num
) { return !_finite(num
) && !_isnan(num
); }
141 inline bool isnan(double num
) { return !!_isnan(num
); }
142 inline bool signbit(double num
) { return _copysign(1.0, num
) < 0; }
144 inline double nextafter(double x
, double y
) { return _nextafter(x
, y
); }
145 inline float nextafterf(float x
, float y
) { return x
> y
? x
- FLT_EPSILON
: x
+ FLT_EPSILON
; }
147 inline double copysign(double x
, double y
) { return _copysign(x
, y
); }
148 inline int isfinite(double x
) { return _finite(x
); }
150 // MSVC's math.h does not currently supply log2.
151 inline double log2(double num
)
153 // This constant is roughly M_LN2, which is not provided by default on Windows.
154 return log(num
) / 0.693147180559945309417232121458176568;
157 // Work around a bug in Win, where atan2(+-infinity, +-infinity) yields NaN instead of specific values.
158 inline double wtf_atan2(double x
, double y
)
160 double posInf
= std::numeric_limits
<double>::infinity();
161 double negInf
= -std::numeric_limits
<double>::infinity();
162 double nan
= std::numeric_limits
<double>::quiet_NaN();
166 if (x
== posInf
&& y
== posInf
)
167 result
= piOverFourDouble
;
168 else if (x
== posInf
&& y
== negInf
)
169 result
= 3 * piOverFourDouble
;
170 else if (x
== negInf
&& y
== posInf
)
171 result
= -piOverFourDouble
;
172 else if (x
== negInf
&& y
== negInf
)
173 result
= -3 * piOverFourDouble
;
175 result
= ::atan2(x
, y
);
180 // Work around a bug in the Microsoft CRT, where fmod(x, +-infinity) yields NaN instead of x.
181 inline double wtf_fmod(double x
, double y
) { return (!isinf(x
) && isinf(y
)) ? x
: fmod(x
, y
); }
183 // Work around a bug in the Microsoft CRT, where pow(NaN, 0) yields NaN instead of 1.
184 inline double wtf_pow(double x
, double y
) { return y
== 0 ? 1 : pow(x
, y
); }
186 #define atan2(x, y) wtf_atan2(x, y)
187 #define fmod(x, y) wtf_fmod(x, y)
188 #define pow(x, y) wtf_pow(x, y)
190 #endif // COMPILER(MSVC)
192 inline double deg2rad(double d
) { return d
* piDouble
/ 180.0; }
193 inline double rad2deg(double r
) { return r
* 180.0 / piDouble
; }
194 inline double deg2grad(double d
) { return d
* 400.0 / 360.0; }
195 inline double grad2deg(double g
) { return g
* 360.0 / 400.0; }
196 inline double turn2deg(double t
) { return t
* 360.0; }
197 inline double deg2turn(double d
) { return d
/ 360.0; }
198 inline double rad2grad(double r
) { return r
* 200.0 / piDouble
; }
199 inline double grad2rad(double g
) { return g
* piDouble
/ 200.0; }
201 inline float deg2rad(float d
) { return d
* piFloat
/ 180.0f
; }
202 inline float rad2deg(float r
) { return r
* 180.0f
/ piFloat
; }
203 inline float deg2grad(float d
) { return d
* 400.0f
/ 360.0f
; }
204 inline float grad2deg(float g
) { return g
* 360.0f
/ 400.0f
; }
205 inline float turn2deg(float t
) { return t
* 360.0f
; }
206 inline float deg2turn(float d
) { return d
/ 360.0f
; }
207 inline float rad2grad(float r
) { return r
* 200.0f
/ piFloat
; }
208 inline float grad2rad(float g
) { return g
* piFloat
/ 200.0f
; }
210 inline int clampToInteger(double x
)
212 const double intMax
= static_cast<double>(std::numeric_limits
<int>::max());
213 const double intMin
= static_cast<double>(std::numeric_limits
<int>::min());
216 return std::numeric_limits
<int>::max();
218 return std::numeric_limits
<int>::min();
219 return static_cast<int>(x
);
222 inline float clampToFloat(double x
)
224 const double floatMax
= static_cast<double>(std::numeric_limits
<float>::max());
225 const double floatMin
= -static_cast<double>(std::numeric_limits
<float>::max());
228 return std::numeric_limits
<float>::max();
230 return -std::numeric_limits
<float>::max();
231 return static_cast<float>(x
);
234 inline int clampToPositiveInteger(double x
)
236 const double intMax
= static_cast<double>(std::numeric_limits
<int>::max());
239 return std::numeric_limits
<int>::max();
242 return static_cast<int>(x
);
245 inline int clampToInteger(float x
)
247 const float intMax
= static_cast<float>(std::numeric_limits
<int>::max());
248 const float intMin
= static_cast<float>(std::numeric_limits
<int>::min());
251 return std::numeric_limits
<int>::max();
253 return std::numeric_limits
<int>::min();
254 return static_cast<int>(x
);
257 inline int clampToPositiveInteger(float x
)
259 const float intMax
= static_cast<float>(std::numeric_limits
<int>::max());
262 return std::numeric_limits
<int>::max();
265 return static_cast<int>(x
);
268 inline int clampToInteger(unsigned x
)
270 const unsigned intMax
= static_cast<unsigned>(std::numeric_limits
<int>::max());
273 return std::numeric_limits
<int>::max();
274 return static_cast<int>(x
);
277 inline bool isWithinIntRange(float x
)
279 return x
> static_cast<float>(std::numeric_limits
<int>::min()) && x
< static_cast<float>(std::numeric_limits
<int>::max());
282 #if !COMPILER(MSVC) && !(COMPILER(RVCT) && PLATFORM(BREWMP)) && !OS(SOLARIS) && !OS(SYMBIAN)
289 #endif // #ifndef WTF_MathExtras_h