From 74f56c45526359fbba3d9fcec80f03522730f26b Mon Sep 17 00:00:00 2001 From: wpbonelli Date: Fri, 22 Dec 2023 10:21:02 -0500 Subject: [PATCH] feat(MathUtil): add root-finding functions (#1521) * move from / for use by PRT ternary solution for DISV grids * modernize (remove gotos, use explicit local variables) * unit test briefly --- autotest/TestMathUtil.f90 | 71 ++++++++- src/Utilities/MathUtil.f90 | 305 ++++++++++++++++++++++++++++++++++++- 2 files changed, 373 insertions(+), 3 deletions(-) diff --git a/autotest/TestMathUtil.f90 b/autotest/TestMathUtil.f90 index b28e9050d8e..b6e3a358a8c 100644 --- a/autotest/TestMathUtil.f90 +++ b/autotest/TestMathUtil.f90 @@ -3,7 +3,8 @@ module TestMathUtil use ConstantsModule, only: DNODATA, DZERO use testdrive, only: check, error_type, new_unittest, test_failed, & to_string, unittest_type - use MathUtilModule, only: is_close, mod_offset + use MathUtilModule, only: is_close, mod_offset, & + zeroch, zerotest, zeroin implicit none private public :: collect_mathutil @@ -17,7 +18,13 @@ subroutine collect_mathutil(testsuite) new_unittest("is_close_symmetric_near_0", & test_is_close_symmetric_near_0), & new_unittest("mod_offset", & - test_mod_offset) & + test_mod_offset), & + new_unittest("zeroch", & + test_zeroch), & + new_unittest("zeroin", & + test_zeroin), & + new_unittest("zerotest", & + test_zerotest) & ] end subroutine collect_mathutil @@ -164,4 +171,64 @@ subroutine test_is_close_symmetric_near_0(error) end subroutine test_is_close_symmetric_near_0 + pure function sine(bet) result(s) + real(DP), intent(in) :: bet + real(DP) :: s + s = sin(bet) + end function sine + + subroutine test_zeroch(error) + type(error_type), allocatable, intent(out) :: error + real(DP), parameter :: pi = 4 * atan(1.0_DP) + real(DP) :: z + + z = zeroch(-1.0_DP, 1.0_DP, sine, 0.001_DP) + call check(error, is_close(z, 0.0_DP, atol=1d-6), & + 'expected 0, got: '//to_string(z)) + + z = zeroch(-4.0_DP, -1.0_DP, sine, 0.001_DP) + call check(error, is_close(z, -pi, atol=1d-6), & + 'expected -pi, got: '//to_string(z)) + + z = zeroch(1.0_DP, 4.0_DP, sine, 0.001_DP) + call check(error, is_close(z, pi, atol=1d-6), & + 'expected pi, got: '//to_string(z)) + end subroutine test_zeroch + + subroutine test_zeroin(error) + type(error_type), allocatable, intent(out) :: error + real(DP), parameter :: pi = 4 * atan(1.0_DP) + real(DP) :: z + + z = zeroin(-1.0_DP, 1.0_DP, sine, 0.001_DP) + call check(error, is_close(z, 0.0_DP, atol=1d-6), & + 'expected 0, got: '//to_string(z)) + + z = zeroin(-4.0_DP, -1.0_DP, sine, 0.001_DP) + call check(error, is_close(z, -pi, atol=1d-6), & + 'expected -pi, got: '//to_string(z)) + + z = zeroin(1.0_DP, 4.0_DP, sine, 0.001_DP) + call check(error, is_close(z, pi, atol=1d-6), & + 'expected pi, got: '//to_string(z)) + end subroutine test_zeroin + + subroutine test_zerotest(error) + type(error_type), allocatable, intent(out) :: error + real(DP), parameter :: pi = 4 * atan(1.0_DP) + real(DP) :: z + + z = zerotest(-1.0_DP, 1.0_DP, sine, 0.001_DP) + call check(error, is_close(z, 0.0_DP, atol=1d-6), & + 'expected 0, got: '//to_string(z)) + + z = zerotest(-4.0_DP, -1.0_DP, sine, 0.001_DP) + call check(error, is_close(z, -pi, atol=1d-6), & + 'expected -pi, got: '//to_string(z)) + + z = zerotest(1.0_DP, 4.0_DP, sine, 0.001_DP) + call check(error, is_close(z, pi, atol=1d-6), & + 'expected pi, got: '//to_string(z)) + end subroutine test_zerotest + end module TestMathUtil diff --git a/src/Utilities/MathUtil.f90 b/src/Utilities/MathUtil.f90 index ad6af25ddd0..0becc593609 100644 --- a/src/Utilities/MathUtil.f90 +++ b/src/Utilities/MathUtil.f90 @@ -7,7 +7,7 @@ module MathUtilModule implicit none private - public :: mod_offset, is_close + public :: mod_offset, is_close, zeroch, zeroin, zerotest interface mod_offset module procedure :: mod_offset_int, mod_offset_dbl @@ -112,4 +112,307 @@ pure function mod_offset_dbl(a, n, d) result(mo) mo = a - n * floor((a - ld) / n) end function mod_offset_dbl + !> @brief Compute zeros on an interval using Chadrupatla's method + !! + !! A zero of the function f{x} is computed in the interval (x0, x1) + !! given tolerance epsa using Chandrupatla's method. FORTRAN code based + !! generally on pseudocode in Scherer, POJ (2013) "Computational Physics: + !! Simulation of Classical and Quantum Systems," 2nd ed., Springer, New York. + !! + !< + function zeroch(x0, x1, f, epsa) result(z) + implicit double precision(a - h, o - z) + ! -- local + real(DP) :: epsm + real(DP) :: a, b, c, t + real(DP) :: aminusb, cminusb + real(DP) :: fa, fb, fc, fm, ft + real(DP) :: faminusfb, fcminusfb + real(DP) :: phi, philo, phihi + real(DP) :: racb, rcab, rbca + real(DP) :: tol, tl, tlc + real(DP) :: xm, xt + + epsm = epsilon(x0) + b = x0 + a = x1 + c = x1 + aminusb = a - b + fb = f(b) + fa = f(a) + fc = f(c) + t = 5d-1 + + do while (.true.) + ! xt = a + t*(b - a) + xt = a - t * aminusb + ft = f(xt) + if (sign(ft, fa) == ft) then + c = a + fc = fa + a = xt + fa = ft + else + c = b + b = a + a = xt + fc = fb + fb = fa + fa = ft + end if + aminusb = a - b + cminusb = c - b + faminusfb = fa - fb + fcminusfb = fc - fb + xm = a + fm = fa + if (dabs(fb) < dabs(fa)) then + xm = b + fm = fb + end if + tol = 2d0 * epsm * dabs(xm) + epsa + ! tl = tol/dabs(b - c) + tl = tol / dabs(cminusb) + if ((tl > 5d-1) .or. (fm == 0d0)) then + z = xm + return + end if + ! xi = (a - b)/(c - b) + xi = aminusb / cminusb + ! phi = (fa - fb)/(fc - fb) + phi = faminusfb / fcminusfb + philo = 1d0 - dsqrt(1d0 - xi) + phihi = dsqrt(xi) + if ((phi > philo) .and. (phi < phihi)) then + ! rab = fa/(fb - fa) + ! rab = -fa/faminusfb + ! rcb = fc/(fb - fc) + ! rcb = -fc/fcminusfb + ! rac = fa/(fc - fa) + ! rbc = fb/(fc - fb) + ! rbc = fb/fcminusfb + ! t = rab*rcb + rac*rbc*(c - a)/(b - a) + ! t = rab*rcb - rac*rbc*(c - a)/aminusb + racb = fa / fcminusfb + rcab = fc / faminusfb + rbca = fb / (fc - fa) + t = racb * (rcab - rbca * (c - a) / aminusb) + if (t < tl) then + t = tl + else + tlc = 1d0 - tl + if (t > tlc) then + t = tlc + end if + end if + else + t = 5d-1 + end if + ! if (t < tl) t = tl + ! if (t > 1d0 - tl) t = 1d0 - tl + end do + end function + + !> @brief Compute a zero of the function f(x) in the interval (x0, x1). + !! + !! A zero of the function f(x) is computed in the interval ax,bx. + !! + !! Input: + !! + !! ax left endpoint of initial interval + !! bx right endpoint of initial interval + !! f function subprogram which evaluates f(x) for any x in + !! the interval ax,bx + !! tol desired length of the interval of uncertainty of the + !! final result (.ge.0.) + !! + !! Output: + !! + !! zeroin abscissa approximating a zero of f in the interval ax,bx + !! + !! it is assumed that f(ax) and f(bx) have opposite signs + !! this is checked, and an error message is printed if this is not + !! satisfied. zeroin returns a zero x in the given interval + !! ax,bx to within a tolerance 4*macheps*abs(x)+tol, where macheps is + !! the relative machine precision defined as the smallest representable + !! number such that 1.+macheps .gt. 1. + !! this function subprogram is a slightly modified translation of + !! the algol 60 procedure zero given in richard brent, algorithms for + !! minimization without derivatives, prentice-hall, inc. (1973). + !< + function zeroin(ax, bx, f, tol) result(z) + implicit double precision(a - h, o - z) + ! -- local + real(DP) :: eps + real(DP) :: a, b, c, d, e, s, p, q + real(DP) :: fa, fb, fc + real(DP) :: tol1 + logical(LGP) :: rs + + eps = epsilon(ax) + tol1 = eps + 1.0d0 + + a = ax + b = bx + fa = f(a) + fb = f(b) + + ! check that f(ax) and f(bx) have different signs + if (.not. ((fa .eq. 0.0d0 .or. fb .eq. 0.0d0) .or. & + (fa * (fb / dabs(fb)) .le. 0.0d0))) & + call pstop(1, 'f(ax) and f(bx) do not have different signs,') + + rs = .true. ! var reset + do while (.true.) + if (rs) then + c = a + fc = fa + d = b - a + e = d + end if + + if (.not. (dabs(fc) .ge. dabs(fb))) then + a = b + b = c + c = a + fa = fb + fb = fc + fc = fa + end if + + tol1 = 2.0d0 * eps * dabs(b) + 0.5d0 * tol + xm = 0.5d0 * (c - b) + if ((dabs(xm) .le. tol1) .or. (fb .eq. 0.0d0)) then + z = b + return + end if + + ! see if a bisection is forced + if ((dabs(e) .ge. tol1) .and. (dabs(fa) .gt. dabs(fb))) then + s = fb / fa + if (a .ne. c) then + ! inverse quadratic interpolation + q = fa / fc + r = fb / fc + p = s * (2.0d0 * xm * q * (q - r) - (b - a) * (r - 1.0d0)) + q = (q - 1.0d0) * (r - 1.0d0) * (s - 1.0d0) + else + ! linear interpolation + p = 2.0d0 * xm * s + q = 1.0d0 - s + end if + + if (p .le. 0.0d0) then + p = -p + else + q = -q + end if + s = e + e = d + if (((2.0d0 * p) .ge. (3.0d0 * xm * q - dabs(tol1 * q))) .or. & + (p .ge. dabs(0.5d0 * s * q))) then + d = xm + e = d + else + d = p / q + end if + else + d = xm + e = d + end if + + a = b + fa = fb + + if (dabs(d) .le. tol1) then + if (xm .le. 0.0d0) then + b = b - tol1 + else + b = b + tol1 + end if + else + b = b + d + end if + + fb = f(b) + rs = (fb * (fc / dabs(fc))) .gt. 0.0d0 + end do + end function zeroin + + !> @brief Compute a zero of the function f(x) in the interval (x0, x1) + function zerotest(x0, x1, f, epsa) result(z) + implicit double precision(a - h, o - z) + ! -- local + real(DP) :: epsm + real(DP) :: ema, emb + real(DP) :: f0 + real(DP) :: tol + real(DP) :: xa, xb, xl + real(DP) :: ya, yb, yl + logical(LGP) :: retainedxa, retainedxb + + epsm = epsilon(x0) + f0 = f(x0) + if (f0 .eq. 0d0) then + z = x0 + return + else if (f0 .lt. 0d0) then + ya = x0 + yb = x1 + xa = f0 + xb = f(yb) + else + ya = x1 + yb = x0 + xa = f(ya) + xb = f0 + end if + ema = 1d0 + emb = 1d0 + retainedxa = .false. + retainedxb = .false. + + do while (.true.) + ! yl = ya - xa*(yb - ya)/(xb - xa) + yl = (ya * xb * emb - yb * xa * ema) / (xb * emb - xa * ema) + tol = 4d0 * epsm * dabs(yl) + epsa + if (dabs(yb - ya) .le. tol) then + z = yl + return + else + xl = f(yl) + if (xl .eq. 0d0) then + z = yl + return + else if (xl .gt. 0d0) then + if (retainedxa) then + ! ema = 1d0 - xl/xb + ! if (ema <= 0d0) ema = 5d-1 + ema = 5d-1 ! kluge illinois + else + ema = 1d0 + end if + emb = 1d0 + yb = yl + xb = xl + retainedxa = .true. + retainedxb = .false. + else + if (retainedxb) then + ! emb = 1d0 - xl/xa + ! if (emb <= 0d0) emb = 5d-1 + emb = 5d-1 ! kluge illinois + else + emb = 1d0 + end if + ema = 1d0 + ya = yl + xa = xl + retainedxa = .false. + retainedxb = .true. + end if + end if + end do + end function + end module MathUtilModule