Skip to content

Commit

Permalink
Adding computation of complete elliptic integrals into amrex::Math (A…
Browse files Browse the repository at this point in the history
…MReX-Codes#4151)

… so that it can be executed on accelerator devices.

## Summary
Currently the parser recognizes comp_ellint_1 and comp_ellint_2, but
only executes on CPU when compiled with gcc. An implementation of the
calculations have been added to amrex::Math to compute these quantities
using the method described here:
https://dlmf.nist.gov/19.8

It is a quadratically converging iterative method that is compatible
with device execution.

Gauss's Arithmetic-Geometric Mean
 
  • Loading branch information
clarkse authored Sep 18, 2024
1 parent acc35e2 commit 23a7f34
Show file tree
Hide file tree
Showing 3 changed files with 77 additions and 14 deletions.
4 changes: 2 additions & 2 deletions Docs/sphinx_documentation/source/Basics.rst
Original file line number Diff line number Diff line change
Expand Up @@ -595,8 +595,8 @@ numbers can be computed with ``min`` and ``max``, respectively. It supports
the Heaviside step function, ``heaviside(x1,x2)`` that gives ``0``, ``x2``,
``1``, for ``x1 < 0``, ``x1 = 0`` and ``x1 > 0``, respectively.
It supports the Bessel function of the first kind of order ``n``
``jn(n,x)``. Complete elliptic integrals of the first and second kind, ``comp_ellint_1`` and ``comp_ellint_2``,
are supported only for gcc and CPUs.
``jn(n,x)``. Complete elliptic integrals of the first and second kind, ``comp_ellint_1(k)`` and ``comp_ellint_2(k)``,
are supported.
There is ``if(a,b,c)`` that gives ``b`` or ``c`` depending on the value of
``a``. A number of comparison operators are supported, including ``<``,
``>``, ``==``, ``!=``, ``<=``, and ``>=``. The Boolean results from
Expand Down
67 changes: 67 additions & 0 deletions Src/Base/AMReX_Math.H
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@
#include <AMReX_GpuQualifiers.H>
#include <AMReX_Extension.H>
#include <AMReX_INT.H>
#include <AMReX_REAL.H>
#include <cmath>
#include <cstdlib>
#include <type_traits>
Expand Down Expand Up @@ -225,6 +226,72 @@ std::uint64_t umulhi (std::uint64_t a, std::uint64_t b)
}
#endif

template <typename T>
AMREX_GPU_HOST_DEVICE AMREX_FORCE_INLINE
T comp_ellint_1 (T k)
{
// Computing K based on DLMF
// https://dlmf.nist.gov/19.8
T tol = std::numeric_limits<T>::epsilon();

T a0 = 1.0;
T g0 = std::sqrt(1.0 - k*k);
T a = a0;
T g = g0;

// Find Arithmetic Geometric mean
while(std::abs(a0 - g0) > tol) {
a = 0.5*(a0 + g0);
g = std::sqrt(a0 * g0);

a0 = a;
g0 = g;
}

return 0.5*pi<T>()/a;
}

template <typename T>
AMREX_GPU_HOST_DEVICE AMREX_FORCE_INLINE
T comp_ellint_2 (T k)
{
// Computing E based on DLMF
// https://dlmf.nist.gov/19.8
T Kcomp = amrex::Math::comp_ellint_1<T>(k);
T tol = std::numeric_limits<T>::epsilon();

// Step Zero
T a0 = 1.0;
T g0 = std::sqrt(1.0 - k*k);
T cn = std::sqrt(a0*a0 - g0*g0);

// Step 1
int n = 1;
T a = 0.5 * (a0 + g0);
T g = std::sqrt(a0*g0);
cn = 0.25*cn*cn/a;

T sum_val = a*a;
a0 = a;
g0 = g;

while(std::abs(cn*cn) > tol) {
// Compute coefficients for this iteration
a = 0.5 * (a0 + g0);
g = std::sqrt(a0*g0);
cn = 0.25*cn*cn/a;

n++;
sum_val -= std::pow(2,n-1)*cn*cn;

// Save a and g for next iteration
a0 = a;
g0 = g;
}

return Kcomp*sum_val;
}

/***************************************************************************************************
* Copyright (c) 2017 - 2024 NVIDIA CORPORATION & AFFILIATES. All rights reserved.
* SPDX-License-Identifier: BSD-3-Clause
Expand Down
20 changes: 8 additions & 12 deletions Src/Base/Parser/AMReX_Parser_Y.H
Original file line number Diff line number Diff line change
Expand Up @@ -349,28 +349,24 @@ AMREX_GPU_HOST_DEVICE AMREX_NO_INLINE
T parser_math_atanh (T a) { return std::atanh(a); }

template <typename T>
AMREX_GPU_HOST_DEVICE AMREX_NO_INLINE
T parser_math_comp_ellint_1 (T a)
AMREX_GPU_HOST_DEVICE AMREX_FORCE_INLINE
T parser_math_comp_ellint_1 (T k)
{
#if defined(__GNUC__) && !defined(__clang__) && !defined(__CUDA_ARCH__) && !defined(__NVCOMPILER)
return std::comp_ellint_1(a);
return std::comp_ellint_1(k);
#else
amrex::ignore_unused(a);
AMREX_ALWAYS_ASSERT_WITH_MESSAGE(false, "parser: comp_ellint_1 only supported with gcc and cpu");
return 0.0;
return amrex::Math::comp_ellint_1<T>(k);
#endif
}

template <typename T>
AMREX_GPU_HOST_DEVICE AMREX_NO_INLINE
T parser_math_comp_ellint_2 (T a)
AMREX_GPU_HOST_DEVICE AMREX_FORCE_INLINE
T parser_math_comp_ellint_2 (T k)
{
#if defined(__GNUC__) && !defined(__clang__) && !defined(__CUDA_ARCH__) && !defined(__NVCOMPILER)
return std::comp_ellint_2(a);
return std::comp_ellint_2(k);
#else
amrex::ignore_unused(a);
AMREX_ALWAYS_ASSERT_WITH_MESSAGE(false, "parser: comp_ellint_2 only supported with gcc and cpu");
return 0.0;
return amrex::Math::comp_ellint_2<T>(k);
#endif
}

Expand Down

0 comments on commit 23a7f34

Please sign in to comment.