Program Listing for File complex_mpc.cpp

Return to documentation for file (symengine/symengine/complex_mpc.cpp)

#include <symengine/complex_mpc.h>
#include <symengine/eval_mpc.h>
#include <symengine/symengine_exception.h>

#ifdef HAVE_SYMENGINE_MPC
namespace SymEngine
{

ComplexMPC::ComplexMPC(mpc_class i) : i{std::move(i)}
{
    SYMENGINE_ASSIGN_TYPEID()
}

hash_t ComplexMPC::__hash__() const
{
    hash_t seed = SYMENGINE_COMPLEX_MPC;
    hash_combine_impl(seed, mpc_realref(i.get_mpc_t()));
    hash_combine_impl(seed, mpc_imagref(i.get_mpc_t()));
    return seed;
}

bool ComplexMPC::__eq__(const Basic &o) const
{
    if (is_a<ComplexMPC>(o)) {
        const ComplexMPC &s = down_cast<const ComplexMPC &>(o);
        if (get_prec() == s.get_prec()) {
            return mpc_cmp(this->i.get_mpc_t(), s.i.get_mpc_t()) == 0;
        }
    }
    return false;
}

int ComplexMPC::compare(const Basic &o) const
{
    SYMENGINE_ASSERT(is_a<ComplexMPC>(o))
    const ComplexMPC &s = down_cast<const ComplexMPC &>(o);
    if (get_prec() == s.get_prec()) {
        int cmp = mpc_cmp(this->i.get_mpc_t(), s.i.get_mpc_t());
        int x = MPC_INEX_RE(cmp), y = MPC_INEX_IM(cmp);
        if (x == 0) {
            if (y != 0)
                return y > 0 ? 1 : -1;
            return 0;
        }
        return x > 0 ? 1 : -1;
    } else {
        return get_prec() > s.get_prec() ? 1 : -1;
    }
}

RCP<const Number> ComplexMPC::real_part() const
{
    mpfr_class t(get_prec());
    mpc_real(t.get_mpfr_t(), i.get_mpc_t(), MPFR_RNDN);
    return real_mpfr(std::move(t));
}

RCP<const Number> ComplexMPC::imaginary_part() const
{
    mpfr_class t(get_prec());
    mpc_imag(t.get_mpfr_t(), i.get_mpc_t(), MPFR_RNDN);
    return real_mpfr(std::move(t));
}

RCP<const Basic> ComplexMPC::conjugate() const
{
    mpc_class t(get_prec());
    mpc_conj(t.get_mpc_t(), i.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::add(const Integer &other) const
{
    mpc_class t(get_prec());
    mpc_set_z(t.get_mpc_t(), get_mpz_t(other.as_integer_class()), MPFR_RNDN);
    mpc_add(t.get_mpc_t(), i.get_mpc_t(), t.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::add(const Rational &other) const
{
    mpc_class t(get_prec());
    mpc_set_q(t.get_mpc_t(), get_mpq_t(other.as_rational_class()), MPFR_RNDN);
    mpc_add(t.get_mpc_t(), i.get_mpc_t(), t.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::add(const Complex &other) const
{
    mpc_class t(get_prec());
    mpc_set_q_q(t.get_mpc_t(), get_mpq_t(other.real_),
                get_mpq_t(other.imaginary_), MPFR_RNDN);
    mpc_add(t.get_mpc_t(), i.get_mpc_t(), t.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::add(const RealDouble &other) const
{
    mpc_class t(get_prec());
    mpc_set_d(t.get_mpc_t(), other.i, MPFR_RNDN);
    mpc_add(t.get_mpc_t(), i.get_mpc_t(), t.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::add(const ComplexDouble &other) const
{
    mpc_class t(get_prec());
    mpc_set_d_d(t.get_mpc_t(), other.i.real(), other.i.imag(), MPFR_RNDN);
    mpc_add(t.get_mpc_t(), i.get_mpc_t(), t.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::add(const RealMPFR &other) const
{
    mpc_class t(std::max(get_prec(), other.get_prec()));
    mpc_add_fr(t.get_mpc_t(), i.get_mpc_t(), other.i.get_mpfr_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::add(const ComplexMPC &other) const
{
    mpc_class t(std::max(get_prec(), other.get_prec()));
    mpc_add(t.get_mpc_t(), i.get_mpc_t(), other.i.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::sub(const Integer &other) const
{
    mpc_class t(get_prec());
    mpc_set_z(t.get_mpc_t(), get_mpz_t(other.as_integer_class()), MPFR_RNDN);
    mpc_sub(t.get_mpc_t(), i.get_mpc_t(), t.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::sub(const Rational &other) const
{
    mpc_class t(get_prec());
    mpc_set_q(t.get_mpc_t(), get_mpq_t(other.as_rational_class()), MPFR_RNDN);
    mpc_sub(t.get_mpc_t(), i.get_mpc_t(), t.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::sub(const Complex &other) const
{
    mpc_class t(get_prec());
    mpc_set_q_q(t.get_mpc_t(), get_mpq_t(other.real_),
                get_mpq_t(other.imaginary_), MPFR_RNDN);
    mpc_sub(t.get_mpc_t(), i.get_mpc_t(), t.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::sub(const RealDouble &other) const
{
    mpc_class t(get_prec());
    mpc_set_d(t.get_mpc_t(), other.i, MPFR_RNDN);
    mpc_sub(t.get_mpc_t(), i.get_mpc_t(), t.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::sub(const ComplexDouble &other) const
{
    mpc_class t(get_prec());
    mpc_set_d_d(t.get_mpc_t(), other.i.real(), other.i.imag(), MPFR_RNDN);
    mpc_sub(t.get_mpc_t(), i.get_mpc_t(), t.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::sub(const RealMPFR &other) const
{
    mpc_class t(std::max(get_prec(), other.get_prec()));
    mpc_sub_fr(t.get_mpc_t(), i.get_mpc_t(), other.i.get_mpfr_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::sub(const ComplexMPC &other) const
{
    mpc_class t(std::max(get_prec(), other.get_prec()));
    mpc_sub(t.get_mpc_t(), i.get_mpc_t(), other.i.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::rsub(const Integer &other) const
{
    mpc_class t(get_prec());
    mpc_set_z(t.get_mpc_t(), get_mpz_t(other.as_integer_class()), MPFR_RNDN);
    mpc_sub(t.get_mpc_t(), t.get_mpc_t(), i.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::rsub(const Rational &other) const
{
    mpc_class t(get_prec());
    mpc_set_q(t.get_mpc_t(), get_mpq_t(other.as_rational_class()), MPFR_RNDN);
    mpc_sub(t.get_mpc_t(), t.get_mpc_t(), i.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::rsub(const Complex &other) const
{
    mpc_class t(get_prec());
    mpc_set_q_q(t.get_mpc_t(), get_mpq_t(other.real_),
                get_mpq_t(other.imaginary_), MPFR_RNDN);
    mpc_sub(t.get_mpc_t(), t.get_mpc_t(), i.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::rsub(const RealDouble &other) const
{
    mpc_class t(get_prec());
    mpc_set_d(t.get_mpc_t(), other.i, MPFR_RNDN);
    mpc_sub(t.get_mpc_t(), t.get_mpc_t(), i.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::rsub(const ComplexDouble &other) const
{
    mpc_class t(get_prec());
    mpc_set_d_d(t.get_mpc_t(), other.i.real(), other.i.imag(), MPFR_RNDN);
    mpc_sub(t.get_mpc_t(), t.get_mpc_t(), i.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::rsub(const RealMPFR &other) const
{
    mpc_class t(std::max(get_prec(), other.get_prec()));
    mpc_fr_sub(t.get_mpc_t(), other.i.get_mpfr_t(), i.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::mul(const Integer &other) const
{
    mpc_class t(get_prec());
    mpc_set_z(t.get_mpc_t(), get_mpz_t(other.as_integer_class()), MPFR_RNDN);
    mpc_mul(t.get_mpc_t(), i.get_mpc_t(), t.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::mul(const Rational &other) const
{
    mpc_class t(get_prec());
    mpc_set_q(t.get_mpc_t(), get_mpq_t(other.as_rational_class()), MPFR_RNDN);
    mpc_mul(t.get_mpc_t(), i.get_mpc_t(), t.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::mul(const Complex &other) const
{
    mpc_class t(get_prec());
    mpc_set_q_q(t.get_mpc_t(), get_mpq_t(other.real_),
                get_mpq_t(other.imaginary_), MPFR_RNDN);
    mpc_mul(t.get_mpc_t(), i.get_mpc_t(), t.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::mul(const RealDouble &other) const
{
    mpc_class t(get_prec());
    mpc_set_d(t.get_mpc_t(), other.i, MPFR_RNDN);
    mpc_mul(t.get_mpc_t(), i.get_mpc_t(), t.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::mul(const ComplexDouble &other) const
{
    mpc_class t(get_prec());
    mpc_set_d_d(t.get_mpc_t(), other.i.real(), other.i.imag(), MPFR_RNDN);
    mpc_mul(t.get_mpc_t(), i.get_mpc_t(), t.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::mul(const RealMPFR &other) const
{
    mpc_class t(std::max(get_prec(), other.get_prec()));
    mpc_mul_fr(t.get_mpc_t(), i.get_mpc_t(), other.i.get_mpfr_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::mul(const ComplexMPC &other) const
{
    mpc_class t(std::max(get_prec(), other.get_prec()));
    mpc_mul(t.get_mpc_t(), i.get_mpc_t(), other.i.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::div(const Integer &other) const
{
    mpc_class t(get_prec());
    mpc_set_z(t.get_mpc_t(), get_mpz_t(other.as_integer_class()), MPFR_RNDN);
    mpc_div(t.get_mpc_t(), i.get_mpc_t(), t.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::div(const Rational &other) const
{
    mpc_class t(get_prec());
    mpc_set_q(t.get_mpc_t(), get_mpq_t(other.as_rational_class()), MPFR_RNDN);
    mpc_div(t.get_mpc_t(), i.get_mpc_t(), t.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::div(const Complex &other) const
{
    mpc_class t(get_prec());
    mpc_set_q_q(t.get_mpc_t(), get_mpq_t(other.real_),
                get_mpq_t(other.imaginary_), MPFR_RNDN);
    mpc_div(t.get_mpc_t(), i.get_mpc_t(), t.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::div(const RealDouble &other) const
{
    mpc_class t(get_prec());
    mpc_set_d(t.get_mpc_t(), other.i, MPFR_RNDN);
    mpc_div(t.get_mpc_t(), i.get_mpc_t(), t.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::div(const ComplexDouble &other) const
{
    mpc_class t(get_prec());
    mpc_set_d_d(t.get_mpc_t(), other.i.real(), other.i.imag(), MPFR_RNDN);
    mpc_div(t.get_mpc_t(), i.get_mpc_t(), t.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::div(const RealMPFR &other) const
{
    mpc_class t(std::max(get_prec(), other.get_prec()));
    mpc_div_fr(t.get_mpc_t(), i.get_mpc_t(), other.i.get_mpfr_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::div(const ComplexMPC &other) const
{
    mpc_class t(std::max(get_prec(), other.get_prec()));
    mpc_div(t.get_mpc_t(), i.get_mpc_t(), other.i.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::rdiv(const Integer &other) const
{
    mpc_class t(get_prec());
    mpc_set_z(t.get_mpc_t(), get_mpz_t(other.as_integer_class()), MPFR_RNDN);
    mpc_div(t.get_mpc_t(), t.get_mpc_t(), i.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::rdiv(const Rational &other) const
{
    mpc_class t(get_prec());
    mpc_set_q(t.get_mpc_t(), get_mpq_t(other.as_rational_class()), MPFR_RNDN);
    mpc_div(t.get_mpc_t(), t.get_mpc_t(), i.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::rdiv(const Complex &other) const
{
    mpc_class t(get_prec());
    mpc_set_q_q(t.get_mpc_t(), get_mpq_t(other.real_),
                get_mpq_t(other.imaginary_), MPFR_RNDN);
    mpc_div(t.get_mpc_t(), t.get_mpc_t(), i.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::rdiv(const RealDouble &other) const
{
    mpc_class t(get_prec());
    mpc_set_d(t.get_mpc_t(), other.i, MPFR_RNDN);
    mpc_div(t.get_mpc_t(), t.get_mpc_t(), i.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::rdiv(const ComplexDouble &other) const
{
    mpc_class t(get_prec());
    mpc_set_d_d(t.get_mpc_t(), other.i.real(), other.i.imag(), MPFR_RNDN);
    mpc_div(t.get_mpc_t(), t.get_mpc_t(), i.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::rdiv(const RealMPFR &other) const
{
    mpc_class t(std::max(get_prec(), other.get_prec()));
    mpc_fr_div(t.get_mpc_t(), other.i.get_mpfr_t(), i.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::pow(const Integer &other) const
{
    mpc_class t(get_prec());
    mpc_set_z(t.get_mpc_t(), get_mpz_t(other.as_integer_class()), MPFR_RNDN);
    mpc_pow(t.get_mpc_t(), i.get_mpc_t(), t.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::pow(const Rational &other) const
{
    mpc_class t(get_prec());
    mpc_set_q(t.get_mpc_t(), get_mpq_t(other.as_rational_class()), MPFR_RNDN);
    mpc_pow(t.get_mpc_t(), i.get_mpc_t(), t.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::pow(const Complex &other) const
{
    mpc_class t(get_prec());
    mpc_set_q_q(t.get_mpc_t(), get_mpq_t(other.real_),
                get_mpq_t(other.imaginary_), MPFR_RNDN);
    mpc_pow(t.get_mpc_t(), i.get_mpc_t(), t.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::pow(const RealDouble &other) const
{
    mpc_class t(get_prec());
    mpc_set_d(t.get_mpc_t(), other.i, MPFR_RNDN);
    mpc_pow(t.get_mpc_t(), i.get_mpc_t(), t.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::pow(const ComplexDouble &other) const
{
    mpc_class t(get_prec());
    mpc_set_d_d(t.get_mpc_t(), other.i.real(), other.i.imag(), MPFR_RNDN);
    mpc_pow(t.get_mpc_t(), i.get_mpc_t(), t.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::pow(const RealMPFR &other) const
{
    mpc_class t(std::max(get_prec(), other.get_prec()));
    mpc_pow_fr(t.get_mpc_t(), i.get_mpc_t(), other.i.get_mpfr_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::pow(const ComplexMPC &other) const
{
    mpc_class t(std::max(get_prec(), other.get_prec()));
    mpc_pow(t.get_mpc_t(), i.get_mpc_t(), other.i.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::rpow(const Integer &other) const
{
    mpc_class t(get_prec());
    mpc_set_z(t.get_mpc_t(), get_mpz_t(other.as_integer_class()), MPFR_RNDN);
    mpc_pow(t.get_mpc_t(), t.get_mpc_t(), i.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::rpow(const Rational &other) const
{
    mpc_class t(get_prec());
    mpc_set_q(t.get_mpc_t(), get_mpq_t(other.as_rational_class()), MPFR_RNDN);
    mpc_pow(t.get_mpc_t(), t.get_mpc_t(), i.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::rpow(const Complex &other) const
{
    mpc_class t(get_prec());
    mpc_set_q_q(t.get_mpc_t(), get_mpq_t(other.real_),
                get_mpq_t(other.imaginary_), MPFR_RNDN);
    mpc_pow(t.get_mpc_t(), t.get_mpc_t(), i.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::rpow(const RealDouble &other) const
{
    mpc_class t(get_prec());
    mpc_set_d(t.get_mpc_t(), other.i, MPFR_RNDN);
    mpc_pow(t.get_mpc_t(), t.get_mpc_t(), i.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::rpow(const ComplexDouble &other) const
{
    mpc_class t(get_prec());
    mpc_set_d_d(t.get_mpc_t(), other.i.real(), other.i.imag(), MPFR_RNDN);
    mpc_pow(t.get_mpc_t(), t.get_mpc_t(), i.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

RCP<const Number> ComplexMPC::rpow(const RealMPFR &other) const
{
    mpc_class t(std::max(get_prec(), other.get_prec()));
    mpc_set_fr(t.get_mpc_t(), other.i.get_mpfr_t(), MPFR_RNDN);
    mpc_pow(t.get_mpc_t(), t.get_mpc_t(), i.get_mpc_t(), MPFR_RNDN);
    return rcp(new ComplexMPC(std::move(t)));
}

class EvaluateMPC : public Evaluate
{
    virtual RCP<const Basic> sin(const Basic &x) const
    {
        SYMENGINE_ASSERT(is_a<ComplexMPC>(x))
        mpc_class t(down_cast<const ComplexMPC &>(x).as_mpc().get_prec());
        mpc_sin(t.get_mpc_t(),
                down_cast<const ComplexMPC &>(x).as_mpc().get_mpc_t(),
                MPFR_RNDN);
        return complex_mpc(std::move(t));
    }
    virtual RCP<const Basic> cos(const Basic &x) const
    {
        SYMENGINE_ASSERT(is_a<ComplexMPC>(x))
        mpc_class t(down_cast<const ComplexMPC &>(x).as_mpc().get_prec());
        mpc_cos(t.get_mpc_t(),
                down_cast<const ComplexMPC &>(x).as_mpc().get_mpc_t(),
                MPFR_RNDN);
        return complex_mpc(std::move(t));
    }
    virtual RCP<const Basic> tan(const Basic &x) const
    {
        SYMENGINE_ASSERT(is_a<ComplexMPC>(x))
        mpc_class t(down_cast<const ComplexMPC &>(x).as_mpc().get_prec());
        mpc_tan(t.get_mpc_t(),
                down_cast<const ComplexMPC &>(x).as_mpc().get_mpc_t(),
                MPFR_RNDN);
        return complex_mpc(std::move(t));
    }
    virtual RCP<const Basic> cot(const Basic &x) const
    {
        SYMENGINE_ASSERT(is_a<ComplexMPC>(x))
        mpc_class t(down_cast<const ComplexMPC &>(x).as_mpc().get_prec());
        mpc_tan(t.get_mpc_t(),
                down_cast<const ComplexMPC &>(x).as_mpc().get_mpc_t(),
                MPFR_RNDN);
        mpc_ui_div(t.get_mpc_t(), 1, t.get_mpc_t(), MPFR_RNDN);
        return complex_mpc(std::move(t));
    }
    virtual RCP<const Basic> sec(const Basic &x) const
    {
        SYMENGINE_ASSERT(is_a<ComplexMPC>(x))
        mpc_class t(down_cast<const ComplexMPC &>(x).as_mpc().get_prec());
        mpc_cos(t.get_mpc_t(),
                down_cast<const ComplexMPC &>(x).as_mpc().get_mpc_t(),
                MPFR_RNDN);
        mpc_ui_div(t.get_mpc_t(), 1, t.get_mpc_t(), MPFR_RNDN);
        return complex_mpc(std::move(t));
    }
    virtual RCP<const Basic> csc(const Basic &x) const
    {
        SYMENGINE_ASSERT(is_a<ComplexMPC>(x))
        mpc_class t(down_cast<const ComplexMPC &>(x).as_mpc().get_prec());
        mpc_sin(t.get_mpc_t(),
                down_cast<const ComplexMPC &>(x).as_mpc().get_mpc_t(),
                MPFR_RNDN);
        mpc_ui_div(t.get_mpc_t(), 1, t.get_mpc_t(), MPFR_RNDN);
        return complex_mpc(std::move(t));
    }
    virtual RCP<const Basic> asin(const Basic &x) const
    {
        SYMENGINE_ASSERT(is_a<ComplexMPC>(x))
        mpc_class t(down_cast<const ComplexMPC &>(x).as_mpc().get_prec());
        mpc_asin(t.get_mpc_t(),
                 down_cast<const ComplexMPC &>(x).as_mpc().get_mpc_t(),
                 MPFR_RNDN);
        return complex_mpc(std::move(t));
    }
    virtual RCP<const Basic> acos(const Basic &x) const
    {
        SYMENGINE_ASSERT(is_a<ComplexMPC>(x))
        mpc_class t(down_cast<const ComplexMPC &>(x).as_mpc().get_prec());
        mpc_acos(t.get_mpc_t(),
                 down_cast<const ComplexMPC &>(x).as_mpc().get_mpc_t(),
                 MPFR_RNDN);
        return complex_mpc(std::move(t));
    }
    virtual RCP<const Basic> atan(const Basic &x) const
    {
        SYMENGINE_ASSERT(is_a<ComplexMPC>(x))
        mpc_class t(down_cast<const ComplexMPC &>(x).as_mpc().get_prec());
        mpc_atan(t.get_mpc_t(),
                 down_cast<const ComplexMPC &>(x).as_mpc().get_mpc_t(),
                 MPFR_RNDN);
        return complex_mpc(std::move(t));
    }
    virtual RCP<const Basic> acot(const Basic &x) const
    {
        SYMENGINE_ASSERT(is_a<ComplexMPC>(x))
        mpc_class t(down_cast<const ComplexMPC &>(x).as_mpc().get_prec());
        mpc_ui_div(t.get_mpc_t(), 1,
                   down_cast<const ComplexMPC &>(x).as_mpc().get_mpc_t(),
                   MPFR_RNDN);
        mpc_atan(t.get_mpc_t(), t.get_mpc_t(), MPFR_RNDN);
        return complex_mpc(std::move(t));
    }
    virtual RCP<const Basic> asec(const Basic &x) const
    {
        SYMENGINE_ASSERT(is_a<ComplexMPC>(x))
        mpc_class t(down_cast<const ComplexMPC &>(x).as_mpc().get_prec());
        mpc_ui_div(t.get_mpc_t(), 1,
                   down_cast<const ComplexMPC &>(x).as_mpc().get_mpc_t(),
                   MPFR_RNDN);
        mpc_acos(t.get_mpc_t(), t.get_mpc_t(), MPFR_RNDN);
        return complex_mpc(std::move(t));
    }
    virtual RCP<const Basic> acsc(const Basic &x) const
    {
        SYMENGINE_ASSERT(is_a<ComplexMPC>(x))
        mpc_class t(down_cast<const ComplexMPC &>(x).as_mpc().get_prec());
        mpc_ui_div(t.get_mpc_t(), 1,
                   down_cast<const ComplexMPC &>(x).as_mpc().get_mpc_t(),
                   MPFR_RNDN);
        mpc_asin(t.get_mpc_t(), t.get_mpc_t(), MPFR_RNDN);
        return complex_mpc(std::move(t));
    }
    virtual RCP<const Basic> sinh(const Basic &x) const
    {
        SYMENGINE_ASSERT(is_a<ComplexMPC>(x))
        mpc_class t(down_cast<const ComplexMPC &>(x).as_mpc().get_prec());
        mpc_sinh(t.get_mpc_t(),
                 down_cast<const ComplexMPC &>(x).as_mpc().get_mpc_t(),
                 MPFR_RNDN);
        return complex_mpc(std::move(t));
    }
    virtual RCP<const Basic> csch(const Basic &x) const
    {
        SYMENGINE_ASSERT(is_a<ComplexMPC>(x))
        mpc_class t(down_cast<const ComplexMPC &>(x).as_mpc().get_prec());
        mpc_sinh(t.get_mpc_t(),
                 down_cast<const ComplexMPC &>(x).as_mpc().get_mpc_t(),
                 MPFR_RNDN);
        mpc_ui_div(t.get_mpc_t(), 1, t.get_mpc_t(), MPFR_RNDN);
        return complex_mpc(std::move(t));
    }
    virtual RCP<const Basic> cosh(const Basic &x) const
    {
        SYMENGINE_ASSERT(is_a<ComplexMPC>(x))
        mpc_class t(down_cast<const ComplexMPC &>(x).as_mpc().get_prec());
        mpc_cosh(t.get_mpc_t(),
                 down_cast<const ComplexMPC &>(x).as_mpc().get_mpc_t(),
                 MPFR_RNDN);
        return complex_mpc(std::move(t));
    }
    virtual RCP<const Basic> sech(const Basic &x) const
    {
        SYMENGINE_ASSERT(is_a<ComplexMPC>(x))
        mpc_class t(down_cast<const ComplexMPC &>(x).as_mpc().get_prec());
        mpc_cosh(t.get_mpc_t(),
                 down_cast<const ComplexMPC &>(x).as_mpc().get_mpc_t(),
                 MPFR_RNDN);
        mpc_ui_div(t.get_mpc_t(), 1, t.get_mpc_t(), MPFR_RNDN);
        return complex_mpc(std::move(t));
    }
    virtual RCP<const Basic> tanh(const Basic &x) const
    {
        SYMENGINE_ASSERT(is_a<ComplexMPC>(x))
        mpc_class t(down_cast<const ComplexMPC &>(x).as_mpc().get_prec());
        mpc_tanh(t.get_mpc_t(),
                 down_cast<const ComplexMPC &>(x).as_mpc().get_mpc_t(),
                 MPFR_RNDN);
        return complex_mpc(std::move(t));
    }
    virtual RCP<const Basic> coth(const Basic &x) const
    {
        SYMENGINE_ASSERT(is_a<ComplexMPC>(x))
        mpc_class t(down_cast<const ComplexMPC &>(x).as_mpc().get_prec());
        mpc_tanh(t.get_mpc_t(),
                 down_cast<const ComplexMPC &>(x).as_mpc().get_mpc_t(),
                 MPFR_RNDN);
        mpc_ui_div(t.get_mpc_t(), 1, t.get_mpc_t(), MPFR_RNDN);
        return complex_mpc(std::move(t));
    }
    virtual RCP<const Basic> asinh(const Basic &x) const
    {
        SYMENGINE_ASSERT(is_a<ComplexMPC>(x))
        mpc_class t(down_cast<const ComplexMPC &>(x).as_mpc().get_prec());
        mpc_asinh(t.get_mpc_t(),
                  down_cast<const ComplexMPC &>(x).as_mpc().get_mpc_t(),
                  MPFR_RNDN);
        return complex_mpc(std::move(t));
    }
    virtual RCP<const Basic> acsch(const Basic &x) const
    {
        SYMENGINE_ASSERT(is_a<ComplexMPC>(x))
        mpc_class t(down_cast<const ComplexMPC &>(x).as_mpc().get_prec());
        mpc_ui_div(t.get_mpc_t(), 1,
                   down_cast<const ComplexMPC &>(x).as_mpc().get_mpc_t(),
                   MPFR_RNDN);
        mpc_asinh(t.get_mpc_t(), t.get_mpc_t(), MPFR_RNDN);
        return complex_mpc(std::move(t));
    }
    virtual RCP<const Basic> acosh(const Basic &x) const
    {
        SYMENGINE_ASSERT(is_a<ComplexMPC>(x))
        mpc_class t(down_cast<const ComplexMPC &>(x).as_mpc().get_prec());
        mpc_acosh(t.get_mpc_t(),
                  down_cast<const ComplexMPC &>(x).as_mpc().get_mpc_t(),
                  MPFR_RNDN);
        return complex_mpc(std::move(t));
    }
    virtual RCP<const Basic> atanh(const Basic &x) const
    {
        SYMENGINE_ASSERT(is_a<ComplexMPC>(x))
        mpc_class t(down_cast<const ComplexMPC &>(x).as_mpc().get_prec());
        mpc_atanh(t.get_mpc_t(),
                  down_cast<const ComplexMPC &>(x).as_mpc().get_mpc_t(),
                  MPFR_RNDN);
        return complex_mpc(std::move(t));
    }
    virtual RCP<const Basic> acoth(const Basic &x) const
    {
        SYMENGINE_ASSERT(is_a<ComplexMPC>(x))
        mpc_class t(down_cast<const ComplexMPC &>(x).as_mpc().get_prec());
        mpc_ui_div(t.get_mpc_t(), 1,
                   down_cast<const ComplexMPC &>(x).as_mpc().get_mpc_t(),
                   MPFR_RNDN);
        mpc_atanh(t.get_mpc_t(), t.get_mpc_t(), MPFR_RNDN);
        return complex_mpc(std::move(t));
    }
    virtual RCP<const Basic> asech(const Basic &x) const
    {
        SYMENGINE_ASSERT(is_a<ComplexMPC>(x))
        mpc_class t(down_cast<const ComplexMPC &>(x).as_mpc().get_prec());
        mpc_ui_div(t.get_mpc_t(), 1,
                   down_cast<const ComplexMPC &>(x).as_mpc().get_mpc_t(),
                   MPFR_RNDN);
        mpc_acosh(t.get_mpc_t(), t.get_mpc_t(), MPFR_RNDN);
        return complex_mpc(std::move(t));
    }
    virtual RCP<const Basic> log(const Basic &x) const
    {
        SYMENGINE_ASSERT(is_a<ComplexMPC>(x))
        mpc_class t(down_cast<const ComplexMPC &>(x).as_mpc().get_prec());
        mpc_log(t.get_mpc_t(),
                down_cast<const ComplexMPC &>(x).as_mpc().get_mpc_t(),
                MPFR_RNDN);
        return complex_mpc(std::move(t));
    }
    virtual RCP<const Basic> abs(const Basic &x) const
    {
        SYMENGINE_ASSERT(is_a<ComplexMPC>(x))
        mpfr_class t(down_cast<const ComplexMPC &>(x).as_mpc().get_prec());
        mpc_abs(t.get_mpfr_t(),
                down_cast<const ComplexMPC &>(x).as_mpc().get_mpc_t(),
                MPFR_RNDN);
        return real_mpfr(std::move(t));
    }
    virtual RCP<const Basic> gamma(Basic const &aConst) const
    {
        throw NotImplementedError("Not Implemented.");
    }
    virtual RCP<const Basic> exp(const Basic &x) const override
    {
        SYMENGINE_ASSERT(is_a<ComplexMPC>(x))
        mpc_class t(down_cast<const ComplexMPC &>(x).as_mpc().get_prec());
        mpc_exp(t.get_mpc_t(),
                down_cast<const ComplexMPC &>(x).as_mpc().get_mpc_t(),
                MPFR_RNDN);
        return complex_mpc(std::move(t));
    }
    virtual RCP<const Basic> floor(const Basic &x) const override
    {
        SYMENGINE_ASSERT(is_a<ComplexMPC>(x))
        integer_class re, im;
        mpfr_get_z(
            get_mpz_t(re),
            mpc_realref(down_cast<const ComplexMPC &>(x).as_mpc().get_mpc_t()),
            MPFR_RNDD);
        mpfr_get_z(
            get_mpz_t(im),
            mpc_imagref(down_cast<const ComplexMPC &>(x).as_mpc().get_mpc_t()),
            MPFR_RNDD);
        mp_demote(re);
        mp_demote(im);
        return Complex::from_two_nums(*integer(std::move(re)),
                                      *integer(std::move(im)));
    }
    virtual RCP<const Basic> ceiling(const Basic &x) const override
    {
        SYMENGINE_ASSERT(is_a<ComplexMPC>(x))
        integer_class re, im;
        mpfr_get_z(
            get_mpz_t(re),
            mpc_realref(down_cast<const ComplexMPC &>(x).as_mpc().get_mpc_t()),
            MPFR_RNDU);
        mpfr_get_z(
            get_mpz_t(im),
            mpc_imagref(down_cast<const ComplexMPC &>(x).as_mpc().get_mpc_t()),
            MPFR_RNDU);
        mp_demote(re);
        mp_demote(im);
        return Complex::from_two_nums(*integer(std::move(re)),
                                      *integer(std::move(im)));
    }
    virtual RCP<const Basic> truncate(const Basic &x) const override
    {
        SYMENGINE_ASSERT(is_a<ComplexMPC>(x))
        integer_class re, im;
        mpfr_get_z(
            get_mpz_t(re),
            mpc_realref(down_cast<const ComplexMPC &>(x).as_mpc().get_mpc_t()),
            MPFR_RNDZ);
        mpfr_get_z(
            get_mpz_t(im),
            mpc_imagref(down_cast<const ComplexMPC &>(x).as_mpc().get_mpc_t()),
            MPFR_RNDZ);
        mp_demote(re);
        mp_demote(im);
        return Complex::from_two_nums(*integer(std::move(re)),
                                      *integer(std::move(im)));
    }
    virtual RCP<const Basic> erf(const Basic &x) const override
    {
        SYMENGINE_ASSERT(is_a<ComplexMPC>(x))
        throw NotImplementedError("erf is not implemented in mpc");
    }
    virtual RCP<const Basic> erfc(const Basic &x) const override
    {
        SYMENGINE_ASSERT(is_a<ComplexMPC>(x))
        throw NotImplementedError("erfc is not implemented in mpc");
    }
};

Evaluate &ComplexMPC::get_eval() const
{
    static EvaluateMPC evaluate_mpc;
    return evaluate_mpc;
}

} // SymEngine
#endif // HAVE_SYMENGINE_MPC