Files
pytorch/c10/test/util/complex_math_test_common.h
2023-03-24 17:24:50 +00:00

668 lines
21 KiB
C++

// Warning: this file is included twice in
// aten/src/ATen/test/cuda_complex_math_test.cu
#include <c10/util/complex.h>
#include <gtest/gtest.h>
#ifndef PI
#define PI 3.141592653589793238463
#endif
#ifndef tol
#define tol 1e-6
#endif
// Exponential functions
C10_DEFINE_TEST(TestExponential, IPi) {
// exp(i*pi) = -1
{
c10::complex<float> e_i_pi = std::exp(c10::complex<float>(0, float(PI)));
C10_ASSERT_NEAR(e_i_pi.real(), -1, tol);
C10_ASSERT_NEAR(e_i_pi.imag(), 0, tol);
}
{
c10::complex<float> e_i_pi = ::exp(c10::complex<float>(0, float(PI)));
C10_ASSERT_NEAR(e_i_pi.real(), -1, tol);
C10_ASSERT_NEAR(e_i_pi.imag(), 0, tol);
}
{
c10::complex<double> e_i_pi = std::exp(c10::complex<double>(0, PI));
C10_ASSERT_NEAR(e_i_pi.real(), -1, tol);
C10_ASSERT_NEAR(e_i_pi.imag(), 0, tol);
}
{
c10::complex<double> e_i_pi = ::exp(c10::complex<double>(0, PI));
C10_ASSERT_NEAR(e_i_pi.real(), -1, tol);
C10_ASSERT_NEAR(e_i_pi.imag(), 0, tol);
}
}
C10_DEFINE_TEST(TestExponential, EulerFormula) {
// exp(ix) = cos(x) + i * sin(x)
{
c10::complex<float> x(0.1, 1.2);
c10::complex<float> e = std::exp(x);
float expected_real = std::exp(x.real()) * std::cos(x.imag());
float expected_imag = std::exp(x.real()) * std::sin(x.imag());
C10_ASSERT_NEAR(e.real(), expected_real, tol);
C10_ASSERT_NEAR(e.imag(), expected_imag, tol);
}
{
c10::complex<float> x(0.1, 1.2);
c10::complex<float> e = ::exp(x);
float expected_real = ::exp(x.real()) * ::cos(x.imag());
float expected_imag = ::exp(x.real()) * ::sin(x.imag());
C10_ASSERT_NEAR(e.real(), expected_real, tol);
C10_ASSERT_NEAR(e.imag(), expected_imag, tol);
}
{
c10::complex<double> x(0.1, 1.2);
c10::complex<double> e = std::exp(x);
float expected_real = std::exp(x.real()) * std::cos(x.imag());
float expected_imag = std::exp(x.real()) * std::sin(x.imag());
C10_ASSERT_NEAR(e.real(), expected_real, tol);
C10_ASSERT_NEAR(e.imag(), expected_imag, tol);
}
{
c10::complex<double> x(0.1, 1.2);
c10::complex<double> e = ::exp(x);
float expected_real = ::exp(x.real()) * ::cos(x.imag());
float expected_imag = ::exp(x.real()) * ::sin(x.imag());
C10_ASSERT_NEAR(e.real(), expected_real, tol);
C10_ASSERT_NEAR(e.imag(), expected_imag, tol);
}
}
C10_DEFINE_TEST(TestExpm1, Normal) {
// expm1(x) = exp(x) - 1
{
c10::complex<float> x(0.1, 1.2);
c10::complex<float> l1 = std::expm1(x);
c10::complex<float> l2 = std::exp(x) - 1.0f;
C10_ASSERT_NEAR(l1.real(), l2.real(), tol);
C10_ASSERT_NEAR(l1.imag(), l2.imag(), tol);
}
{
c10::complex<double> x(0.1, 1.2);
c10::complex<double> l1 = std::expm1(x);
c10::complex<double> l2 = std::exp(x) - 1.0;
C10_ASSERT_NEAR(l1.real(), l2.real(), tol);
C10_ASSERT_NEAR(l1.imag(), l2.imag(), tol);
}
}
C10_DEFINE_TEST(TestExpm1, Small) {
// expm1(x) = exp(x) - 1
// expm1(x) provides greater precision than exp(x) - 1 for small values of x
{
c10::complex<float> x(1e-30, 1e-30);
c10::complex<float> l1 = std::expm1(x);
C10_ASSERT_NEAR(l1.real(), 1e-30, tol);
C10_ASSERT_NEAR(l1.imag(), 1e-30, tol);
}
{
c10::complex<double> x(1e-100, 1e-100);
c10::complex<double> l1 = std::expm1(x);
C10_ASSERT_NEAR(l1.real(), 1e-30, tol);
C10_ASSERT_NEAR(l1.imag(), 1e-30, tol);
}
}
C10_DEFINE_TEST(TestLog, Definition) {
// log(x) = log(r) + i*theta
{
c10::complex<float> x(1.2, 3.4);
c10::complex<float> l = std::log(x);
float expected_real = std::log(std::abs(x));
float expected_imag = std::arg(x);
C10_ASSERT_NEAR(l.real(), expected_real, tol);
C10_ASSERT_NEAR(l.imag(), expected_imag, tol);
}
{
c10::complex<float> x(1.2, 3.4);
c10::complex<float> l = ::log(x);
float expected_real = ::log(std::abs(x));
float expected_imag = std::arg(x);
C10_ASSERT_NEAR(l.real(), expected_real, tol);
C10_ASSERT_NEAR(l.imag(), expected_imag, tol);
}
{
c10::complex<double> x(1.2, 3.4);
c10::complex<double> l = std::log(x);
float expected_real = std::log(std::abs(x));
float expected_imag = std::arg(x);
C10_ASSERT_NEAR(l.real(), expected_real, tol);
C10_ASSERT_NEAR(l.imag(), expected_imag, tol);
}
{
c10::complex<double> x(1.2, 3.4);
c10::complex<double> l = ::log(x);
float expected_real = ::log(std::abs(x));
float expected_imag = std::arg(x);
C10_ASSERT_NEAR(l.real(), expected_real, tol);
C10_ASSERT_NEAR(l.imag(), expected_imag, tol);
}
}
C10_DEFINE_TEST(TestLog10, Rev) {
// log10(10^x) = x
{
c10::complex<float> x(0.1, 1.2);
c10::complex<float> l = std::log10(std::pow(float(10), x));
C10_ASSERT_NEAR(l.real(), float(0.1), tol);
C10_ASSERT_NEAR(l.imag(), float(1.2), tol);
}
{
c10::complex<float> x(0.1, 1.2);
c10::complex<float> l = ::log10(::pow(float(10), x));
C10_ASSERT_NEAR(l.real(), float(0.1), tol);
C10_ASSERT_NEAR(l.imag(), float(1.2), tol);
}
{
c10::complex<double> x(0.1, 1.2);
c10::complex<double> l = std::log10(std::pow(double(10), x));
C10_ASSERT_NEAR(l.real(), double(0.1), tol);
C10_ASSERT_NEAR(l.imag(), double(1.2), tol);
}
{
c10::complex<double> x(0.1, 1.2);
c10::complex<double> l = ::log10(::pow(double(10), x));
C10_ASSERT_NEAR(l.real(), double(0.1), tol);
C10_ASSERT_NEAR(l.imag(), double(1.2), tol);
}
}
C10_DEFINE_TEST(TestLog2, Rev) {
// log2(2^x) = x
{
c10::complex<float> x(0.1, 1.2);
c10::complex<float> l = std::log2(std::pow(float(2), x));
C10_ASSERT_NEAR(l.real(), float(0.1), tol);
C10_ASSERT_NEAR(l.imag(), float(1.2), tol);
}
{
c10::complex<float> x(0.1, 1.2);
c10::complex<float> l = ::log2(std::pow(float(2), x));
C10_ASSERT_NEAR(l.real(), float(0.1), tol);
C10_ASSERT_NEAR(l.imag(), float(1.2), tol);
}
{
c10::complex<double> x(0.1, 1.2);
c10::complex<double> l = std::log2(std::pow(double(2), x));
C10_ASSERT_NEAR(l.real(), double(0.1), tol);
C10_ASSERT_NEAR(l.imag(), double(1.2), tol);
}
{
c10::complex<double> x(0.1, 1.2);
c10::complex<double> l = ::log2(std::pow(double(2), x));
C10_ASSERT_NEAR(l.real(), double(0.1), tol);
C10_ASSERT_NEAR(l.imag(), double(1.2), tol);
}
}
C10_DEFINE_TEST(TestLog1p, Normal) {
// log1p(x) = log(1 + x)
{
c10::complex<float> x(0.1, 1.2);
c10::complex<float> l1 = std::log1p(x);
c10::complex<float> l2 = std::log(1.0f + x);
C10_ASSERT_NEAR(l1.real(), l2.real(), tol);
C10_ASSERT_NEAR(l1.imag(), l2.imag(), tol);
}
{
c10::complex<double> x(0.1, 1.2);
c10::complex<double> l1 = std::log1p(x);
c10::complex<double> l2 = std::log(1.0 + x);
C10_ASSERT_NEAR(l1.real(), l2.real(), tol);
C10_ASSERT_NEAR(l1.imag(), l2.imag(), tol);
}
}
C10_DEFINE_TEST(TestLog1p, Small) {
// log(1 + x) ~ x for |x| << 1
{
c10::complex<float> x(1e-9, 2e-9);
c10::complex<float> l = std::log1p(x);
C10_ASSERT_NEAR(l.real() / x.real(), 1, tol);
C10_ASSERT_NEAR(l.imag() / x.imag(), 1, tol);
}
{
c10::complex<double> x(1e-100, 2e-100);
c10::complex<double> l = std::log1p(x);
C10_ASSERT_NEAR(l.real() / x.real(), 1, tol);
C10_ASSERT_NEAR(l.imag() / x.imag(), 1, tol);
}
}
C10_DEFINE_TEST(TestLog1p, Extreme) {
// log(1 + x) ~ x for |x| << 1 and in the brink of overflow / underflow
{
c10::complex<float> x(-1, 1e-30);
c10::complex<float> l = std::log1p(x);
C10_ASSERT_NEAR(l.real(), -69.07755278982137, tol);
C10_ASSERT_NEAR(l.imag(), 1.5707963267948966, tol);
}
{
c10::complex<float> x(-1, 1e30);
c10::complex<float> l = std::log1p(x);
C10_ASSERT_NEAR(l.real(), 69.07755278982137, tol);
C10_ASSERT_NEAR(l.imag(), 1.5707963267948966, tol);
}
{
c10::complex<float> x(1e30, 1);
c10::complex<float> l = std::log1p(x);
C10_ASSERT_NEAR(l.real(), 69.07755278982137, tol);
C10_ASSERT_NEAR(l.imag(), 1e-30, tol);
}
{
c10::complex<float> x(1e-30, 1);
c10::complex<float> l = std::log1p(x);
C10_ASSERT_NEAR(l.real(), 0.34657359027997264, tol);
C10_ASSERT_NEAR(l.imag(), 0.7853981633974483, tol);
}
{
c10::complex<float> x(1e30, 1e30);
c10::complex<float> l = std::log1p(x);
C10_ASSERT_NEAR(l.real(), 69.42412638010134, tol);
C10_ASSERT_NEAR(l.imag(), 0.7853981633974483, tol);
}
{
c10::complex<float> x(1e-38, 1e-38);
c10::complex<float> l = std::log1p(x);
C10_ASSERT_NEAR(l.real(), 1e-38, tol);
C10_ASSERT_NEAR(l.imag(), 1e-38, tol);
}
{
c10::complex<float> x(1e-38, 2e-30);
c10::complex<float> l = std::log1p(x);
C10_ASSERT_NEAR(l.real(), 1e-30, tol);
C10_ASSERT_NEAR(l.imag(), 2e-30, tol);
}
{
c10::complex<double> x(-1, 1e-250);
c10::complex<double> l = std::log1p(x);
C10_ASSERT_NEAR(l.real(), -575.6462732485114, tol);
C10_ASSERT_NEAR(l.imag(), 1.5707963267948966, tol);
}
{
c10::complex<double> x(-1, 1e250);
c10::complex<double> l = std::log1p(x);
C10_ASSERT_NEAR(l.real(), 575.6462732485114, tol);
C10_ASSERT_NEAR(l.imag(), 1.5707963267948966, tol);
}
{
c10::complex<double> x(1e250, 1);
c10::complex<double> l = std::log1p(x);
C10_ASSERT_NEAR(l.real(), 575.6462732485114, tol);
C10_ASSERT_NEAR(l.imag(), 1e-250, tol);
}
{
c10::complex<double> x(1e-250, 1);
c10::complex<double> l = std::log1p(x);
C10_ASSERT_NEAR(l.real(), 0.34657359027997264, tol);
C10_ASSERT_NEAR(l.imag(), 0.7853981633974483, tol);
}
{
c10::complex<double> x(1e250, 1e250);
c10::complex<double> l = std::log1p(x);
C10_ASSERT_NEAR(l.real(), 575.9928468387914, tol);
C10_ASSERT_NEAR(l.imag(), 0.7853981633974483, tol);
}
{
c10::complex<double> x(1e-250, 1e-250);
c10::complex<double> l = std::log1p(x);
C10_ASSERT_NEAR(l.real(), 1e-250, tol);
C10_ASSERT_NEAR(l.imag(), 1e-250, tol);
}
{
c10::complex<double> x(1e-250, 2e-250);
c10::complex<double> l = std::log1p(x);
C10_ASSERT_NEAR(l.real(), 1e-250, tol);
C10_ASSERT_NEAR(l.imag(), 2e-250, tol);
}
{
c10::complex<double> x(2e-308, 1.5e-250);
c10::complex<double> l = std::log1p(x);
C10_ASSERT_NEAR(l.real(), 2e-308, tol);
C10_ASSERT_NEAR(l.imag(), 1.5e-308, tol);
}
}
// Power functions
C10_DEFINE_TEST(TestPowSqrt, Equal) {
// x^0.5 = sqrt(x)
{
c10::complex<float> x(0.1, 1.2);
c10::complex<float> y = std::pow(x, float(0.5));
c10::complex<float> z = std::sqrt(x);
C10_ASSERT_NEAR(y.real(), z.real(), tol);
C10_ASSERT_NEAR(y.imag(), z.imag(), tol);
}
{
c10::complex<float> x(0.1, 1.2);
c10::complex<float> y = ::pow(x, float(0.5));
c10::complex<float> z = ::sqrt(x);
C10_ASSERT_NEAR(y.real(), z.real(), tol);
C10_ASSERT_NEAR(y.imag(), z.imag(), tol);
}
{
c10::complex<double> x(0.1, 1.2);
c10::complex<double> y = std::pow(x, double(0.5));
c10::complex<double> z = std::sqrt(x);
C10_ASSERT_NEAR(y.real(), z.real(), tol);
C10_ASSERT_NEAR(y.imag(), z.imag(), tol);
}
{
c10::complex<double> x(0.1, 1.2);
c10::complex<double> y = ::pow(x, double(0.5));
c10::complex<double> z = ::sqrt(x);
C10_ASSERT_NEAR(y.real(), z.real(), tol);
C10_ASSERT_NEAR(y.imag(), z.imag(), tol);
}
}
C10_DEFINE_TEST(TestPow, Square) {
// x^2 = x * x
{
c10::complex<float> x(0.1, 1.2);
c10::complex<float> y = std::pow(x, float(2));
c10::complex<float> z = x * x;
C10_ASSERT_NEAR(y.real(), z.real(), tol);
C10_ASSERT_NEAR(y.imag(), z.imag(), tol);
}
{
c10::complex<float> x(0.1, 1.2);
c10::complex<float> y = ::pow(x, float(2));
c10::complex<float> z = x * x;
C10_ASSERT_NEAR(y.real(), z.real(), tol);
C10_ASSERT_NEAR(y.imag(), z.imag(), tol);
}
{
c10::complex<double> x(0.1, 1.2);
c10::complex<double> y = std::pow(x, double(2));
c10::complex<double> z = x * x;
C10_ASSERT_NEAR(y.real(), z.real(), tol);
C10_ASSERT_NEAR(y.imag(), z.imag(), tol);
}
{
c10::complex<double> x(0.1, 1.2);
c10::complex<double> y = ::pow(x, double(2));
c10::complex<double> z = x * x;
C10_ASSERT_NEAR(y.real(), z.real(), tol);
C10_ASSERT_NEAR(y.imag(), z.imag(), tol);
}
}
// Trigonometric functions and hyperbolic functions
C10_DEFINE_TEST(TestSinCosSinhCosh, Identity) {
// sin(x + i * y) = sin(x) * cosh(y) + i * cos(x) * sinh(y)
// cos(x + i * y) = cos(x) * cosh(y) - i * sin(x) * sinh(y)
{
c10::complex<float> x(0.1, 1.2);
c10::complex<float> y = std::sin(x);
float expected_real = std::sin(x.real()) * std::cosh(x.imag());
float expected_imag = std::cos(x.real()) * std::sinh(x.imag());
C10_ASSERT_NEAR(y.real(), expected_real, tol);
C10_ASSERT_NEAR(y.imag(), expected_imag, tol);
}
{
c10::complex<float> x(0.1, 1.2);
c10::complex<float> y = ::sin(x);
float expected_real = ::sin(x.real()) * ::cosh(x.imag());
float expected_imag = ::cos(x.real()) * ::sinh(x.imag());
C10_ASSERT_NEAR(y.real(), expected_real, tol);
C10_ASSERT_NEAR(y.imag(), expected_imag, tol);
}
{
c10::complex<float> x(0.1, 1.2);
c10::complex<float> y = std::cos(x);
float expected_real = std::cos(x.real()) * std::cosh(x.imag());
float expected_imag = -std::sin(x.real()) * std::sinh(x.imag());
C10_ASSERT_NEAR(y.real(), expected_real, tol);
C10_ASSERT_NEAR(y.imag(), expected_imag, tol);
}
{
c10::complex<float> x(0.1, 1.2);
c10::complex<float> y = ::cos(x);
float expected_real = ::cos(x.real()) * ::cosh(x.imag());
float expected_imag = -::sin(x.real()) * ::sinh(x.imag());
C10_ASSERT_NEAR(y.real(), expected_real, tol);
C10_ASSERT_NEAR(y.imag(), expected_imag, tol);
}
{
c10::complex<double> x(0.1, 1.2);
c10::complex<double> y = std::sin(x);
float expected_real = std::sin(x.real()) * std::cosh(x.imag());
float expected_imag = std::cos(x.real()) * std::sinh(x.imag());
C10_ASSERT_NEAR(y.real(), expected_real, tol);
C10_ASSERT_NEAR(y.imag(), expected_imag, tol);
}
{
c10::complex<double> x(0.1, 1.2);
c10::complex<double> y = ::sin(x);
float expected_real = ::sin(x.real()) * ::cosh(x.imag());
float expected_imag = ::cos(x.real()) * ::sinh(x.imag());
C10_ASSERT_NEAR(y.real(), expected_real, tol);
C10_ASSERT_NEAR(y.imag(), expected_imag, tol);
}
{
c10::complex<double> x(0.1, 1.2);
c10::complex<double> y = std::cos(x);
float expected_real = std::cos(x.real()) * std::cosh(x.imag());
float expected_imag = -std::sin(x.real()) * std::sinh(x.imag());
C10_ASSERT_NEAR(y.real(), expected_real, tol);
C10_ASSERT_NEAR(y.imag(), expected_imag, tol);
}
{
c10::complex<double> x(0.1, 1.2);
c10::complex<double> y = ::cos(x);
float expected_real = ::cos(x.real()) * ::cosh(x.imag());
float expected_imag = -::sin(x.real()) * ::sinh(x.imag());
C10_ASSERT_NEAR(y.real(), expected_real, tol);
C10_ASSERT_NEAR(y.imag(), expected_imag, tol);
}
}
C10_DEFINE_TEST(TestTan, Identity) {
// tan(x) = sin(x) / cos(x)
{
c10::complex<float> x(0.1, 1.2);
c10::complex<float> y = std::tan(x);
c10::complex<float> z = std::sin(x) / std::cos(x);
C10_ASSERT_NEAR(y.real(), z.real(), tol);
C10_ASSERT_NEAR(y.imag(), z.imag(), tol);
}
{
c10::complex<float> x(0.1, 1.2);
c10::complex<float> y = ::tan(x);
c10::complex<float> z = ::sin(x) / ::cos(x);
C10_ASSERT_NEAR(y.real(), z.real(), tol);
C10_ASSERT_NEAR(y.imag(), z.imag(), tol);
}
{
c10::complex<double> x(0.1, 1.2);
c10::complex<double> y = std::tan(x);
c10::complex<double> z = std::sin(x) / std::cos(x);
C10_ASSERT_NEAR(y.real(), z.real(), tol);
C10_ASSERT_NEAR(y.imag(), z.imag(), tol);
}
{
c10::complex<double> x(0.1, 1.2);
c10::complex<double> y = ::tan(x);
c10::complex<double> z = ::sin(x) / ::cos(x);
C10_ASSERT_NEAR(y.real(), z.real(), tol);
C10_ASSERT_NEAR(y.imag(), z.imag(), tol);
}
}
C10_DEFINE_TEST(TestTanh, Identity) {
// tanh(x) = sinh(x) / cosh(x)
{
c10::complex<float> x(0.1, 1.2);
c10::complex<float> y = std::tanh(x);
c10::complex<float> z = std::sinh(x) / std::cosh(x);
C10_ASSERT_NEAR(y.real(), z.real(), tol);
C10_ASSERT_NEAR(y.imag(), z.imag(), tol);
}
{
c10::complex<float> x(0.1, 1.2);
c10::complex<float> y = ::tanh(x);
c10::complex<float> z = ::sinh(x) / ::cosh(x);
C10_ASSERT_NEAR(y.real(), z.real(), tol);
C10_ASSERT_NEAR(y.imag(), z.imag(), tol);
}
{
c10::complex<double> x(0.1, 1.2);
c10::complex<double> y = std::tanh(x);
c10::complex<double> z = std::sinh(x) / std::cosh(x);
C10_ASSERT_NEAR(y.real(), z.real(), tol);
C10_ASSERT_NEAR(y.imag(), z.imag(), tol);
}
{
c10::complex<double> x(0.1, 1.2);
c10::complex<double> y = ::tanh(x);
c10::complex<double> z = ::sinh(x) / ::cosh(x);
C10_ASSERT_NEAR(y.real(), z.real(), tol);
C10_ASSERT_NEAR(y.imag(), z.imag(), tol);
}
}
// Rev trigonometric functions
C10_DEFINE_TEST(TestRevTrigonometric, Rev) {
// asin(sin(x)) = x
// acos(cos(x)) = x
// atan(tan(x)) = x
{
c10::complex<float> x(0.5, 0.6);
c10::complex<float> s = std::sin(x);
c10::complex<float> ss = std::asin(s);
c10::complex<float> c = std::cos(x);
c10::complex<float> cc = std::acos(c);
c10::complex<float> t = std::tan(x);
c10::complex<float> tt = std::atan(t);
C10_ASSERT_NEAR(x.real(), ss.real(), tol);
C10_ASSERT_NEAR(x.imag(), ss.imag(), tol);
C10_ASSERT_NEAR(x.real(), cc.real(), tol);
C10_ASSERT_NEAR(x.imag(), cc.imag(), tol);
C10_ASSERT_NEAR(x.real(), tt.real(), tol);
C10_ASSERT_NEAR(x.imag(), tt.imag(), tol);
}
{
c10::complex<float> x(0.5, 0.6);
c10::complex<float> s = ::sin(x);
c10::complex<float> ss = ::asin(s);
c10::complex<float> c = ::cos(x);
c10::complex<float> cc = ::acos(c);
c10::complex<float> t = ::tan(x);
c10::complex<float> tt = ::atan(t);
C10_ASSERT_NEAR(x.real(), ss.real(), tol);
C10_ASSERT_NEAR(x.imag(), ss.imag(), tol);
C10_ASSERT_NEAR(x.real(), cc.real(), tol);
C10_ASSERT_NEAR(x.imag(), cc.imag(), tol);
C10_ASSERT_NEAR(x.real(), tt.real(), tol);
C10_ASSERT_NEAR(x.imag(), tt.imag(), tol);
}
{
c10::complex<double> x(0.5, 0.6);
c10::complex<double> s = std::sin(x);
c10::complex<double> ss = std::asin(s);
c10::complex<double> c = std::cos(x);
c10::complex<double> cc = std::acos(c);
c10::complex<double> t = std::tan(x);
c10::complex<double> tt = std::atan(t);
C10_ASSERT_NEAR(x.real(), ss.real(), tol);
C10_ASSERT_NEAR(x.imag(), ss.imag(), tol);
C10_ASSERT_NEAR(x.real(), cc.real(), tol);
C10_ASSERT_NEAR(x.imag(), cc.imag(), tol);
C10_ASSERT_NEAR(x.real(), tt.real(), tol);
C10_ASSERT_NEAR(x.imag(), tt.imag(), tol);
}
{
c10::complex<double> x(0.5, 0.6);
c10::complex<double> s = ::sin(x);
c10::complex<double> ss = ::asin(s);
c10::complex<double> c = ::cos(x);
c10::complex<double> cc = ::acos(c);
c10::complex<double> t = ::tan(x);
c10::complex<double> tt = ::atan(t);
C10_ASSERT_NEAR(x.real(), ss.real(), tol);
C10_ASSERT_NEAR(x.imag(), ss.imag(), tol);
C10_ASSERT_NEAR(x.real(), cc.real(), tol);
C10_ASSERT_NEAR(x.imag(), cc.imag(), tol);
C10_ASSERT_NEAR(x.real(), tt.real(), tol);
C10_ASSERT_NEAR(x.imag(), tt.imag(), tol);
}
}
// Rev hyperbolic functions
C10_DEFINE_TEST(TestRevHyperbolic, Rev) {
// asinh(sinh(x)) = x
// acosh(cosh(x)) = x
// atanh(tanh(x)) = x
{
c10::complex<float> x(0.5, 0.6);
c10::complex<float> s = std::sinh(x);
c10::complex<float> ss = std::asinh(s);
c10::complex<float> c = std::cosh(x);
c10::complex<float> cc = std::acosh(c);
c10::complex<float> t = std::tanh(x);
c10::complex<float> tt = std::atanh(t);
C10_ASSERT_NEAR(x.real(), ss.real(), tol);
C10_ASSERT_NEAR(x.imag(), ss.imag(), tol);
C10_ASSERT_NEAR(x.real(), cc.real(), tol);
C10_ASSERT_NEAR(x.imag(), cc.imag(), tol);
C10_ASSERT_NEAR(x.real(), tt.real(), tol);
C10_ASSERT_NEAR(x.imag(), tt.imag(), tol);
}
{
c10::complex<float> x(0.5, 0.6);
c10::complex<float> s = ::sinh(x);
c10::complex<float> ss = ::asinh(s);
c10::complex<float> c = ::cosh(x);
c10::complex<float> cc = ::acosh(c);
c10::complex<float> t = ::tanh(x);
c10::complex<float> tt = ::atanh(t);
C10_ASSERT_NEAR(x.real(), ss.real(), tol);
C10_ASSERT_NEAR(x.imag(), ss.imag(), tol);
C10_ASSERT_NEAR(x.real(), cc.real(), tol);
C10_ASSERT_NEAR(x.imag(), cc.imag(), tol);
C10_ASSERT_NEAR(x.real(), tt.real(), tol);
C10_ASSERT_NEAR(x.imag(), tt.imag(), tol);
}
{
c10::complex<double> x(0.5, 0.6);
c10::complex<double> s = std::sinh(x);
c10::complex<double> ss = std::asinh(s);
c10::complex<double> c = std::cosh(x);
c10::complex<double> cc = std::acosh(c);
c10::complex<double> t = std::tanh(x);
c10::complex<double> tt = std::atanh(t);
C10_ASSERT_NEAR(x.real(), ss.real(), tol);
C10_ASSERT_NEAR(x.imag(), ss.imag(), tol);
C10_ASSERT_NEAR(x.real(), cc.real(), tol);
C10_ASSERT_NEAR(x.imag(), cc.imag(), tol);
C10_ASSERT_NEAR(x.real(), tt.real(), tol);
C10_ASSERT_NEAR(x.imag(), tt.imag(), tol);
}
{
c10::complex<double> x(0.5, 0.6);
c10::complex<double> s = ::sinh(x);
c10::complex<double> ss = ::asinh(s);
c10::complex<double> c = ::cosh(x);
c10::complex<double> cc = ::acosh(c);
c10::complex<double> t = ::tanh(x);
c10::complex<double> tt = ::atanh(t);
C10_ASSERT_NEAR(x.real(), ss.real(), tol);
C10_ASSERT_NEAR(x.imag(), ss.imag(), tol);
C10_ASSERT_NEAR(x.real(), cc.real(), tol);
C10_ASSERT_NEAR(x.imag(), cc.imag(), tol);
C10_ASSERT_NEAR(x.real(), tt.real(), tol);
C10_ASSERT_NEAR(x.imag(), tt.imag(), tol);
}
}