| 
									
										
										
										
											2020-03-30 10:34:21 +02:00
										 |  |  | #include <catch2/catch_approx.hpp>
 | 
					
						
							|  |  |  | #include <catch2/internal/catch_enforce.hpp>
 | 
					
						
							| 
									
										
										
										
											2017-07-19 10:01:06 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-01 07:30:11 +01:00
										 |  |  | #include <cmath>
 | 
					
						
							| 
									
										
										
										
											2017-07-19 10:01:06 +02:00
										 |  |  | #include <limits>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-01 07:30:11 +01:00
										 |  |  | namespace { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Performs equivalent check of std::fabs(lhs - rhs) <= margin
 | 
					
						
							|  |  |  | // But without the subtraction to allow for INFINITY in comparison
 | 
					
						
							|  |  |  | bool marginComparison(double lhs, double rhs, double margin) { | 
					
						
							|  |  |  |     return (lhs + margin >= rhs) && (rhs + margin >= lhs); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-19 10:01:06 +02:00
										 |  |  | namespace Catch { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Approx::Approx ( double value ) | 
					
						
							|  |  |  |     :   m_epsilon( std::numeric_limits<float>::epsilon()*100 ), | 
					
						
							|  |  |  |         m_margin( 0.0 ), | 
					
						
							| 
									
										
										
										
											2017-10-26 09:19:57 +02:00
										 |  |  |         m_scale( 0.0 ), | 
					
						
							| 
									
										
										
										
											2017-07-19 10:01:06 +02:00
										 |  |  |         m_value( value ) | 
					
						
							|  |  |  |     {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Approx Approx::custom() { | 
					
						
							|  |  |  |         return Approx( 0 ); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-21 15:42:40 +02:00
										 |  |  |     Approx Approx::operator-() const { | 
					
						
							|  |  |  |         auto temp(*this); | 
					
						
							|  |  |  |         temp.m_value = -temp.m_value; | 
					
						
							|  |  |  |         return temp; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-19 10:01:06 +02:00
										 |  |  |     std::string Approx::toString() const { | 
					
						
							| 
									
										
										
										
											2017-11-07 18:01:10 +00:00
										 |  |  |         ReusableStringStream rss; | 
					
						
							|  |  |  |         rss << "Approx( " << ::Catch::Detail::stringify( m_value ) << " )"; | 
					
						
							|  |  |  |         return rss.str(); | 
					
						
							| 
									
										
										
										
											2017-07-19 10:01:06 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-01 07:30:11 +01:00
										 |  |  |     bool Approx::equalityComparisonImpl(const double other) const { | 
					
						
							|  |  |  |         // First try with fixed margin, then compute margin based on epsilon, scale and Approx's value
 | 
					
						
							|  |  |  |         // Thanks to Richard Harris for his help refining the scaled margin value
 | 
					
						
							| 
									
										
										
										
											2019-10-04 13:23:14 +02:00
										 |  |  |         return marginComparison(m_value, other, m_margin) | 
					
						
							|  |  |  |             || marginComparison(m_value, other, m_epsilon * (m_scale + std::fabs(std::isinf(m_value)? 0 : m_value))); | 
					
						
							| 
									
										
										
										
											2017-11-01 07:30:11 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-13 08:54:23 -07:00
										 |  |  |     void Approx::setMargin(double newMargin) { | 
					
						
							|  |  |  |         CATCH_ENFORCE(newMargin >= 0, | 
					
						
							|  |  |  |             "Invalid Approx::margin: " << newMargin << '.' | 
					
						
							| 
									
										
										
										
											2018-09-03 10:03:47 +02:00
										 |  |  |             << " Approx::Margin has to be non-negative."); | 
					
						
							| 
									
										
										
										
											2019-01-13 08:54:23 -07:00
										 |  |  |         m_margin = newMargin; | 
					
						
							| 
									
										
										
										
											2018-09-03 10:15:49 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-13 08:54:23 -07:00
										 |  |  |     void Approx::setEpsilon(double newEpsilon) { | 
					
						
							|  |  |  |         CATCH_ENFORCE(newEpsilon >= 0 && newEpsilon <= 1.0, | 
					
						
							|  |  |  |             "Invalid Approx::epsilon: " << newEpsilon << '.' | 
					
						
							| 
									
										
										
										
											2018-09-03 10:03:47 +02:00
										 |  |  |             << " Approx::epsilon has to be in [0, 1]"); | 
					
						
							| 
									
										
										
										
											2019-01-13 08:54:23 -07:00
										 |  |  |         m_epsilon = newEpsilon; | 
					
						
							| 
									
										
										
										
											2018-09-03 10:15:49 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-21 15:42:40 +02:00
										 |  |  | namespace literals { | 
					
						
							| 
									
										
										
										
											2020-02-03 15:07:59 +01:00
										 |  |  |     Approx operator "" _a(long double val) { | 
					
						
							|  |  |  |         return Approx(val); | 
					
						
							| 
									
										
										
										
											2018-05-21 15:42:40 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-02-03 15:07:59 +01:00
										 |  |  |     Approx operator "" _a(unsigned long long val) { | 
					
						
							|  |  |  |         return Approx(val); | 
					
						
							| 
									
										
										
										
											2018-05-21 15:42:40 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | } // end namespace literals
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-03 15:07:59 +01:00
										 |  |  | std::string StringMaker<Catch::Approx>::convert(Catch::Approx const& value) { | 
					
						
							| 
									
										
										
										
											2017-07-19 10:01:06 +02:00
										 |  |  |     return value.toString(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // end namespace Catch
 |