//@HEADER // ************************************************************************ // // Kokkos v. 4.0 // Copyright (2022) National Technology & Engineering // Solutions of Sandia, LLC (NTESS). // // Under the terms of Contract DE-NA0003525 with NTESS, // the U.S. Government retains certain rights in this software. // // Part of Kokkos, under the Apache License v2.0 with LLVM Exceptions. // See https://kokkos.org/LICENSE for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //@HEADER #include #include #include namespace Test { // Test construction and assignment template struct TestComplexConstruction { Kokkos::View *, ExecSpace> d_results; typename Kokkos::View *, ExecSpace>::HostMirror h_results; void testit() { d_results = Kokkos::View *, ExecSpace>( "TestComplexConstruction", 10); h_results = Kokkos::create_mirror_view(d_results); Kokkos::parallel_for(Kokkos::RangePolicy(0, 1), *this); Kokkos::fence(); Kokkos::deep_copy(h_results, d_results); ASSERT_FLOAT_EQ(h_results(0).real(), 1.5); ASSERT_FLOAT_EQ(h_results(0).imag(), 2.5); ASSERT_FLOAT_EQ(h_results(1).real(), 1.5); ASSERT_FLOAT_EQ(h_results(1).imag(), 2.5); ASSERT_FLOAT_EQ(h_results(2).real(), 0.0); ASSERT_FLOAT_EQ(h_results(2).imag(), 0.0); ASSERT_FLOAT_EQ(h_results(3).real(), 3.5); ASSERT_FLOAT_EQ(h_results(3).imag(), 0.0); ASSERT_FLOAT_EQ(h_results(4).real(), 4.5); ASSERT_FLOAT_EQ(h_results(4).imag(), 5.5); ASSERT_FLOAT_EQ(h_results(5).real(), 1.5); ASSERT_FLOAT_EQ(h_results(5).imag(), 2.5); ASSERT_FLOAT_EQ(h_results(6).real(), 4.5); ASSERT_FLOAT_EQ(h_results(6).imag(), 5.5); ASSERT_FLOAT_EQ(h_results(7).real(), 7.5); ASSERT_FLOAT_EQ(h_results(7).imag(), 0.0); ASSERT_FLOAT_EQ(h_results(8).real(), double(8)); ASSERT_FLOAT_EQ(h_results(8).imag(), 0.0); // Copy construction conversion between // Kokkos::complex and std::complex doesn't compile Kokkos::complex a(1.5, 2.5), b(3.25, 5.25), r_kk; std::complex sa(a), sb(3.25, 5.25), r; r = a; r_kk = a; ASSERT_FLOAT_EQ(r.real(), r_kk.real()); ASSERT_FLOAT_EQ(r.imag(), r_kk.imag()); r = sb * a; r_kk = b * a; ASSERT_FLOAT_EQ(r.real(), r_kk.real()); ASSERT_FLOAT_EQ(r.imag(), r_kk.imag()); r = sa; r_kk = a; ASSERT_FLOAT_EQ(r.real(), r_kk.real()); ASSERT_FLOAT_EQ(r.imag(), r_kk.imag()); } KOKKOS_INLINE_FUNCTION void operator()(const int & /*i*/) const { Kokkos::complex a(1.5, 2.5); d_results(0) = a; Kokkos::complex b(a); d_results(1) = b; Kokkos::complex c = Kokkos::complex(); d_results(2) = c; Kokkos::complex d(3.5); d_results(3) = d; Kokkos::complex a_v(4.5, 5.5); d_results(4) = a_v; Kokkos::complex b_v(a); d_results(5) = b_v; Kokkos::complex e(a_v); d_results(6) = e; d_results(7) = double(7.5); d_results(8) = int(8); } }; TEST(TEST_CATEGORY, complex_construction) { TestComplexConstruction test; test.testit(); } // Test Math FUnction template struct TestComplexBasicMath { Kokkos::View *, ExecSpace> d_results; typename Kokkos::View *, ExecSpace>::HostMirror h_results; void testit() { d_results = Kokkos::View *, ExecSpace>( "TestComplexBasicMath", 24); h_results = Kokkos::create_mirror_view(d_results); Kokkos::parallel_for(Kokkos::RangePolicy(0, 1), *this); Kokkos::fence(); Kokkos::deep_copy(h_results, d_results); std::complex a(1.5, 2.5); std::complex b(3.25, 5.75); std::complex d(1.0, 2.0); double c = 9.3; int e = 2; std::complex r; r = a + b; ASSERT_FLOAT_EQ(h_results(0).real(), r.real()); ASSERT_FLOAT_EQ(h_results(0).imag(), r.imag()); r = a - b; ASSERT_FLOAT_EQ(h_results(1).real(), r.real()); ASSERT_FLOAT_EQ(h_results(1).imag(), r.imag()); r = a * b; ASSERT_FLOAT_EQ(h_results(2).real(), r.real()); ASSERT_FLOAT_EQ(h_results(2).imag(), r.imag()); r = a / b; #ifndef KOKKOS_WORKAROUND_OPENMPTARGET_CLANG ASSERT_FLOAT_EQ(h_results(3).real(), r.real()); ASSERT_FLOAT_EQ(h_results(3).imag(), r.imag()); #endif r = d + a; ASSERT_FLOAT_EQ(h_results(4).real(), r.real()); ASSERT_FLOAT_EQ(h_results(4).imag(), r.imag()); r = d - a; ASSERT_FLOAT_EQ(h_results(5).real(), r.real()); ASSERT_FLOAT_EQ(h_results(5).imag(), r.imag()); r = d * a; ASSERT_FLOAT_EQ(h_results(6).real(), r.real()); ASSERT_FLOAT_EQ(h_results(6).imag(), r.imag()); r = d / a; ASSERT_FLOAT_EQ(h_results(7).real(), r.real()); ASSERT_FLOAT_EQ(h_results(7).imag(), r.imag()); r = a + c; ASSERT_FLOAT_EQ(h_results(8).real(), r.real()); ASSERT_FLOAT_EQ(h_results(8).imag(), r.imag()); r = a - c; ASSERT_FLOAT_EQ(h_results(9).real(), r.real()); ASSERT_FLOAT_EQ(h_results(9).imag(), r.imag()); r = a * c; ASSERT_FLOAT_EQ(h_results(10).real(), r.real()); ASSERT_FLOAT_EQ(h_results(10).imag(), r.imag()); r = a / c; ASSERT_FLOAT_EQ(h_results(11).real(), r.real()); ASSERT_FLOAT_EQ(h_results(11).imag(), r.imag()); r = d + c; ASSERT_FLOAT_EQ(h_results(12).real(), r.real()); ASSERT_FLOAT_EQ(h_results(12).imag(), r.imag()); r = d - c; ASSERT_FLOAT_EQ(h_results(13).real(), r.real()); ASSERT_FLOAT_EQ(h_results(13).imag(), r.imag()); r = d * c; ASSERT_FLOAT_EQ(h_results(14).real(), r.real()); ASSERT_FLOAT_EQ(h_results(14).imag(), r.imag()); r = d / c; ASSERT_FLOAT_EQ(h_results(15).real(), r.real()); ASSERT_FLOAT_EQ(h_results(15).imag(), r.imag()); r = c + a; ASSERT_FLOAT_EQ(h_results(16).real(), r.real()); ASSERT_FLOAT_EQ(h_results(16).imag(), r.imag()); r = c - a; ASSERT_FLOAT_EQ(h_results(17).real(), r.real()); ASSERT_FLOAT_EQ(h_results(17).imag(), r.imag()); r = c * a; ASSERT_FLOAT_EQ(h_results(18).real(), r.real()); ASSERT_FLOAT_EQ(h_results(18).imag(), r.imag()); r = c / a; #ifndef KOKKOS_WORKAROUND_OPENMPTARGET_CLANG ASSERT_FLOAT_EQ(h_results(19).real(), r.real()); ASSERT_FLOAT_EQ(h_results(19).imag(), r.imag()); #endif r = a; /* r = a+e; */ ASSERT_FLOAT_EQ(h_results(20).real(), r.real() + e); ASSERT_FLOAT_EQ(h_results(20).imag(), r.imag()); /* r = a-e; */ ASSERT_FLOAT_EQ(h_results(21).real(), r.real() - e); ASSERT_FLOAT_EQ(h_results(21).imag(), r.imag()); /* r = a*e; */ ASSERT_FLOAT_EQ(h_results(22).real(), r.real() * e); ASSERT_FLOAT_EQ(h_results(22).imag(), r.imag() * e); /* r = a/e; */ ASSERT_FLOAT_EQ(h_results(23).real(), r.real() / 2); ASSERT_FLOAT_EQ(h_results(23).imag(), r.imag() / e); } KOKKOS_INLINE_FUNCTION void operator()(const int & /*i*/) const { Kokkos::complex a(1.5, 2.5); Kokkos::complex b(3.25, 5.75); // Basic math complex / complex d_results(0) = a + b; d_results(1) = a - b; d_results(2) = a * b; d_results(3) = a / b; d_results(4).real(1.0); d_results(4).imag(2.0); d_results(4) += a; d_results(5) = Kokkos::complex(1.0, 2.0); d_results(5) -= a; d_results(6) = Kokkos::complex(1.0, 2.0); d_results(6) *= a; d_results(7) = Kokkos::complex(1.0, 2.0); d_results(7) /= a; // Basic math complex / scalar double c = 9.3; d_results(8) = a + c; d_results(9) = a - c; d_results(10) = a * c; d_results(11) = a / c; d_results(12).real(1.0); d_results(12).imag(2.0); d_results(12) += c; d_results(13) = Kokkos::complex(1.0, 2.0); d_results(13) -= c; d_results(14) = Kokkos::complex(1.0, 2.0); d_results(14) *= c; d_results(15) = Kokkos::complex(1.0, 2.0); d_results(15) /= c; // Basic math scalar / complex d_results(16) = c + a; d_results(17) = c - a; d_results(18) = c * a; d_results(19) = c / a; int e = 2; d_results(20) = a + e; d_results(21) = a - e; d_results(22) = a * e; d_results(23) = a / e; } }; TEST(TEST_CATEGORY, complex_basic_math) { TestComplexBasicMath test; test.testit(); } template struct TestComplexSpecialFunctions { Kokkos::View *, ExecSpace> d_results; typename Kokkos::View *, ExecSpace>::HostMirror h_results; void testit() { d_results = Kokkos::View *, ExecSpace>( "TestComplexSpecialFunctions", 20); h_results = Kokkos::create_mirror_view(d_results); Kokkos::parallel_for(Kokkos::RangePolicy(0, 1), *this); Kokkos::fence(); Kokkos::deep_copy(h_results, d_results); std::complex a(1.5, 2.5); double c = 9.3; std::complex r; r = a; ASSERT_FLOAT_EQ(h_results(0).real(), r.real()); ASSERT_FLOAT_EQ(h_results(0).imag(), r.imag()); r = std::sqrt(a); ASSERT_FLOAT_EQ(h_results(1).real(), r.real()); ASSERT_FLOAT_EQ(h_results(1).imag(), r.imag()); r = std::pow(a, c); ASSERT_FLOAT_EQ(h_results(2).real(), r.real()); ASSERT_FLOAT_EQ(h_results(2).imag(), r.imag()); r = std::abs(a); ASSERT_FLOAT_EQ(h_results(3).real(), r.real()); ASSERT_FLOAT_EQ(h_results(3).imag(), r.imag()); r = std::exp(a); ASSERT_FLOAT_EQ(h_results(4).real(), r.real()); ASSERT_FLOAT_EQ(h_results(4).imag(), r.imag()); r = Kokkos::exp(a); ASSERT_FLOAT_EQ(h_results(4).real(), r.real()); ASSERT_FLOAT_EQ(h_results(4).imag(), r.imag()); #ifndef KOKKOS_WORKAROUND_OPENMPTARGET_CLANG r = std::log(a); ASSERT_FLOAT_EQ(h_results(5).real(), r.real()); ASSERT_FLOAT_EQ(h_results(5).imag(), r.imag()); r = std::sin(a); ASSERT_FLOAT_EQ(h_results(6).real(), r.real()); ASSERT_FLOAT_EQ(h_results(6).imag(), r.imag()); r = std::cos(a); ASSERT_FLOAT_EQ(h_results(7).real(), r.real()); ASSERT_FLOAT_EQ(h_results(7).imag(), r.imag()); r = std::tan(a); ASSERT_FLOAT_EQ(h_results(8).real(), r.real()); ASSERT_FLOAT_EQ(h_results(8).imag(), r.imag()); r = std::sinh(a); ASSERT_FLOAT_EQ(h_results(9).real(), r.real()); ASSERT_FLOAT_EQ(h_results(9).imag(), r.imag()); r = std::cosh(a); ASSERT_FLOAT_EQ(h_results(10).real(), r.real()); ASSERT_FLOAT_EQ(h_results(10).imag(), r.imag()); r = std::tanh(a); ASSERT_FLOAT_EQ(h_results(11).real(), r.real()); ASSERT_FLOAT_EQ(h_results(11).imag(), r.imag()); r = std::asinh(a); ASSERT_FLOAT_EQ(h_results(12).real(), r.real()); ASSERT_FLOAT_EQ(h_results(12).imag(), r.imag()); r = std::acosh(a); ASSERT_FLOAT_EQ(h_results(13).real(), r.real()); ASSERT_FLOAT_EQ(h_results(13).imag(), r.imag()); // atanh // Work around a bug in gcc 5.3.1 where the compiler cannot compute atanh r = {0.163481616851666003, 1.27679502502111284}; ASSERT_FLOAT_EQ(h_results(14).real(), r.real()); ASSERT_FLOAT_EQ(h_results(14).imag(), r.imag()); r = std::asin(a); ASSERT_FLOAT_EQ(h_results(15).real(), r.real()); ASSERT_FLOAT_EQ(h_results(15).imag(), r.imag()); r = std::acos(a); ASSERT_FLOAT_EQ(h_results(16).real(), r.real()); ASSERT_FLOAT_EQ(h_results(16).imag(), r.imag()); // atan // Work around a bug in gcc 5.3.1 where the compiler cannot compute atan r = {1.380543138238714, 0.2925178131625636}; ASSERT_FLOAT_EQ(h_results(17).real(), r.real()); ASSERT_FLOAT_EQ(h_results(17).imag(), r.imag()); // log10 r = std::log10(a); ASSERT_FLOAT_EQ(h_results(18).real(), r.real()); ASSERT_FLOAT_EQ(h_results(18).imag(), r.imag()); #endif } KOKKOS_INLINE_FUNCTION void operator()(const int & /*i*/) const { Kokkos::complex a(1.5, 2.5); Kokkos::complex b(3.25, 5.75); double c = 9.3; d_results(0) = Kokkos::complex(Kokkos::real(a), Kokkos::imag(a)); d_results(1) = Kokkos::sqrt(a); d_results(2) = Kokkos::pow(a, c); d_results(3) = Kokkos::abs(a); d_results(4) = Kokkos::exp(a); d_results(5) = Kokkos::log(a); d_results(6) = Kokkos::sin(a); d_results(7) = Kokkos::cos(a); d_results(8) = Kokkos::tan(a); d_results(9) = Kokkos::sinh(a); d_results(10) = Kokkos::cosh(a); d_results(11) = Kokkos::tanh(a); d_results(12) = Kokkos::asinh(a); d_results(13) = Kokkos::acosh(a); d_results(14) = Kokkos::atanh(a); d_results(15) = Kokkos::asin(a); d_results(16) = Kokkos::acos(a); d_results(17) = Kokkos::atan(a); d_results(18) = Kokkos::log10(a); } }; void testComplexIO() { Kokkos::complex z = {3.14, 1.41}; std::stringstream ss; ss << z; ASSERT_EQ(ss.str(), "(3.14,1.41)"); ss.str("1 (2) (3,4)"); ss.clear(); ss >> z; ASSERT_EQ(z, (Kokkos::complex{1, 0})); ss >> z; ASSERT_EQ(z, (Kokkos::complex{2, 0})); ss >> z; ASSERT_EQ(z, (Kokkos::complex{3, 4})); } TEST(TEST_CATEGORY, complex_special_funtions) { TestComplexSpecialFunctions test; test.testit(); } TEST(TEST_CATEGORY, complex_io) { testComplexIO(); } TEST(TEST_CATEGORY, complex_trivially_copyable) { // Kokkos::complex is trivially copyable when RealType is // trivially copyable using RealType = double; // clang claims compatibility with gcc 4.2.1 but all versions tested know // about std::is_trivially_copyable. ASSERT_TRUE(std::is_trivially_copyable>::value || !std::is_trivially_copyable::value); } template struct TestBugPowAndLogComplex { Kokkos::View *, ExecSpace> d_pow; Kokkos::View *, ExecSpace> d_log; TestBugPowAndLogComplex() : d_pow("pow", 2), d_log("log", 2) { test(); } void test() { Kokkos::parallel_for(Kokkos::RangePolicy(0, 1), *this); auto h_pow = Kokkos::create_mirror_view_and_copy(Kokkos::HostSpace(), d_pow); ASSERT_FLOAT_EQ(h_pow(0).real(), 18); ASSERT_FLOAT_EQ(h_pow(0).imag(), 26); ASSERT_FLOAT_EQ(h_pow(1).real(), -18); ASSERT_FLOAT_EQ(h_pow(1).imag(), 26); auto h_log = Kokkos::create_mirror_view_and_copy(Kokkos::HostSpace(), d_log); ASSERT_FLOAT_EQ(h_log(0).real(), 1.151292546497023); ASSERT_FLOAT_EQ(h_log(0).imag(), 0.3217505543966422); ASSERT_FLOAT_EQ(h_log(1).real(), 1.151292546497023); ASSERT_FLOAT_EQ(h_log(1).imag(), 2.819842099193151); } KOKKOS_FUNCTION void operator()(int) const { d_pow(0) = Kokkos::pow(Kokkos::complex(+3., 1.), 3.); d_pow(1) = Kokkos::pow(Kokkos::complex(-3., 1.), 3.); d_log(0) = Kokkos::log(Kokkos::complex(+3., 1.)); d_log(1) = Kokkos::log(Kokkos::complex(-3., 1.)); } }; TEST(TEST_CATEGORY, complex_issue_3865) { TestBugPowAndLogComplex(); } #ifdef KOKKOS_ENABLE_OPENMPTARGET // FIXME_OPENMPTARGET TEST(TEST_CATEGORY, complex_issue_3867) { ASSERT_EQ(Kokkos::pow(Kokkos::complex(2., 1.), 3.), Kokkos::pow(Kokkos::complex(2., 1.), 3)); ASSERT_EQ( Kokkos::pow(Kokkos::complex(2., 1.), 3.), Kokkos::pow(Kokkos::complex(2., 1.), Kokkos::complex(3))); auto x = Kokkos::pow(Kokkos::complex(2, 1), Kokkos::complex(-3, 4)); auto y = Kokkos::complex( std::pow(std::complex(2, 1), std::complex(-3, 4))); ASSERT_FLOAT_EQ(x.real(), y.real()); ASSERT_FLOAT_EQ(x.imag(), y.imag()); #define CHECK_POW_COMPLEX_PROMOTION(ARGTYPE1, ARGTYPE2, RETURNTYPE) \ static_assert( \ std::is_same(), \ std::declval()))>::value, \ ""); \ static_assert( \ std::is_same(), \ std::declval()))>::value, \ ""); CHECK_POW_COMPLEX_PROMOTION(Kokkos::complex, long double, Kokkos::complex); CHECK_POW_COMPLEX_PROMOTION(Kokkos::complex, double, Kokkos::complex); CHECK_POW_COMPLEX_PROMOTION(Kokkos::complex, float, Kokkos::complex); CHECK_POW_COMPLEX_PROMOTION(Kokkos::complex, int, Kokkos::complex); CHECK_POW_COMPLEX_PROMOTION(Kokkos::complex, long double, Kokkos::complex); CHECK_POW_COMPLEX_PROMOTION(Kokkos::complex, double, Kokkos::complex); CHECK_POW_COMPLEX_PROMOTION(Kokkos::complex, float, Kokkos::complex); CHECK_POW_COMPLEX_PROMOTION(Kokkos::complex, int, Kokkos::complex); CHECK_POW_COMPLEX_PROMOTION(Kokkos::complex, long double, Kokkos::complex); CHECK_POW_COMPLEX_PROMOTION(Kokkos::complex, double, Kokkos::complex); CHECK_POW_COMPLEX_PROMOTION(Kokkos::complex, float, Kokkos::complex); CHECK_POW_COMPLEX_PROMOTION(Kokkos::complex, int, Kokkos::complex); #undef CHECK_POW_COMPLEX_PROMOTION } #endif TEST(TEST_CATEGORY, complex_operations_arithmetic_types_overloads) { static_assert(Kokkos::real(1) == 1.); static_assert(Kokkos::real(2.f) == 2.f); static_assert(Kokkos::real(3.) == 3.); static_assert(Kokkos::real(4.l) == 4.l); static_assert((std::is_same::value)); static_assert((std::is_same::value)); static_assert((std::is_same::value)); static_assert( (std::is_same::value)); static_assert(Kokkos::imag(1) == 0.); static_assert(Kokkos::imag(2.f) == 0.f); static_assert(Kokkos::imag(3.) == 0.); static_assert(Kokkos::imag(4.l) == 0.l); static_assert((std::is_same::value)); static_assert((std::is_same::value)); static_assert((std::is_same::value)); static_assert( (std::is_same::value)); // FIXME in principle could be checked at compile time too ASSERT_EQ(Kokkos::conj(1), Kokkos::complex(1)); ASSERT_EQ(Kokkos::conj(2.f), Kokkos::complex(2.f)); ASSERT_EQ(Kokkos::conj(3.), Kokkos::complex(3.)); // long double has size 12 but Kokkos::complex requires 2*sizeof(T) to be a // power of two. #ifndef KOKKOS_IMPL_32BIT ASSERT_EQ(Kokkos::conj(4.l), Kokkos::complex(4.l)); static_assert(( std::is_same>::value)); #endif static_assert((std::is_same>::value)); static_assert((std::is_same>::value)); static_assert((std::is_same>::value)); } } // namespace Test