/* Copyright (C) 2013 Tom Bachmann This file is part of FLINT. FLINT is free software: you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License (LGPL) as published by the Free Software Foundation; either version 2.1 of the License, or (at your option) any later version. See . */ #ifndef CXX_FMPZXX_H #define CXX_FMPZXX_H #include #include #include "flintxx/evaluation_tools.h" #include "flintxx/expression.h" #include "flintxx/expression_traits.h" #include "flintxx/flint_classes.h" #include "flintxx/frandxx.h" #include "flintxx/ltuple.h" #include "flintxx/stdmath.h" #include "fmpz.h" #undef clrbit /* no idea where these are coming from */ #undef setbit // TODO functions for addmul? inhomogeneous addmul? // TODO use evaluate_n in immediate functions? // TODO fmpz_abs_ubound_ui_2exp, lbound // TODO cmpabs // TODO xgcd_partial namespace flint { FLINT_DEFINE_BINOP(cdiv_q) FLINT_DEFINE_BINOP(fdiv_r) FLINT_DEFINE_BINOP(tdiv_q) FLINT_DEFINE_BINOP(fdiv_r_2exp) FLINT_DEFINE_BINOP(tdiv_q_2exp) FLINT_DEFINE_UNOP(fac) FLINT_DEFINE_UNOP(fib) FLINT_DEFINE_BINOP(rfac) FLINT_DEFINE_BINOP(bin) FLINT_DEFINE_BINOP(gcd) FLINT_DEFINE_BINOP(lcm) FLINT_DEFINE_BINOP(invmod) FLINT_DEFINE_BINOP(negmod) FLINT_DEFINE_THREEARY(mul2) FLINT_DEFINE_THREEARY(divexact2) FLINT_DEFINE_THREEARY(powm) FLINT_DEFINE_THREEARY(mul_tdiv_q_2exp) FLINT_DEFINE_BINOP(fdiv_qr) FLINT_DEFINE_BINOP(tdiv_qr) FLINT_DEFINE_BINOP(sqrtmod) FLINT_DEFINE_UNOP(sqrtrem) FLINT_DEFINE_BINOP(gcdinv) FLINT_DEFINE_BINOP(xgcd) FLINT_DEFINE_BINOP(remove) FLINT_DEFINE_FIVEARY(fmpzxx_bit_unpack) FLINT_DEFINE_THREEARY(fmpzxx_bit_unpack_unsigned) namespace mp { template struct enable_all_fmpzxx; } template class fmpzxx_expression : public expression, Operation, Data> { public: typedef expression, Operation, Data> base_t; FLINTXX_DEFINE_BASICS(fmpzxx_expression) FLINTXX_DEFINE_CTORS(fmpzxx_expression) FLINTXX_DEFINE_C_REF(fmpzxx_expression, fmpz, _fmpz) // these only make sense with fmpzxx FLINTXX_DEFINE_RANDFUNC(fmpz, randbits) FLINTXX_DEFINE_RANDFUNC(fmpz, randtest) FLINTXX_DEFINE_RANDFUNC(fmpz, randtest_unsigned) FLINTXX_DEFINE_RANDFUNC(fmpz, randtest_not_zero) template static fmpzxx_expression randm(frandxx& state, const Fmpz& m) { fmpzxx_expression res; fmpz_randm(res._fmpz(), state._data(), m.evaluate()._fmpz()); return res; } template static fmpzxx_expression randtest_mod(frandxx& state, const Fmpz& m) { fmpzxx_expression res; fmpz_randtest_mod(res._fmpz(), state._data(), m.evaluate()._fmpz()); return res; } template static fmpzxx_expression randtest_mod_signed(frandxx& state, const Fmpz& m) { fmpzxx_expression res; fmpz_randtest_mod_signed(res._fmpz(), state._data(), m.evaluate()._fmpz()); return res; } // TODO would these make more sense static? void set_ui_smod(mp_limb_t x, mp_limb_t m) { fmpz_set_ui_smod(this->_fmpz(), x, m); } void set_uiui(mp_limb_t hi, mp_limb_t lo) { fmpz_set_uiui(this->_fmpz(), hi, lo); } void neg_uiui(mp_limb_t hi, mp_limb_t lo) { fmpz_neg_uiui(this->_fmpz(), hi, lo); } // these only make sense with fmpzxx/fmpzxx_ref void clrbit(ulong i) {fmpz_clrbit(_fmpz(), i);} void combit(ulong i) {fmpz_combit(_fmpz(), i);} void setbit(ulong i) {fmpz_setbit(_fmpz(), i);} void set_zero() {fmpz_zero(_fmpz());} void set_one() {fmpz_one(_fmpz());} // These make sense with all expressions, but cause evaluation double get_d_2exp(slong& exp) const { return fmpz_get_d_2exp(&exp, this->evaluate()._fmpz()); } bool is_zero() const { return fmpz_is_zero(this->evaluate()._fmpz()); } bool is_one() const { return fmpz_is_one(this->evaluate()._fmpz()); } bool is_pm1() const { return fmpz_is_pm1(this->evaluate()._fmpz()); } bool is_even() const { return fmpz_is_even(this->evaluate()._fmpz()); } bool is_odd() const { return fmpz_is_odd(this->evaluate()._fmpz()); } bool is_square() const { return fmpz_is_square(this->evaluate()._fmpz()); } int popcnt() const { return fmpz_popcnt(this->evaluate()._fmpz()); } bool is_probabprime() const { return fmpz_is_probabprime(this->evaluate()._fmpz()); } bool is_prime_pseudosquare() const { return fmpz_is_prime_pseudosquare(this->evaluate()._fmpz()); } bool abs_fits_ui() const { return fmpz_abs_fits_ui(this->evaluate()._fmpz()); } bool fits_si() const { return fmpz_fits_si(this->evaluate()._fmpz()); } bool tstbit(ulong i) const { return fmpz_tstbit(this->evaluate()._fmpz(), i); } template typename mp::enable_all_fmpzxx::type divisible(const T2& t2) const { return fmpz_divisible(this->evaluate()._fmpz(), t2.evaluate()._fmpz()); } template typename mp::enable_if, bool>::type divisible(const T2& t2) const { return fmpz_divisible_si(this->evaluate()._fmpz(), t2); } template typename mp::enable_all_fmpzxx::type clog(const Fmpz2& b) const { return fmpz_clog(this->evaluate()._fmpz(), b.evaluate()._fmpz()); } template typename mp::enable_if, slong>::type clog(Int b) const { return fmpz_clog_ui(this->evaluate()._fmpz(), b); } template typename mp::enable_all_fmpzxx::type flog(const Fmpz2& b) const { return fmpz_flog(this->evaluate()._fmpz(), b.evaluate()._fmpz()); } template typename mp::enable_if, slong>::type flog(Int b) const { return fmpz_flog_ui(this->evaluate()._fmpz(), b); } double dlog() const { return fmpz_dlog(this->evaluate()._fmpz()); } template typename mp::enable_all_fmpzxx::type jacobi(const Fmpz2& p) const { return fmpz_jacobi(this->evaluate()._fmpz(), p.evaluate()._fmpz()); } size_t sizeinbase(int b) const {return fmpz_sizeinbase(this->evaluate()._fmpz(), b);} flint_bitcnt_t bits() const {return fmpz_bits(this->evaluate()._fmpz());} flint_bitcnt_t size() const {return fmpz_size(this->evaluate()._fmpz());} flint_bitcnt_t val2() const {return fmpz_val2(this->evaluate()._fmpz());} int sgn() const {return fmpz_sgn(this->evaluate()._fmpz());} // lazy function forwarding FLINTXX_DEFINE_MEMBER_3OP(divexact2) FLINTXX_DEFINE_MEMBER_3OP(mul2) FLINTXX_DEFINE_MEMBER_3OP(mul_tdiv_q_2exp) FLINTXX_DEFINE_MEMBER_3OP(powm) FLINTXX_DEFINE_MEMBER_BINOP(cdiv_q) FLINTXX_DEFINE_MEMBER_BINOP(divexact) FLINTXX_DEFINE_MEMBER_BINOP(fdiv_qr) FLINTXX_DEFINE_MEMBER_BINOP(fdiv_r) FLINTXX_DEFINE_MEMBER_BINOP(fdiv_r_2exp) FLINTXX_DEFINE_MEMBER_BINOP(gcd) FLINTXX_DEFINE_MEMBER_BINOP(gcdinv) FLINTXX_DEFINE_MEMBER_BINOP(invmod) FLINTXX_DEFINE_MEMBER_BINOP(lcm) FLINTXX_DEFINE_MEMBER_BINOP(negmod) FLINTXX_DEFINE_MEMBER_BINOP(pow) FLINTXX_DEFINE_MEMBER_BINOP(remove) FLINTXX_DEFINE_MEMBER_BINOP(rfac) FLINTXX_DEFINE_MEMBER_BINOP(root) FLINTXX_DEFINE_MEMBER_BINOP(sqrtmod) FLINTXX_DEFINE_MEMBER_BINOP(tdiv_q) FLINTXX_DEFINE_MEMBER_BINOP(tdiv_q_2exp) FLINTXX_DEFINE_MEMBER_BINOP(tdiv_qr) FLINTXX_DEFINE_MEMBER_BINOP(xgcd) FLINTXX_DEFINE_MEMBER_UNOP(abs) FLINTXX_DEFINE_MEMBER_UNOP(sqrt) // FLINTXX_DEFINE_MEMBER_UNOP(sqrtrem) // TODO FLINTXX_DEFINE_MEMBER_5OP(CRT) template static FLINT_FIVEARY_ENABLE_RETTYPE(fmpzxx_bit_unpack, Arg1, Arg2, flint_bitcnt_t, int, bool) bit_unpack(const Arg1& arr, const Arg2& bits, flint_bitcnt_t shift = 0, int negate = 0, bool borrow = false) { return fmpzxx_bit_unpack(arr, bits, shift, negate, borrow); } template static FLINT_THREEARY_ENABLE_RETTYPE(fmpzxx_bit_unpack_unsigned, Arg1, Arg2, flint_bitcnt_t) bit_unpack_unsigned(const Arg1& arr, const Arg2& bits, flint_bitcnt_t shift = 0) { return fmpzxx_bit_unpack_unsigned(arr, bits, shift); } }; namespace detail { struct fmpz_data; } typedef fmpzxx_expression fmpzxx; typedef fmpzxx_expression > fmpzxx_ref; typedef fmpzxx_expression > fmpzxx_srcref; namespace detail { struct fmpz_data { typedef fmpz_t& data_ref_t; typedef const fmpz_t& data_srcref_t; fmpz_t inner; fmpz_data() {fmpz_init(inner);} ~fmpz_data() {fmpz_clear(inner);} fmpz_data(const fmpz_data& o) {fmpz_init_set(inner, o.inner);} template fmpz_data(const T& t) { init(t); } template typename mp::enable_if >::type init(T t) { fmpz_init_set_ui(inner, t); } template typename mp::enable_if >::type init(T t) { fmpz_init(inner); fmpz_set_si(inner, t); } void init(double d) { fmpz_init(inner); fmpz_set_d(inner, d); } void init(const char* str) { fmpz_init(inner); fmpz_set_str(inner, str, 10); } void init(const fmpzxx_srcref& r) { fmpz_init_set(inner, r._fmpz()); } }; } // detail /////////////////////////////////////////////////////////////////// // HELPERS /////////////////////////////////////////////////////////////////// namespace traits { template struct is_fmpzxx : mp::or_< traits::is_T_expr, flint_classes::is_source > { }; } // traits namespace mp { template struct all_fmpzxx : mp::and_, all_fmpzxx > { }; template struct all_fmpzxx : traits::is_fmpzxx { }; template struct enable_all_fmpzxx : mp::enable_if, Out> { }; } // mp /////////////////////////////////////////////////////////////////// // RULES /////////////////////////////////////////////////////////////////// namespace rules { #define FMPZXX_COND_S FLINTXX_COND_S(fmpzxx) #define FMPZXX_COND_T FLINTXX_COND_T(fmpzxx) FLINT_DEFINE_DOIT_COND2(assignment, FMPZXX_COND_T, FMPZXX_COND_S, fmpz_set(to._fmpz(), from._fmpz())) FLINT_DEFINE_DOIT_COND2(assignment, FMPZXX_COND_T, traits::is_unsigned_integer, fmpz_set_ui(to._fmpz(), from)) FLINT_DEFINE_DOIT_COND2(assignment, FMPZXX_COND_T, traits::is_signed_integer, fmpz_set_si(to._fmpz(), from)) FLINTXX_DEFINE_CMP(fmpzxx, fmpz_cmp(e1._fmpz(), e2._fmpz())) template struct cmp, traits::is_signed_integer > >::type> { static int get(const T& v, const U& t) { return fmpz_cmp_si(v._fmpz(), t); } }; template struct cmp >::type> { static int get(const fmpzxx& v, const T& t) { return fmpz_cmp_ui(v._fmpz(), t); } }; FLINTXX_DEFINE_ASSIGN_STR(fmpzxx, fmpz_set_str(to._fmpz(), from, 10)) FLINTXX_DEFINE_TO_STR(fmpzxx, fmpz_get_str(0, base, from._fmpz())) FLINTXX_DEFINE_SWAP(fmpzxx, fmpz_swap(e1._fmpz(), e2._fmpz())) FLINT_DEFINE_PRINT_COND(FMPZXX_COND_S, fmpz_fprint(to, from._fmpz())) FLINT_DEFINE_READ_COND(FMPZXX_COND_T, fmpz_fread(from, to._fmpz())) FLINT_DEFINE_GET_COND(conversion, slong, FMPZXX_COND_S, fmpz_get_si(from._fmpz())) FLINT_DEFINE_GET_COND(conversion, ulong, FMPZXX_COND_S, fmpz_get_ui(from._fmpz())) FLINT_DEFINE_GET_COND(conversion, double, FMPZXX_COND_S, fmpz_get_d(from._fmpz())) FLINT_DEFINE_BINARY_EXPR_COND2(plus, fmpzxx, FMPZXX_COND_S, FMPZXX_COND_S, fmpz_add(to._fmpz(), e1._fmpz(), e2._fmpz())) FLINT_DEFINE_CBINARY_EXPR_COND2(plus, fmpzxx, FMPZXX_COND_S, traits::is_unsigned_integer, fmpz_add_ui(to._fmpz(), e1._fmpz(), e2)) FLINT_DEFINE_CBINARY_EXPR_COND2(plus, fmpzxx, FMPZXX_COND_S, traits::is_signed_integer, fmpz_add_si(to._fmpz(), e1._fmpz(), e2)) FLINT_DEFINE_CBINARY_EXPR_COND2(times, fmpzxx, FMPZXX_COND_S, FMPZXX_COND_S, fmpz_mul(to._fmpz(), e1._fmpz(), e2._fmpz())) FLINT_DEFINE_CBINARY_EXPR_COND2(times, fmpzxx, FMPZXX_COND_S, traits::is_unsigned_integer, fmpz_mul_ui(to._fmpz(), e1._fmpz(), e2)) FLINT_DEFINE_CBINARY_EXPR_COND2(times, fmpzxx, FMPZXX_COND_S, traits::is_signed_integer, fmpz_mul_si(to._fmpz(), e1._fmpz(), e2)) FLINT_DEFINE_BINARY_EXPR_COND2(minus, fmpzxx, FMPZXX_COND_S, FMPZXX_COND_S, fmpz_sub(to._fmpz(), e1._fmpz(), e2._fmpz())) FLINT_DEFINE_BINARY_EXPR_COND2(minus, fmpzxx, FMPZXX_COND_S, traits::is_unsigned_integer, fmpz_sub_ui(to._fmpz(), e1._fmpz(), e2)) FLINT_DEFINE_BINARY_EXPR_COND2(minus, fmpzxx, FMPZXX_COND_S, traits::is_signed_integer, fmpz_sub_si(to._fmpz(), e1._fmpz(), e2)) FLINT_DEFINE_BINARY_EXPR_COND2(divided_by, fmpzxx, FMPZXX_COND_S, FMPZXX_COND_S, fmpz_fdiv_q(to._fmpz(), e1._fmpz(), e2._fmpz())) FLINT_DEFINE_BINARY_EXPR_COND2(divided_by, fmpzxx, FMPZXX_COND_S, traits::is_unsigned_integer, fmpz_fdiv_q_ui(to._fmpz(), e1._fmpz(), e2)) FLINT_DEFINE_BINARY_EXPR_COND2(divided_by, fmpzxx, FMPZXX_COND_S, traits::is_signed_integer, fmpz_fdiv_q_si(to._fmpz(), e1._fmpz(), e2)) // TODO this interpretation of mod is not the same as for builtin types! FLINT_DEFINE_BINARY_EXPR_COND2(modulo, fmpzxx, FMPZXX_COND_S, FMPZXX_COND_S, fmpz_mod(to._fmpz(), e1._fmpz(), e2._fmpz())) FLINT_DEFINE_BINARY_EXPR_COND2(modulo, fmpzxx, FMPZXX_COND_S, traits::is_unsigned_integer, fmpz_mod_ui(to._fmpz(), e1._fmpz(), e2)) FLINT_DEFINE_BINARY_EXPR_COND2(binary_and, fmpzxx, FMPZXX_COND_S, FMPZXX_COND_S, fmpz_and(to._fmpz(), e1._fmpz(), e2._fmpz())) FLINT_DEFINE_BINARY_EXPR_COND2(binary_or, fmpzxx, FMPZXX_COND_S, FMPZXX_COND_S, fmpz_or(to._fmpz(), e1._fmpz(), e2._fmpz())) FLINT_DEFINE_BINARY_EXPR_COND2(binary_xor, fmpzxx, FMPZXX_COND_S, FMPZXX_COND_S, fmpz_xor(to._fmpz(), e1._fmpz(), e2._fmpz())) FLINT_DEFINE_UNARY_EXPR_COND(negate, fmpzxx, FMPZXX_COND_S, fmpz_neg(to._fmpz(), from._fmpz())) FLINT_DEFINE_UNARY_EXPR_COND(complement, fmpzxx, FMPZXX_COND_S, fmpz_complement(to._fmpz(), from._fmpz())) namespace rdetail { template void fmpzxx_shift(Fmpz1& to, const Fmpz2& from, T howmuch) { if(howmuch < 0) fmpz_fdiv_q_2exp(to._fmpz(), from._fmpz(), -howmuch); else fmpz_mul_2exp(to._fmpz(), from._fmpz(), howmuch); } } // rdetail FLINT_DEFINE_BINARY_EXPR_COND2(shift, fmpzxx, FMPZXX_COND_S, traits::is_integer, rdetail::fmpzxx_shift(to, e1, e2)) } // rules FLINTXX_DEFINE_TERNARY(fmpzxx, fmpz_addmul(to._fmpz(), e1._fmpz(), e2._fmpz()), fmpz_submul(to._fmpz(), e1._fmpz(), e2._fmpz()), FLINTXX_UNADORNED_MAKETYPES) /////////////////////////////////////////////////////////////////////////// // FUNCTIONS /////////////////////////////////////////////////////////////////////////// // These functions evaluate immediately, and (often) do not yield fmpzxxs template inline typename mp::enable_all_fmpzxx::type divisible(const T1& t1, const T2& t2) { return t1.divisible(t2); } template inline typename mp::enable_all_fmpzxx::type clog(const Fmpz1& x, const Fmpz2& b) { return x.clog(b); } template inline typename mp::enable_all_fmpzxx::type flog(const Fmpz1& x, const Fmpz2& b) { return x.flog(b); } template inline typename mp::enable_if, double>::type dlog(const Fmpz& x) { return x.dlog(); } template inline typename mp::enable_all_fmpzxx::type jacobi(const Fmpz1& a, const Fmpz2& p) { return a.jacobi(p); } template inline typename mp::enable_if, size_t>::type sizeinbase(const Fmpz& a, int b) { return a.sizeinbase(b); } template inline typename mp::enable_if, flint_bitcnt_t>::type bits(const Fmpz& a) { return a.bits(); } template inline typename mp::enable_if, flint_bitcnt_t>::type val2(const Fmpz& a) { return a.val2(); } template inline typename mp::enable_if, flint_bitcnt_t>::type size(const Fmpz& a) { return a.size(); } template inline typename mp::enable_if, int>::type sgn(const Fmpz& a) { return a.sgn(); } template inline bool bit_pack(std::vector& arr, flint_bitcnt_t bits, const Fmpz& coeff, flint_bitcnt_t shift = 0, int negate = 0, bool borrow = false, typename mp::enable_if >::type* = 0) { return fmpz_bit_pack(&arr.front(), shift, bits, coeff.evaluate()._fmpz(), negate, borrow); } // These functions are evaluated lazily namespace rules { FLINT_DEFINE_BINARY_EXPR_COND2(rfac_op, fmpzxx, FMPZXX_COND_S, traits::is_unsigned_integer, fmpz_rfac_ui(to._fmpz(), e1._fmpz(), e2)) FLINT_DEFINE_UNARY_EXPR_COND(fac_op, fmpzxx, traits::is_unsigned_integer, fmpz_fac_ui(to._fmpz(), from)) FLINT_DEFINE_UNARY_EXPR_COND(fib_op, fmpzxx, traits::is_unsigned_integer, fmpz_fib_ui(to._fmpz(), from)) FLINT_DEFINE_BINARY_EXPR_COND2(gcd_op, fmpzxx, FMPZXX_COND_S, FMPZXX_COND_S, fmpz_gcd(to._fmpz(), e1._fmpz(), e2._fmpz())) FLINT_DEFINE_BINARY_EXPR_COND2(lcm_op, fmpzxx, FMPZXX_COND_S, FMPZXX_COND_S, fmpz_lcm(to._fmpz(), e1._fmpz(), e2._fmpz())) template struct binary_expression< T1, typename mp::enable_if< mp::and_< traits::is_unsigned_integer, traits::is_unsigned_integer >, operations::bin_op>::type, T2> { typedef fmpzxx return_t; template static void doit(V& to, const T1& t1, const T2& t2) { fmpz_bin_uiui(to._fmpz(), t1, t2); } }; #define FMPZXX_DEFINE_DIVFUNCS(name) \ FLINT_DEFINE_BINARY_EXPR_COND2(name##_op, fmpzxx, FMPZXX_COND_S, FMPZXX_COND_S, \ fmpz_##name(to._fmpz(), e1._fmpz(), e2._fmpz())) \ FLINT_DEFINE_BINARY_EXPR_COND2(name##_op, fmpzxx, FMPZXX_COND_S, \ traits::is_signed_integer, \ fmpz_##name##_si(to._fmpz(), e1._fmpz(), e2)) \ FLINT_DEFINE_BINARY_EXPR_COND2(name##_op, fmpzxx, FMPZXX_COND_S, \ traits::is_unsigned_integer, \ fmpz_##name##_ui(to._fmpz(), e1._fmpz(), e2)) FMPZXX_DEFINE_DIVFUNCS(cdiv_q) FMPZXX_DEFINE_DIVFUNCS(tdiv_q) FMPZXX_DEFINE_DIVFUNCS(divexact) FLINT_DEFINE_BINARY_EXPR_COND2(fdiv_r_op, fmpzxx, FMPZXX_COND_S, FMPZXX_COND_S, fmpz_fdiv_r(to._fmpz(), e1._fmpz(), e2._fmpz())) FLINT_DEFINE_BINARY_EXPR_COND2(tdiv_q_2exp_op, fmpzxx, FMPZXX_COND_S, traits::is_unsigned_integer, fmpz_tdiv_q_2exp(to._fmpz(), e1._fmpz(), e2)) FLINT_DEFINE_BINARY_EXPR_COND2(fdiv_r_2exp_op, fmpzxx, FMPZXX_COND_S, traits::is_unsigned_integer, fmpz_fdiv_r_2exp(to._fmpz(), e1._fmpz(), e2)) FLINT_DEFINE_BINARY_EXPR_COND2(invmod_op, fmpzxx, FMPZXX_COND_S, FMPZXX_COND_S, fmpz_invmod(to._fmpz(), e1._fmpz(), e2._fmpz())) FLINT_DEFINE_BINARY_EXPR_COND2(negmod_op, fmpzxx, FMPZXX_COND_S, FMPZXX_COND_S, fmpz_negmod(to._fmpz(), e1._fmpz(), e2._fmpz())) FLINT_DEFINE_THREEARY_EXPR_COND3(mul2_op, fmpzxx, FMPZXX_COND_S, traits::is_unsigned_integer, traits::is_unsigned_integer, fmpz_mul2_uiui(to._fmpz(), e1._fmpz(), e2, e3)) FLINT_DEFINE_THREEARY_EXPR_COND3(divexact2_op, fmpzxx, FMPZXX_COND_S, traits::is_unsigned_integer, traits::is_unsigned_integer, fmpz_divexact2_uiui(to._fmpz(), e1._fmpz(), e2, e3)) FLINT_DEFINE_THREEARY_EXPR_COND3(powm_op, fmpzxx, FMPZXX_COND_S, traits::is_unsigned_integer, FMPZXX_COND_S, fmpz_powm_ui(to._fmpz(), e1._fmpz(), e2, e3._fmpz())) FLINT_DEFINE_THREEARY_EXPR_COND3(powm_op, fmpzxx, FMPZXX_COND_S, FMPZXX_COND_S, FMPZXX_COND_S, fmpz_powm(to._fmpz(), e1._fmpz(), e2._fmpz(), e3._fmpz())) FLINT_DEFINE_THREEARY_EXPR_COND3(mul_tdiv_q_2exp_op, fmpzxx, FMPZXX_COND_S, FMPZXX_COND_S, traits::is_unsigned_integer, fmpz_mul_tdiv_q_2exp(to._fmpz(), e1._fmpz(), e2._fmpz(), e3)) FLINT_DEFINE_THREEARY_EXPR_COND3(mul_tdiv_q_2exp_op, fmpzxx, FMPZXX_COND_S, traits::fits_into_slong, traits::is_unsigned_integer, fmpz_mul_si_tdiv_q_2exp(to._fmpz(), e1._fmpz(), e2, e3)) // TODO addmul, submul? namespace rdetail { typedef make_ltuple::type>::type fmpzxx_pair; typedef make_ltuple::type>::type fmpzxx_triple; typedef make_ltuple::type>::type bool_fmpzxx_pair; } // rdetail FLINT_DEFINE_BINARY_EXPR_COND2(fdiv_qr_op, rdetail::fmpzxx_pair, FMPZXX_COND_S, FMPZXX_COND_S, fmpz_fdiv_qr(to.template get<0>()._fmpz(), to.template get<1>()._fmpz(), e1._fmpz(), e2._fmpz())) FLINT_DEFINE_BINARY_EXPR_COND2(tdiv_qr_op, rdetail::fmpzxx_pair, FMPZXX_COND_S, FMPZXX_COND_S, fmpz_tdiv_qr(to.template get<0>()._fmpz(), to.template get<1>()._fmpz(), e1._fmpz(), e2._fmpz())) FLINT_DEFINE_BINARY_EXPR_COND2(sqrtmod_op, rdetail::bool_fmpzxx_pair, FMPZXX_COND_S, FMPZXX_COND_S, to.template get<0>() = fmpz_sqrtmod( to.template get<1>()._fmpz(), e1._fmpz(), e2._fmpz())) FLINT_DEFINE_UNARY_EXPR_COND(sqrtrem_op, rdetail::fmpzxx_pair, FMPZXX_COND_S, fmpz_sqrtrem(to.template get<0>()._fmpz(), to.template get<1>()._fmpz(), from._fmpz())) FLINT_DEFINE_BINARY_EXPR_COND2(gcdinv_op, rdetail::fmpzxx_pair, FMPZXX_COND_S, FMPZXX_COND_S, fmpz_gcdinv(to.template get<0>()._fmpz(), to.template get<1>()._fmpz(), e1._fmpz(), e2._fmpz())) FLINT_DEFINE_BINARY_EXPR_COND2(xgcd_op, rdetail::fmpzxx_triple, FMPZXX_COND_S, FMPZXX_COND_S, fmpz_xgcd( to.template get<0>()._fmpz(), to.template get<1>()._fmpz(), to.template get<2>()._fmpz(), e1._fmpz(), e2._fmpz())) namespace rdetail { template struct is_mplimb_t_vec : mp::equal_types > { }; } FLINT_DEFINE_FIVEARY_EXPR_COND5(fmpzxx_bit_unpack_op, rdetail::bool_fmpzxx_pair, rdetail::is_mplimb_t_vec, traits::fits_into_flint_bitcnt_t, traits::fits_into_flint_bitcnt_t, traits::is_integer, tools::is_bool, to.template get<0>() = fmpz_bit_unpack(to.template get<1>()._fmpz(), &e1.front(), e3, e2, e4, e5)) FLINT_DEFINE_THREEARY_EXPR_COND3(fmpzxx_bit_unpack_unsigned_op, fmpzxx, rdetail::is_mplimb_t_vec, traits::fits_into_flint_bitcnt_t, traits::fits_into_flint_bitcnt_t, fmpz_bit_unpack_unsigned(to._fmpz(), &e1.front(), e3, e2)) // standard math functions (c/f stdmath.h) FLINT_DEFINE_BINARY_EXPR_COND2(pow_op, fmpzxx, FMPZXX_COND_S, traits::is_unsigned_integer, fmpz_pow_ui(to._fmpz(), e1._fmpz(), e2)) FLINT_DEFINE_BINARY_EXPR_COND2(root_op, fmpzxx, FMPZXX_COND_S, traits::fits_into_slong, fmpz_root(to._fmpz(), e1._fmpz(), e2)) FLINT_DEFINE_UNARY_EXPR_COND(sqrt_op, fmpzxx, FMPZXX_COND_S, fmpz_sqrt(to._fmpz(), from._fmpz())) FLINT_DEFINE_UNARY_EXPR_COND(abs_op, fmpzxx, FMPZXX_COND_S, fmpz_abs(to._fmpz(), from._fmpz())) namespace rdetail { typedef make_ltuple::type>::type slong_fmpzxx_pair; } // rdetail FLINT_DEFINE_BINARY_EXPR_COND2(remove_op, rdetail::slong_fmpzxx_pair, FMPZXX_COND_S, FMPZXX_COND_S, to.template get<0>() = fmpz_remove(to.template get<1>()._fmpz(), e1._fmpz(), e2._fmpz())) } // rules // chinese remaindering // TODO should this use nmod? class fmpz_combxx { private: fmpz_comb_t comb; mutable fmpz_comb_temp_t tmp; // not copyable fmpz_combxx(const fmpz_combxx&); public: fmpz_combxx(const std::vector& v) { fmpz_comb_init(comb, &v.front(), v.size()); fmpz_comb_temp_init(tmp, comb); } ~fmpz_combxx() { fmpz_comb_temp_clear(tmp); fmpz_comb_clear(comb); } const fmpz_comb_t& _comb() const {return comb;} fmpz_comb_temp_t& _temp() const {return tmp;} }; // TODO make lazy somehow? template inline typename mp::enable_if >::type multi_mod(std::vector& out, const Fmpz& in, const fmpz_combxx& comb) { fmpz_multi_mod_ui(&out.front(), in.evaluate()._fmpz(), comb._comb(), comb._temp()); } namespace rules { FLINT_DEFINE_FIVEARY_EXPR_COND5(CRT_op, fmpzxx, FMPZXX_COND_S, FMPZXX_COND_S, traits::is_unsigned_integer, traits::is_unsigned_integer, tools::is_bool, fmpz_CRT_ui(to._fmpz(), e1._fmpz(), e2._fmpz(), e3, e4, e5)) FLINT_DEFINE_THREEARY_EXPR(multi_CRT_op, fmpzxx, std::vector, fmpz_combxx, bool, fmpz_multi_CRT_ui(to._fmpz(), &e1.front(), e2._comb(), e2._temp(), e3)) } // rules } // flint #include "fmpz_factorxx.h" #endif