Files
safe_numerics/include/safe_integer.hpp
Robert Ramey 12f5843f90 Fixed problems in testing addition operations
More carefully defined the concepts.  That is the concept of the library is now clear:

Trap at either compile or runtime integer operations which yield a result which differs from the normal arithmetic result.

Pending on this check in:
a) only addition is fixed up so far
b) uses BOOST_TYPEOF... rather than decltype(... - final decision sill pending.
c) comments out some compile time checks which should be re-instated.
d) fixed tests so that they actually work!
2014-04-05 12:22:25 -07:00

87 lines
2.3 KiB
C++

#ifndef BOOST_NUMERIC_SAFE_INTEGER_HPP
#define BOOST_NUMERIC_SAFE_INTEGER_HPP
// MS compatible compilers support #pragma once
#if defined(_MSC_VER) && (_MSC_VER >= 1020)
# pragma once
#endif
// Copyright (c) 2012 Robert Ramey
//
// Distributed under 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)
#include <boost/config.hpp> // BOOST_NOEXCEPT
#include <boost/limits.hpp>
#include <boost/integer_traits.hpp>
#include <boost/mpl/if.hpp>
#include "safe_range.hpp"
#include "numeric.hpp"
namespace boost {
namespace numeric {
namespace detail{
template<class T>
struct safe_integer_base {
typedef typename boost::mpl::if_<
boost::numeric::is_signed<T>,
boost::numeric::safe_signed_range<
static_cast<boost::intmax_t>(boost::integer_traits<T>::const_min),
static_cast<boost::intmax_t>(boost::integer_traits<T>::const_max)
>,
boost::numeric::safe_unsigned_range<
static_cast<boost::uintmax_t>(boost::integer_traits<T>::const_min),
static_cast<boost::uintmax_t>(boost::integer_traits<T>::const_max)
>
>::type type;
};
} // detail
template<class T>
struct safe : public detail::safe_integer_base<T>::type {
typedef typename detail::safe_integer_base<T>::type base_type;
// verify that std::numeric_limits has been specialized for this type
BOOST_STATIC_ASSERT_MSG(
std::numeric_limits<T>::is_specialized,
"std::numeric_limits<T> has not been specialized for this type"
);
// verify that T is an integer type
BOOST_STATIC_ASSERT_MSG(
std::numeric_limits<T>::is_integer,
"T is not an integer type"
);
safe(){}
template<class U>
safe(const U & u) :
detail::safe_integer_base<T>::type(u)
{}
};
} // numeric
} // boost
namespace std {
template<class T>
class numeric_limits< boost::numeric::safe<T> > : public
numeric_limits<T>
{
typedef boost::numeric::safe<T> SI;
public:
BOOST_STATIC_CONSTEXPR SI min() BOOST_NOEXCEPT { return numeric_limits<T>::min(); }
BOOST_STATIC_CONSTEXPR SI max() BOOST_NOEXCEPT { return numeric_limits<T>::max(); }
BOOST_STATIC_CONSTEXPR SI lowest() BOOST_NOEXCEPT { return numeric_limits<T>::min(); }
};
} // std
#endif // BOOST_NUMERIC_SAFE_INTEGER_HPP