| // Copyright John Maddock 2010. |
| // Use, modification and distribution are subject to the |
| // Boost Software License, Version 1.0. (See accompanying file |
| // LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) |
| |
| #ifdef _MSC_VER |
| # pragma once |
| #endif |
| |
| #ifndef BOOST_MATH_CONSTANTS_INFO_INCLUDED |
| #define BOOST_MATH_CONSTANTS_INFO_INCLUDED |
| |
| #include <boost/math/constants/constants.hpp> |
| #include <iostream> |
| #include <iomanip> |
| #ifndef BOOST_MATH_NO_RTTI |
| #include <typeinfo> |
| #endif |
| |
| namespace boost{ namespace math{ namespace constants{ |
| |
| namespace detail{ |
| |
| template <class T> |
| const char* nameof(BOOST_MATH_EXPLICIT_TEMPLATE_TYPE_SPEC(T)) |
| { |
| #ifndef BOOST_MATH_NO_RTTI |
| return typeid(T).name(); |
| #else |
| return "unknown"; |
| #endif |
| } |
| template <> |
| const char* nameof<float>(BOOST_MATH_EXPLICIT_TEMPLATE_TYPE_SPEC(float)) |
| { |
| return "float"; |
| } |
| template <> |
| const char* nameof<double>(BOOST_MATH_EXPLICIT_TEMPLATE_TYPE_SPEC(double)) |
| { |
| return "double"; |
| } |
| template <> |
| const char* nameof<long double>(BOOST_MATH_EXPLICIT_TEMPLATE_TYPE_SPEC(long double)) |
| { |
| return "long double"; |
| } |
| |
| } |
| |
| template <class T, class Policy> |
| void print_info_on_type(std::ostream& os = std::cout BOOST_MATH_APPEND_EXPLICIT_TEMPLATE_TYPE_SPEC(T) BOOST_MATH_APPEND_EXPLICIT_TEMPLATE_TYPE_SPEC(Policy)) |
| { |
| using detail::nameof; |
| #ifdef _MSC_VER |
| #pragma warning(push) |
| #pragma warning(disable:4127) |
| #endif |
| os << |
| "Information on the Implementation and Handling of \n" |
| "Mathematical Constants for Type " << nameof<T>() << |
| "\n\n" |
| "Checking for std::numeric_limits<" << nameof<T>() << "> specialisation: " << |
| (std::numeric_limits<T>::is_specialized ? "yes" : "no") << std::endl; |
| if(std::numeric_limits<T>::is_specialized) |
| { |
| os << |
| "std::numeric_limits<" << nameof<T>() << ">::digits reports that the radix is " << std::numeric_limits<T>::radix << ".\n"; |
| if (std::numeric_limits<T>::radix == 2) |
| { |
| os << |
| "std::numeric_limits<" << nameof<T>() << ">::digits reports that the precision is \n" << std::numeric_limits<T>::digits << " binary digits.\n"; |
| } |
| else if (std::numeric_limits<T>::radix == 10) |
| { |
| os << |
| "std::numeric_limits<" << nameof<T>() << ">::digits reports that the precision is \n" << std::numeric_limits<T>::digits10 << " decimal digits.\n"; |
| os << |
| "std::numeric_limits<" << nameof<T>() << ">::digits reports that the precision is \n" |
| << std::numeric_limits<T>::digits * 1000L /301L << " binary digits.\n"; // divide by log2(10) - about 3 bits per decimal digit. |
| } |
| else |
| { |
| os << "Unknown radix = " << std::numeric_limits<T>::radix << "\n"; |
| } |
| } |
| typedef typename boost::math::policies::precision<T, Policy>::type precision_type; |
| if(precision_type::value) |
| { |
| if (std::numeric_limits<T>::radix == 2) |
| { |
| os << |
| "boost::math::policies::precision<" << nameof<T>() << ", " << nameof<Policy>() << " reports that the compile time precision is \n" << precision_type::value << " binary digits.\n"; |
| } |
| else if (std::numeric_limits<T>::radix == 10) |
| { |
| os << |
| "boost::math::policies::precision<" << nameof<T>() << ", " << nameof<Policy>() << " reports that the compile time precision is \n" << precision_type::value << " binary digits.\n"; |
| } |
| else |
| { |
| os << "Unknown radix = " << std::numeric_limits<T>::radix << "\n"; |
| } |
| } |
| else |
| { |
| os << |
| "boost::math::policies::precision<" << nameof<T>() << ", Policy> \n" |
| "reports that there is no compile type precision available.\n" |
| "boost::math::tools::digits<" << nameof<T>() << ">() \n" |
| "reports that the current runtime precision is \n" << |
| boost::math::tools::digits<T>() << " binary digits.\n"; |
| } |
| |
| typedef typename construction_traits<T, Policy>::type construction_type; |
| |
| switch(construction_type::value) |
| { |
| case 0: |
| os << |
| "No compile time precision is available, the construction method \n" |
| "will be decided at runtime and results will not be cached \n" |
| "- this may lead to poor runtime performance.\n" |
| "Current runtime precision indicates that\n"; |
| if(boost::math::tools::digits<T>() > max_string_digits) |
| { |
| os << "the constant will be recalculated on each call.\n"; |
| } |
| else |
| { |
| os << "the constant will be constructed from a string on each call.\n"; |
| } |
| break; |
| case 1: |
| os << |
| "The constant will be constructed from a float.\n"; |
| break; |
| case 2: |
| os << |
| "The constant will be constructed from a double.\n"; |
| break; |
| case 3: |
| os << |
| "The constant will be constructed from a long double.\n"; |
| break; |
| case 4: |
| os << |
| "The constant will be constructed from a string (and the result cached).\n"; |
| break; |
| default: |
| os << |
| "The constant will be calculated (and the result cached).\n"; |
| break; |
| } |
| os << std::endl; |
| #ifdef _MSC_VER |
| #pragma warning(pop) |
| #endif |
| } |
| |
| template <class T> |
| void print_info_on_type(std::ostream& os = std::cout BOOST_MATH_APPEND_EXPLICIT_TEMPLATE_TYPE_SPEC(T)) |
| { |
| print_info_on_type<T, boost::math::policies::policy<> >(os); |
| } |
| |
| }}} // namespaces |
| |
| #endif // BOOST_MATH_CONSTANTS_INFO_INCLUDED |