GM6000 Digital Heater Controller Branch: main
SDX-1330
real.h
Go to the documentation of this file.
1#ifndef Cpl_Math_real_h_
2#define Cpl_Math_real_h_
3/*-----------------------------------------------------------------------------
4* This file is part of the Colony.Core Project. The Colony.Core Project is an
5* open source project with a BSD type of licensing agreement. See the license
6* agreement (license.txt) in the top/ directory or on the Internet at
7* http://integerfox.com/colony.core/license.txt
8*
9* Copyright (c) 2014-2022 John T. Taylor
10*
11* Redistributions of the source code must retain the above copyright notice.
12*----------------------------------------------------------------------------*/
13/** @file
14
15 This file contains a collection of methods comparing, manipulating, etc.
16 floating point numbers (i.e. floats and doubles).
17
18*/
19
20#include "colony_config.h"
21#include <cmath>
22#include <cfloat>
23
24#undef abs
25
26/** This symbols provides the default Epsilon value when testing for 'almost
27 equal' between to float numbers. Note: This is a GLOBAL setting.
28 */
29#ifndef CPL_MATH_REAL_FLOAT_EPSILON
30#define CPL_MATH_REAL_FLOAT_EPSILON (FLT_EPSILON)
31#endif
32
33 /** This symbols provides the default Epsilon value when testing for 'almost
34 equal' between to double numbers. Note: This is a GLOBAL setting.
35 */
36#ifndef CPL_MATH_REAL_DOUBLE_EPSILON
37#define CPL_MATH_REAL_DOUBLE_EPSILON (DBL_EPSILON)
38#endif
39
40
41
42///
43namespace Cpl {
44///
45namespace Math {
46
47
48/** This template function implements a 'almost equals' comparison function for
49 floating points numbers. See the following link for why an 'almost equals'
50 function is necessary:
51
52 http://www.cygnus-software.com/papers/comparingfloats/comparingfloats.htm
53 */
54template <class T>
55bool almostEquals( T a, T b, T epsilon )
56{
57 // Filter out NaN (always return false since by the IEEE standard NaN != NaN)
58 if ( std::isnan( a ) || std::isnan( b ) )
59 {
60 return false;
61 }
62
63 // Try the 'quick' case (also handles infinites)
64 if ( a == b )
65 {
66 return true;
67 }
68
69
70 // Handle the case of one of the arguments being zero or close to zero
71 T diff = std::abs( a - b );
72 if ( ( a == (T) ( 0.0 ) || b == (T) ( 0.0 ) ) && diff < epsilon )
73 {
74 return diff < ( epsilon * epsilon );
75 }
76
77 // Do a relative error check
78 T absA = std::abs( a );
79 T absB = std::abs( b );
80 return ( diff / ( absA + absB ) ) < epsilon;
81}
82
83
84/** This method is short hand for almostEquals<float> AND provides a default epsilon
85 */
86inline bool areFloatsEqual( float a, float b, float epsilon = CPL_MATH_REAL_FLOAT_EPSILON )
87{
88 return almostEquals<float>( a, b, epsilon );
89}
90
91
92/** This method is short hand for almostEquals<double> AND provides a default epsilon
93 */
94inline bool areDoublesEqual( double a, double b, double epsilon = CPL_MATH_REAL_DOUBLE_EPSILON )
95{
96 return almostEquals<double>( a, b, epsilon );
97}
98
99
100}; // end namespaces
101};
102#endif // end header latch
bool almostEquals(T a, T b, T epsilon)
This template function implements a 'almost equals' comparison function for floating points numbers.
Definition real.h:55
bool areFloatsEqual(float a, float b, float epsilon=CPL_MATH_REAL_FLOAT_EPSILON)
This method is short hand for almostEquals<float> AND provides a default epsilon.
Definition real.h:86
bool areDoublesEqual(double a, double b, double epsilon=CPL_MATH_REAL_DOUBLE_EPSILON)
This method is short hand for almostEquals<double> AND provides a default epsilon.
Definition real.h:94
The 'Cpl' namespace is the root name space for the Colony.
Definition Api16.h:20
#define CPL_MATH_REAL_FLOAT_EPSILON
This symbols provides the default Epsilon value when testing for 'almost equal' between to float numb...
Definition real.h:30
#define CPL_MATH_REAL_DOUBLE_EPSILON
This symbols provides the default Epsilon value when testing for 'almost equal' between to double num...
Definition real.h:37