//This file is part of Bertini 2. // //mpfr_extensions.hpp is free software: you can redistribute it and/or modify //it under the terms of the GNU General Public License as published by //the Free Software Foundation, either version 3 of the License, or //(at your option) any later version. // //mpfr_extensions.hpp is distributed in the hope that it will be useful, //but WITHOUT ANY WARRANTY; without even the implied warranty of //MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the //GNU General Public License for more details. // //You should have received a copy of the GNU General Public License //along with mpfr_extensions.hpp. If not, see . // // Copyright(C) 2015 - 2021 by Bertini2 Development Team // // See for a copy of the license, // as well as COPYING. Bertini2 is provided with permitted // additional terms in the b2/licenses/ directory. // individual authors of this file include: // silviana amethyst, university of wisconsin - eau claire /** \file mpfr_extensions.hpp \brief Extensions to the Boost.Multiprecision library. Particularly includes Boost.Serialize code for the mpfr_float, gmp_rational, and gmp_int types. */ #ifndef BERTINI_MPFR_EXTENSIONS_HPP #define BERTINI_MPFR_EXTENSIONS_HPP #include #ifdef B2_FORBID_MIXED_ARITHMETIC #include "bertini2/forbid_mixed_arithmetic.hpp" #endif #include #include #include #include #include #include "bertini2/double_extensions.hpp" namespace bertini{ #ifdef BMP_EXPRESSION_TEMPLATES using mpfr_float = boost::multiprecision::number, boost::multiprecision::et_on>; using mpz_int = boost::multiprecision::number; using mpq_rational = boost::multiprecision::number; #else using mpfr_float = boost::multiprecision::number, boost::multiprecision::et_off>; using mpz_int = boost::multiprecision::number; using mpq_rational = boost::multiprecision::number; #endif /** \brief Get the precision of a real number. For mpfr_floats, this calls the precision member method for mpfr_float. */ inline auto Precision(mpfr_float const& num) { return num.precision(); } /** \brief Change the precision of a real number. For mpfr_floats, this calls the precision member method for mpfr_float. */ inline void Precision(mpfr_float & num, unsigned prec) { num.precision(prec); } } // overloads of the `max` function. namespace bertini{ /** \brief Three-argument form of `max`. \param a Input one \param b Input two \param c Input three */ template T max(T const& a, T const& b, T const& c) { using std::max; return max(max(a,b),c); } /** \brief Four-argument form of `max`. \param a Input one \param b Input two \param c Input three \param d Input four */ template T max(T const& a, T const& b, T const& c, T const& d) { using std::max; using bertini::max; return max(max(a,b,c),d); } /** \brief Five-argument form of `max`. \param a Input one \param b Input two \param c Input three \param d Input four \param e Input five */ template T max(T const& a, T const& b, T const& c, T const& d, T const& e) { using std::max; using bertini::max; return max(max(a,b,c,d),e); } } // the following code block extends serialization to the mpfr_float class from boost::multiprecision namespace boost { namespace serialization { /** Save a mpfr_float type to a boost archive. */ template void save(Archive& ar, ::boost::multiprecision::backends::mpfr_float_backend<0> const& r, unsigned /*version*/) { unsigned num_digits(r.precision()); ar & num_digits; std::string tmp = r.str(0,std::ios::scientific); ar & tmp; } /** Load a mpfr_float type from a boost archive. */ template void load(Archive& ar, ::boost::multiprecision::backends::mpfr_float_backend<0>& r, unsigned /*version*/) { unsigned num_digits; ar & num_digits; r.precision(num_digits); std::string tmp; ar & tmp; r = tmp.c_str(); } /** Save a gmp_rational type to a boost archive. */ template void save(Archive& ar, ::boost::multiprecision::backends::gmp_rational const& r, unsigned /*version*/) { std::string tmp = r.str(0,std::ios::scientific); ar & tmp; } /** Load a gmp_rational type from a boost archive. */ template void load(Archive& ar, ::boost::multiprecision::backends::gmp_rational& r, unsigned /*version*/) { std::string tmp; ar & tmp; r = tmp.c_str(); } /** Save a gmp_int type to a boost archive. */ template void save(Archive& ar, ::boost::multiprecision::backends::gmp_int const& r, unsigned /*version*/) { std::string tmp = r.str(0,std::ios::scientific); ar & tmp; } /** Load a gmp_int type from a boost archive. */ template void load(Archive& ar, ::boost::multiprecision::backends::gmp_int& r, unsigned /*version*/) { std::string tmp; ar & tmp; r = tmp.c_str(); } } } // re: namespaces BOOST_SERIALIZATION_SPLIT_FREE(::boost::multiprecision::backends::mpfr_float_backend<0>) BOOST_SERIALIZATION_SPLIT_FREE(::boost::multiprecision::backends::gmp_rational) BOOST_SERIALIZATION_SPLIT_FREE(::boost::multiprecision::backends::gmp_int) // if you wish to use et_on with Boost.Multiprecision with Eigen 3.2.x or earlier, you must apply a patch to Boost.MP related to bug 11149, and use the following non-standard lines. // https://svn.boost.org/trac/boost/ticket/11149 #define EIGEN_DEVICE_FUNC // to make Eigen 3.3 happy... ugh, this is likely to break CUDA usage with Bertini2, if that ever happens. #include #ifdef BMP_EXPRESSION_TEMPLATES #if (!EIGEN_VERSION_AT_LEAST(3,2,92)) // version of 3.3-beta1 is 3,2,92. namespace std{ using boost::multiprecision::min; //error receiver: please see https://svn.boost.org/trac/boost/ticket/11149 for information about these using statements in std namespace. using boost::multiprecision::max; //3 options: ./configure --disable-expression_templates, use Boost 1.61, or patch earlier Boost versions to resolve this. } #endif #endif #endif