164 lines
5.4 KiB
C++
164 lines
5.4 KiB
C++
// 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>
|
|
#include <typeinfo>
|
|
|
|
namespace boost{ namespace math{ namespace constants{
|
|
|
|
namespace detail{
|
|
|
|
template <class T>
|
|
const char* nameof(BOOST_MATH_EXPLICIT_TEMPLATE_TYPE_SPEC(T))
|
|
{
|
|
return typeid(T).name();
|
|
}
|
|
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 BOOST_MSVC
|
|
#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 BOOST_MSVC
|
|
#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
|