1 // (C) Copyright John Maddock 2007.
2 // Use, modification and distribution are subject to the
3 // Boost Software License, Version 1.0. (See accompanying file
4 // LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
5
6 #include <pch.hpp>
7
8 #include <boost/math/concepts/real_concept.hpp>
9 #define BOOST_TEST_MAIN
10 #include <boost/test/unit_test.hpp>
11 #include <boost/math/tools/test.hpp>
12 #include <boost/test/tools/floating_point_comparison.hpp>
13 #include <boost/math/special_functions/round.hpp>
14 #include <boost/math/special_functions/next.hpp>
15 #include <boost/math/special_functions/trunc.hpp>
16 #include <boost/math/special_functions/modf.hpp>
17 #include <boost/math/special_functions/sign.hpp>
18 #include <boost/random/mersenne_twister.hpp>
19 #include <iostream>
20 #include <iomanip>
21
22 boost::mt19937 rng;
23
24 template <class T>
get_random()25 T get_random()
26 {
27 //
28 // Fill all the bits in T with random values,
29 // likewise set the exponent to a random value
30 // that will still fit inside a T, and always
31 // have a remainder as well as an integer part.
32 //
33 int bits = boost::math::tools::digits<T>();
34 int shift = 0;
35 int exponent = rng() % (bits - 4);
36 T result = 0;
37 while(bits > 0)
38 {
39 result += ldexp(static_cast<T>(rng()), shift);
40 shift += std::numeric_limits<int>::digits;
41 bits -= std::numeric_limits<int>::digits;
42 }
43 return rng() & 1u ? -ldexp(frexp(result, &bits), exponent) : ldexp(frexp(result, &bits), exponent);
44 }
45
46 template <class T, class U>
check_within_half(T a,U u)47 void check_within_half(T a, U u)
48 {
49 BOOST_MATH_STD_USING
50 if(fabs(a-u) > 0.5f)
51 {
52 BOOST_ERROR("Rounded result differed by more than 0.5 from the original");
53 std::cerr << "Values were: " << std::setprecision(35) << std::setw(40)
54 << std::left << a << u << std::endl;
55 }
56 if((fabs(a - u) == 0.5f) && (fabs(static_cast<T>(u)) < fabs(a)))
57 {
58 BOOST_ERROR("Rounded result was towards zero with boost::round");
59 std::cerr << "Values were: " << std::setprecision(35) << std::setw(40)
60 << std::left << a << u << std::endl;
61 }
62 }
63
64 //
65 // We may not have an abs overload for long long so provide a fall back:
66 //
67 template <class T>
safe_abs(T const & v...)68 inline T safe_abs(T const& v ...)
69 {
70 return v < 0 ? -v : v;
71 }
72
73 template <class T, class U>
check_trunc_result(T a,U u)74 void check_trunc_result(T a, U u)
75 {
76 BOOST_MATH_STD_USING
77 if(fabs(a-u) >= 1)
78 {
79 BOOST_ERROR("Rounded result differed by more than 1 from the original");
80 std::cerr << "Values were: " << std::setprecision(35) << std::setw(40)
81 << std::left << a << u << std::endl;
82 }
83 if(abs(a) < safe_abs(u))
84 {
85 BOOST_ERROR("Truncated result had larger absolute value than the original");
86 std::cerr << "Values were: " << std::setprecision(35) << std::setw(40)
87 << std::left << a << u << std::endl;
88 }
89 if(fabs(static_cast<T>(u)) > fabs(a))
90 {
91 BOOST_ERROR("Rounded result was away from zero with boost::trunc");
92 std::cerr << "Values were: " << std::setprecision(35) << std::setw(40)
93 << std::left << a << u << std::endl;
94 }
95 }
96
97 template <class T, class U>
check_modf_result(T a,T fract,U ipart)98 void check_modf_result(T a, T fract, U ipart)
99 {
100 BOOST_MATH_STD_USING
101 if(fract + ipart != a)
102 {
103 BOOST_ERROR("Fractional and integer results do not add up to the original value");
104 std::cerr << "Values were: " << std::setprecision(35) << " "
105 << std::left << a << ipart << " " << fract << std::endl;
106 }
107 if((boost::math::sign(a) != boost::math::sign(fract)) && boost::math::sign(fract))
108 {
109 BOOST_ERROR("Original and fractional parts have differing signs");
110 std::cerr << "Values were: " << std::setprecision(35) << " "
111 << std::left << a << ipart << " " << fract << std::endl;
112 }
113 if((boost::math::sign(a) != boost::math::sign(ipart)) && boost::math::sign(ipart))
114 {
115 BOOST_ERROR("Original and integer parts have differing signs");
116 std::cerr << "Values were: " << std::setprecision(35) << " "
117 << std::left << a << ipart << " " << ipart << std::endl;
118 }
119 if(fabs(a-ipart) >= 1)
120 {
121 BOOST_ERROR("Rounded result differed by more than 1 from the original");
122 std::cerr << "Values were: " << std::setprecision(35) << std::setw(40)
123 << std::left << a << ipart << std::endl;
124 }
125 }
126
127 template <class T>
test_round_number(T arg)128 void test_round_number(T arg)
129 {
130 BOOST_MATH_STD_USING
131 #ifdef BOOST_HAS_LONG_LONG
132 using boost::math::llround; using boost::math::lltrunc;
133 #endif
134
135 T r = round(arg);
136 check_within_half(arg, r);
137 r = trunc(arg);
138 check_trunc_result(arg, r);
139 T frac = boost::math::modf(arg, &r);
140 check_modf_result(arg, frac, r);
141
142 if(abs(r) < (std::numeric_limits<int>::max)())
143 {
144 int i = iround(arg);
145 check_within_half(arg, i);
146 i = itrunc(arg);
147 check_trunc_result(arg, T(i));
148 r = boost::math::modf(arg, &i);
149 check_modf_result(arg, r, i);
150 }
151 if(std::numeric_limits<T>::digits >= std::numeric_limits<int>::digits)
152 {
153 int si = iround(static_cast<T>((std::numeric_limits<int>::max)()));
154 check_within_half(static_cast<T>((std::numeric_limits<int>::max)()), si);
155 si = iround(static_cast<T>((std::numeric_limits<int>::min)()));
156 check_within_half(static_cast<T>((std::numeric_limits<int>::min)()), si);
157 si = itrunc(static_cast<T>((std::numeric_limits<int>::max)()));
158 check_trunc_result(static_cast<T>((std::numeric_limits<int>::max)()), T(si));
159 si = itrunc(static_cast<T>((std::numeric_limits<int>::min)()));
160 check_trunc_result(static_cast<T>((std::numeric_limits<int>::min)()), T(si));
161 }
162 if(abs(r) < (std::numeric_limits<long>::max)())
163 {
164 long l = lround(arg);
165 check_within_half(arg, l);
166 l = ltrunc(arg);
167 check_trunc_result(arg, T(l));
168 r = boost::math::modf(arg, &l);
169 check_modf_result(arg, r, l);
170 }
171 if(std::numeric_limits<T>::digits >= std::numeric_limits<long>::digits)
172 {
173 long k = lround(static_cast<T>((std::numeric_limits<long>::max)()));
174 check_within_half(static_cast<T>((std::numeric_limits<long>::max)()), k);
175 k = lround(static_cast<T>((std::numeric_limits<long>::min)()));
176 check_within_half(static_cast<T>((std::numeric_limits<long>::min)()), k);
177 k = ltrunc(static_cast<T>((std::numeric_limits<long>::max)()));
178 check_trunc_result(static_cast<T>((std::numeric_limits<long>::max)()), T(k));
179 k = ltrunc(static_cast<T>((std::numeric_limits<long>::min)()));
180 check_trunc_result(static_cast<T>((std::numeric_limits<long>::min)()), T(k));
181 }
182
183 #ifdef BOOST_HAS_LONG_LONG
184 if(abs(r) < (std::numeric_limits<boost::long_long_type>::max)())
185 {
186 boost::long_long_type ll = llround(arg);
187 check_within_half(arg, ll);
188 ll = lltrunc(arg);
189 check_trunc_result(arg, T(ll));
190 r = boost::math::modf(arg, &ll);
191 check_modf_result(arg, r, ll);
192 }
193 if(std::numeric_limits<T>::digits >= std::numeric_limits<boost::long_long_type>::digits)
194 {
195 boost::long_long_type j = llround(static_cast<T>((std::numeric_limits<boost::long_long_type>::max)()));
196 check_within_half(static_cast<T>((std::numeric_limits<boost::long_long_type>::max)()), j);
197 j = llround(static_cast<T>((std::numeric_limits<boost::long_long_type>::min)()));
198 check_within_half(static_cast<T>((std::numeric_limits<boost::long_long_type>::min)()), j);
199 j = lltrunc(static_cast<T>((std::numeric_limits<boost::long_long_type>::max)()));
200 check_trunc_result(static_cast<T>((std::numeric_limits<boost::long_long_type>::max)()), T(j));
201 j = lltrunc(static_cast<T>((std::numeric_limits<boost::long_long_type>::min)()));
202 check_trunc_result(static_cast<T>((std::numeric_limits<boost::long_long_type>::min)()), T(j));
203 }
204 #endif
205 }
206
207 template <class T>
test_round(T,const char * name)208 void test_round(T, const char* name )
209 {
210 BOOST_MATH_STD_USING
211 #ifdef BOOST_HAS_LONG_LONG
212 using boost::math::llround; using boost::math::lltrunc;
213 #endif
214
215 std::cout << "Testing rounding with type " << name << std::endl;
216
217 for(int i = 0; i < 1000; ++i)
218 {
219 T arg = get_random<T>();
220 test_round_number<T>(arg);
221 }
222 //
223 // Finish off by testing the error handlers:
224 //
225 BOOST_MATH_CHECK_THROW(iround(static_cast<T>(1e20)), boost::math::rounding_error);
226 BOOST_MATH_CHECK_THROW(iround(static_cast<T>(-1e20)), boost::math::rounding_error);
227 BOOST_MATH_CHECK_THROW(lround(static_cast<T>(1e20)), boost::math::rounding_error);
228 BOOST_MATH_CHECK_THROW(lround(static_cast<T>(-1e20)), boost::math::rounding_error);
229 #ifdef BOOST_HAS_LONG_LONG
230 BOOST_MATH_CHECK_THROW(llround(static_cast<T>(1e20)), boost::math::rounding_error);
231 BOOST_MATH_CHECK_THROW(llround(static_cast<T>(-1e20)), boost::math::rounding_error);
232 #endif
233 if(std::numeric_limits<T>::has_infinity)
234 {
235 BOOST_MATH_CHECK_THROW(round(std::numeric_limits<T>::infinity()), boost::math::rounding_error);
236 BOOST_MATH_CHECK_THROW(iround(std::numeric_limits<T>::infinity()), boost::math::rounding_error);
237 BOOST_MATH_CHECK_THROW(iround(-std::numeric_limits<T>::infinity()), boost::math::rounding_error);
238 BOOST_MATH_CHECK_THROW(lround(std::numeric_limits<T>::infinity()), boost::math::rounding_error);
239 BOOST_MATH_CHECK_THROW(lround(-std::numeric_limits<T>::infinity()), boost::math::rounding_error);
240 #ifdef BOOST_HAS_LONG_LONG
241 BOOST_MATH_CHECK_THROW(llround(std::numeric_limits<T>::infinity()), boost::math::rounding_error);
242 BOOST_MATH_CHECK_THROW(llround(-std::numeric_limits<T>::infinity()), boost::math::rounding_error);
243 #endif
244 }
245 if(std::numeric_limits<T>::has_quiet_NaN)
246 {
247 BOOST_MATH_CHECK_THROW(round(std::numeric_limits<T>::quiet_NaN()), boost::math::rounding_error);
248 BOOST_MATH_CHECK_THROW(iround(std::numeric_limits<T>::quiet_NaN()), boost::math::rounding_error);
249 BOOST_MATH_CHECK_THROW(lround(std::numeric_limits<T>::quiet_NaN()), boost::math::rounding_error);
250 #ifdef BOOST_HAS_LONG_LONG
251 BOOST_MATH_CHECK_THROW(llround(std::numeric_limits<T>::quiet_NaN()), boost::math::rounding_error);
252 #endif
253 }
254 BOOST_MATH_CHECK_THROW(itrunc(static_cast<T>(1e20)), boost::math::rounding_error);
255 BOOST_MATH_CHECK_THROW(itrunc(static_cast<T>(-1e20)), boost::math::rounding_error);
256 BOOST_MATH_CHECK_THROW(ltrunc(static_cast<T>(1e20)), boost::math::rounding_error);
257 BOOST_MATH_CHECK_THROW(ltrunc(static_cast<T>(-1e20)), boost::math::rounding_error);
258 #ifdef BOOST_HAS_LONG_LONG
259 BOOST_MATH_CHECK_THROW(lltrunc(static_cast<T>(1e20)), boost::math::rounding_error);
260 BOOST_MATH_CHECK_THROW(lltrunc(static_cast<T>(-1e20)), boost::math::rounding_error);
261 #endif
262 if(std::numeric_limits<T>::has_infinity)
263 {
264 BOOST_MATH_CHECK_THROW(trunc(std::numeric_limits<T>::infinity()), boost::math::rounding_error);
265 BOOST_MATH_CHECK_THROW(itrunc(std::numeric_limits<T>::infinity()), boost::math::rounding_error);
266 BOOST_MATH_CHECK_THROW(itrunc(-std::numeric_limits<T>::infinity()), boost::math::rounding_error);
267 BOOST_MATH_CHECK_THROW(ltrunc(std::numeric_limits<T>::infinity()), boost::math::rounding_error);
268 BOOST_MATH_CHECK_THROW(ltrunc(-std::numeric_limits<T>::infinity()), boost::math::rounding_error);
269 #ifdef BOOST_HAS_LONG_LONG
270 BOOST_MATH_CHECK_THROW(lltrunc(std::numeric_limits<T>::infinity()), boost::math::rounding_error);
271 BOOST_MATH_CHECK_THROW(lltrunc(-std::numeric_limits<T>::infinity()), boost::math::rounding_error);
272 #endif
273 }
274 if(std::numeric_limits<T>::has_quiet_NaN)
275 {
276 BOOST_MATH_CHECK_THROW(trunc(std::numeric_limits<T>::quiet_NaN()), boost::math::rounding_error);
277 BOOST_MATH_CHECK_THROW(itrunc(std::numeric_limits<T>::quiet_NaN()), boost::math::rounding_error);
278 BOOST_MATH_CHECK_THROW(ltrunc(std::numeric_limits<T>::quiet_NaN()), boost::math::rounding_error);
279 #ifdef BOOST_HAS_LONG_LONG
280 BOOST_MATH_CHECK_THROW(lltrunc(std::numeric_limits<T>::quiet_NaN()), boost::math::rounding_error);
281 #endif
282 }
283 if(std::numeric_limits<T>::digits >= std::numeric_limits<int>::digits)
284 {
285 BOOST_MATH_CHECK_THROW(itrunc(static_cast<T>((std::numeric_limits<int>::max)()) + 1), boost::math::rounding_error);
286 BOOST_MATH_CHECK_THROW(itrunc(static_cast<T>((std::numeric_limits<int>::min)()) - 1), boost::math::rounding_error);
287 }
288 if(std::numeric_limits<T>::digits >= std::numeric_limits<long>::digits)
289 {
290 BOOST_MATH_CHECK_THROW(ltrunc(static_cast<T>((std::numeric_limits<long>::max)()) + 1), boost::math::rounding_error);
291 BOOST_MATH_CHECK_THROW(ltrunc(static_cast<T>((std::numeric_limits<long>::min)()) - 1), boost::math::rounding_error);
292 }
293 #ifndef BOOST_NO_LONG_LONG
294 if(std::numeric_limits<T>::digits >= std::numeric_limits<boost::long_long_type>::digits)
295 {
296 BOOST_MATH_CHECK_THROW(lltrunc(static_cast<T>((std::numeric_limits<boost::long_long_type>::max)()) + 1), boost::math::rounding_error);
297 BOOST_MATH_CHECK_THROW(lltrunc(static_cast<T>((std::numeric_limits<boost::long_long_type>::min)()) - 1), boost::math::rounding_error);
298 }
299 #endif
300 if(std::numeric_limits<T>::digits >= std::numeric_limits<int>::digits)
301 {
302 BOOST_MATH_CHECK_THROW(iround(static_cast<T>((std::numeric_limits<int>::max)()) + 1), boost::math::rounding_error);
303 BOOST_MATH_CHECK_THROW(iround(static_cast<T>((std::numeric_limits<int>::min)()) - 1), boost::math::rounding_error);
304 }
305 if(std::numeric_limits<T>::digits >= std::numeric_limits<long>::digits)
306 {
307 BOOST_MATH_CHECK_THROW(lround(static_cast<T>((std::numeric_limits<long>::max)()) + 1), boost::math::rounding_error);
308 BOOST_MATH_CHECK_THROW(lround(static_cast<T>((std::numeric_limits<long>::min)()) - 1), boost::math::rounding_error);
309 }
310 #ifndef BOOST_NO_LONG_LONG
311 if(std::numeric_limits<T>::digits >= std::numeric_limits<boost::long_long_type>::digits)
312 {
313 BOOST_MATH_CHECK_THROW(llround(static_cast<T>((std::numeric_limits<boost::long_long_type>::max)()) + 1), boost::math::rounding_error);
314 BOOST_MATH_CHECK_THROW(llround(static_cast<T>((std::numeric_limits<boost::long_long_type>::min)()) - 1), boost::math::rounding_error);
315 }
316 #endif
317 //
318 // try non-throwing error handlers:
319 //
320 boost::math::policies::policy<boost::math::policies::rounding_error<boost::math::policies::ignore_error> > pol;
321
322 if(std::numeric_limits<T>::digits >= std::numeric_limits<int>::digits)
323 {
324 BOOST_CHECK_EQUAL(iround((std::numeric_limits<int>::max)() + T(1.0), pol), (std::numeric_limits<int>::max)());
325 BOOST_CHECK_EQUAL(iround((std::numeric_limits<int>::min)() - T(1.0), pol), (std::numeric_limits<int>::min)());
326 BOOST_CHECK_EQUAL(itrunc((std::numeric_limits<int>::max)() + T(1.0), pol), (std::numeric_limits<int>::max)());
327 BOOST_CHECK_EQUAL(itrunc((std::numeric_limits<int>::min)() - T(1.0), pol), (std::numeric_limits<int>::min)());
328 }
329 if(std::numeric_limits<T>::digits >= std::numeric_limits<long>::digits)
330 {
331 BOOST_CHECK_EQUAL(lround((std::numeric_limits<long>::max)() + T(1.0), pol), (std::numeric_limits<long>::max)());
332 BOOST_CHECK_EQUAL(lround((std::numeric_limits<long>::min)() - T(1.0), pol), (std::numeric_limits<long>::min)());
333 BOOST_CHECK_EQUAL(ltrunc((std::numeric_limits<long>::max)() + T(1.0), pol), (std::numeric_limits<long>::max)());
334 BOOST_CHECK_EQUAL(ltrunc((std::numeric_limits<long>::min)() - T(1.0), pol), (std::numeric_limits<long>::min)());
335 }
336 #ifndef BOOST_NO_LONG_LONG
337 if(std::numeric_limits<T>::digits >= std::numeric_limits<long long>::digits)
338 {
339 BOOST_CHECK_EQUAL(llround((std::numeric_limits<long long>::max)() + T(1.0), pol), (std::numeric_limits<long long>::max)());
340 BOOST_CHECK_EQUAL(llround((std::numeric_limits<long long>::min)() - T(1.0), pol), (std::numeric_limits<long long>::min)());
341 BOOST_CHECK_EQUAL(lltrunc((std::numeric_limits<long long>::max)() + T(1.0), pol), (std::numeric_limits<long long>::max)());
342 BOOST_CHECK_EQUAL(lltrunc((std::numeric_limits<long long>::min)() - T(1.0), pol), (std::numeric_limits<long long>::min)());
343 }
344 #endif
345 // Again with bigger value:
346 T big = 1e20f;
347 BOOST_CHECK_EQUAL(iround(big, pol), (std::numeric_limits<int>::max)());
348 BOOST_CHECK_EQUAL(lround(big, pol), (std::numeric_limits<long>::max)());
349 BOOST_CHECK_EQUAL(iround(-big, pol), (std::numeric_limits<int>::min)());
350 BOOST_CHECK_EQUAL(lround(-big, pol), (std::numeric_limits<long>::min)());
351 BOOST_CHECK_EQUAL(itrunc(big, pol), (std::numeric_limits<int>::max)());
352 BOOST_CHECK_EQUAL(ltrunc(big, pol), (std::numeric_limits<long>::max)());
353 BOOST_CHECK_EQUAL(itrunc(-big, pol), (std::numeric_limits<int>::min)());
354 BOOST_CHECK_EQUAL(ltrunc(-big, pol), (std::numeric_limits<long>::min)());
355 #ifndef BOOST_NO_LONG_LONG
356 BOOST_CHECK_EQUAL(llround(big, pol), (std::numeric_limits<long long>::max)());
357 BOOST_CHECK_EQUAL(llround(-big, pol), (std::numeric_limits<long long>::min)());
358 BOOST_CHECK_EQUAL(lltrunc(big, pol), (std::numeric_limits<long long>::max)());
359 BOOST_CHECK_EQUAL(lltrunc(-big, pol), (std::numeric_limits<long long>::min)());
360 #endif
361
362 //
363 // Special cases that we know can go bad:
364 //
365 T half = 0.5f;
366 half = boost::math::float_prior(half);
367 test_round_number(half);
368 half = -0.5f;
369 half = boost::math::float_next(half);
370 test_round_number(half);
371
372 if(std::numeric_limits<T>::is_specialized)
373 {
374 //
375 // Odd and even integer values:
376 //
377 T val;
378 for(int i = 2; i < std::numeric_limits<T>::max_exponent; ++i)
379 {
380 val = ldexp(T(1), i);
381 test_round_number(val);
382 ++val;
383 test_round_number(val);
384 val = -val;
385 test_round_number(val);
386 ++val;
387 test_round_number(val);
388 }
389 }
390 }
391
BOOST_AUTO_TEST_CASE(test_main)392 BOOST_AUTO_TEST_CASE( test_main )
393 {
394 test_round(0.1F, "float");
395 test_round(0.1, "double");
396 #ifndef BOOST_MATH_NO_LONG_DOUBLE_MATH_FUNCTIONS
397 test_round(0.1L, "long double");
398 test_round(boost::math::concepts::real_concept(0.1), "real_concept");
399 #else
400 std::cout << "<note>The long double tests have been disabled on this platform "
401 "either because the long double overloads of the usual math functions are "
402 "not available at all, or because they are too inaccurate for these tests "
403 "to pass.</note>" << std::endl;
404 #endif
405
406 // test rounding of direct predecessor/successor of 0.5/-0.5 for float and double
407 test_round_number(-0.4999999701976776123046875f);
408 BOOST_CHECK_EQUAL(boost::math::round(-0.4999999701976776123046875f), 0.0f);
409
410 test_round_number(0.4999999701976776123046875f);
411 BOOST_CHECK_EQUAL(boost::math::round(0.4999999701976776123046875f), 0.0f);
412
413 BOOST_CHECK_EQUAL(boost::math::round(-0.499999999999999944488848768742172978818416595458984375), 0.0);
414 test_round_number(-0.499999999999999944488848768742172978818416595458984375);
415
416 BOOST_CHECK_EQUAL(boost::math::round(0.499999999999999944488848768742172978818416595458984375), 0.0);
417 test_round_number(0.499999999999999944488848768742172978818416595458984375);
418
419 // test rounding of integer numbers on the edge of the float/double mantissa width
420 BOOST_CHECK_EQUAL(boost::math::round(-16777215.0f), -16777215.0f);
421 test_round_number(-16777215.0f);
422
423 BOOST_CHECK_EQUAL(boost::math::round(-16777213.0f), -16777213.0f);
424 test_round_number(-16777213.0f);
425
426 BOOST_CHECK_EQUAL(boost::math::round(-8388611.0f), -8388611.0f);
427 test_round_number(-8388611.0f);
428
429 BOOST_CHECK_EQUAL(boost::math::round(-8388609.0f), -8388609.0f);
430 test_round_number(-8388609.0f);
431
432 BOOST_CHECK_EQUAL(boost::math::round(8388609.0f), 8388609.0f);
433 test_round_number(8388609.0f);
434
435 BOOST_CHECK_EQUAL(boost::math::round(8388611.0f), 8388611.0f);
436 test_round_number(8388611.0f);
437
438 BOOST_CHECK_EQUAL(boost::math::round(16777213.0f), 16777213.0f);
439 test_round_number(16777213.0f);
440
441 BOOST_CHECK_EQUAL(boost::math::round(16777215.0f), 16777215.0f);
442 test_round_number(16777215.0f);
443
444 BOOST_CHECK_EQUAL(boost::math::round(-9007199254740993.0), -9007199254740993.0);
445 test_round_number(-9007199254740993.0);
446
447 BOOST_CHECK_EQUAL(boost::math::round(-9007199254740991.0), -9007199254740991.0);
448 test_round_number(-9007199254740991.0);
449
450 BOOST_CHECK_EQUAL(boost::math::round(-4503599627370499.0), -4503599627370499.0);
451 test_round_number(-4503599627370499.0);
452
453 BOOST_CHECK_EQUAL(boost::math::round(-4503599627370497.0), -4503599627370497.0);
454 test_round_number(-4503599627370497.0);
455
456 BOOST_CHECK_EQUAL(boost::math::round(4503599627370497.0), 4503599627370497.0);
457 test_round_number(4503599627370497.0);
458
459 BOOST_CHECK_EQUAL(boost::math::round(4503599627370499.0), 4503599627370499.0);
460 test_round_number(4503599627370499.0);
461
462 BOOST_CHECK_EQUAL(boost::math::round(9007199254740991.0), 9007199254740991.0);
463 test_round_number(9007199254740991.0);
464
465 BOOST_CHECK_EQUAL(boost::math::round(9007199254740993.0), 9007199254740993.0);
466 test_round_number(9007199254740993.0);
467 }
468