From a965c46025718a4e0a51c6e1279a856d21226467 Mon Sep 17 00:00:00 2001 From: Martin Diehl Date: Wed, 6 Mar 2019 15:57:42 +0100 Subject: [PATCH] improved functions for floating point comparison - less stric tolerance for comparison to zero - better readable - avoid "merge" on optional arguments (not safe) - "equal" and "notEqual" are now symmetric (assignment of <= and < is arbitrary) --- src/prec.f90 | 104 ++++++++++++++++++++++++++++++++++++--------------- 1 file changed, 74 insertions(+), 30 deletions(-) diff --git a/src/prec.f90 b/src/prec.f90 index ab47f8e34..36f92b7bd 100644 --- a/src/prec.f90 +++ b/src/prec.f90 @@ -79,6 +79,8 @@ module prec integer(pInt), pointer, dimension(:,:) :: p end type + real(pReal), private, parameter :: DBL_EPSILON = 2.220446049250313E-16_pReal !< minimum positive number such that 1.0 + DBL_EPSILON /= 1.0. + real(pReal), private, parameter :: DBL_MIN = 2.2250738585072014E-308_pReal !< smallest normalized floating point number public :: & prec_init, & @@ -126,12 +128,19 @@ end subroutine prec_init !-------------------------------------------------------------------------------------------------- logical elemental pure function dEq(a,b,tol) - implicit none - real(pReal), intent(in) :: a,b - real(pReal), intent(in), optional :: tol - real(pReal), parameter :: eps = 2.220446049250313E-16 ! DBL_EPSILON in C + implicit none + real(pReal), intent(in) :: a,b + real(pReal), intent(in), optional :: tol + real(pReal) :: eps + + if (present(tol)) then + eps = tol + else + eps = DBL_EPSILON * maxval(abs([a,b])) + endif + + dEq = merge(.True.,.False.,abs(a-b) < eps) - dEq = merge(.True.,.False.,abs(a-b) <= merge(tol,eps,present(tol))*maxval(abs([a,b]))) end function dEq @@ -143,12 +152,19 @@ end function dEq !-------------------------------------------------------------------------------------------------- logical elemental pure function dNeq(a,b,tol) - implicit none - real(pReal), intent(in) :: a,b - real(pReal), intent(in), optional :: tol - real(pReal), parameter :: eps = 2.220446049250313E-16 ! DBL_EPSILON in C + implicit none + real(pReal), intent(in) :: a,b + real(pReal), intent(in), optional :: tol + real(pReal) :: eps + + if (present(tol)) then + eps = tol + else + eps = DBL_EPSILON * maxval(abs([a,b])) + endif + + dNeq = merge(.False.,.True.,abs(a-b) <= eps) - dNeq = merge(.False.,.True.,abs(a-b) <= merge(tol,eps,present(tol))*maxval(abs([a,b]))) end function dNeq @@ -160,12 +176,19 @@ end function dNeq !-------------------------------------------------------------------------------------------------- logical elemental pure function dEq0(a,tol) - implicit none - real(pReal), intent(in) :: a - real(pReal), intent(in), optional :: tol - real(pReal), parameter :: eps = 2.2250738585072014E-308 ! smallest non-denormalized number + implicit none + real(pReal), intent(in) :: a + real(pReal), intent(in), optional :: tol + real(pReal) :: eps + + if (present(tol)) then + eps = tol + else + eps = DBL_MIN * 10.0_pReal + endif + + dEq0 = merge(.True.,.False.,abs(a) < eps) - dEq0 = merge(.True.,.False.,abs(a) <= merge(tol,eps,present(tol))) end function dEq0 @@ -177,12 +200,19 @@ end function dEq0 !-------------------------------------------------------------------------------------------------- logical elemental pure function dNeq0(a,tol) - implicit none - real(pReal), intent(in) :: a - real(pReal), intent(in), optional :: tol - real(pReal), parameter :: eps = 2.2250738585072014E-308 ! smallest non-denormalized number + implicit none + real(pReal), intent(in) :: a + real(pReal), intent(in), optional :: tol + real(pReal) :: eps + + if (present(tol)) then + eps = tol + else + eps = DBL_MIN * 10.0_pReal + endif + + dNeq0 = merge(.False.,.True.,abs(a) <= eps) - dNeq0 = merge(.False.,.True.,abs(a) <= merge(tol,eps,present(tol))) end function dNeq0 @@ -195,12 +225,19 @@ end function dNeq0 !-------------------------------------------------------------------------------------------------- logical elemental pure function cEq(a,b,tol) - implicit none - complex(pReal), intent(in) :: a,b - real(pReal), intent(in), optional :: tol - real(pReal), parameter :: eps = 2.220446049250313E-16 ! DBL_EPSILON in C + implicit none + complex(pReal), intent(in) :: a,b + real(pReal), intent(in), optional :: tol + real(pReal) :: eps + + if (present(tol)) then + eps = tol + else + eps = DBL_EPSILON * maxval(abs([a,b])) + endif + + cEq = merge(.True.,.False.,abs(a-b) < eps) - cEq = merge(.True.,.False.,abs(a-b) <= merge(tol,eps,present(tol))*maxval(abs([a,b]))) end function cEq @@ -213,12 +250,19 @@ end function cEq !-------------------------------------------------------------------------------------------------- logical elemental pure function cNeq(a,b,tol) - implicit none - complex(pReal), intent(in) :: a,b - real(pReal), intent(in), optional :: tol - real(pReal), parameter :: eps = 2.220446049250313E-16 ! DBL_EPSILON in C + implicit none + complex(pReal), intent(in) :: a,b + real(pReal), intent(in), optional :: tol + real(pReal) :: eps + + if (present(tol)) then + eps = tol + else + eps = DBL_EPSILON * maxval(abs([a,b])) + endif + + cNeq = merge(.False.,.True.,abs(a-b) <= eps) - cNeq = merge(.False.,.True.,abs(a-b) <= merge(tol,eps,present(tol))*maxval(abs([a,b]))) end function cNeq end module prec