#ifndef BOOST_NUMERIC_SAFE_BASE_HPP #define BOOST_NUMERIC_SAFE_BASE_HPP // 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 #include // is_integral, enable_if, conditional #include // BOOST_CLANG #include "concept/exception_policy.hpp" #include "concept/promotion_policy.hpp" #include "safe_common.hpp" #include "exception_policies.hpp" #include "boost/concept/assert.hpp" namespace boost { namespace safe_numerics { ///////////////////////////////////////////////////////////////// // forward declarations to support friend function declarations // in safe_base template< class Stored, Stored Min, Stored Max, class P, // promotion polic class E // exception policy > class safe_base; template< class T, T Min, T Max, class P, class E > struct is_safe > : public std::true_type {}; template< class T, T Min, T Max, class P, class E > struct get_promotion_policy > { using type = P; }; template< class T, T Min, T Max, class P, class E > struct get_exception_policy > { using type = E; }; template< class T, T Min, T Max, class P, class E > struct base_type > { using type = T; }; template< class T, T Min, T Max, class P, class E > constexpr T base_value( const safe_base & st ) { return static_cast(st); } template< typename T, T N, class P, // promotion policy class E // exception policy > class safe_literal_impl; // works for both GCC and clang #if BOOST_CLANG==1 #pragma GCC diagnostic push #pragma GCC diagnostic ignored "-Wmismatched-tags" #endif ///////////////////////////////////////////////////////////////// // Main implementation template< class Stored, Stored Min, Stored Max, class P, // promotion polic class E // exception policy > class safe_base { private: BOOST_CONCEPT_ASSERT((PromotionPolicy

)); BOOST_CONCEPT_ASSERT((ExceptionPolicy)); Stored m_t; template< class StoredX, StoredX MinX, StoredX MaxX, class PX, // promotion polic class EX // exception policy > friend class safe_base; friend class std::numeric_limits; template constexpr Stored validated_cast(const T & t) const; template constexpr Stored validated_cast( const safe_literal_impl & t ) const; // stream support template void output(std::basic_ostream & os) const; // note usage of friend declaration to mark function as // a global function rather than a member function. If // this is not done, the compiler will confuse this with // a member operator overload on the << operator. Weird // I know. But it's documented here // http://en.cppreference.com/w/cpp/language/friend // under the heading "Template friend operators" template friend std::basic_ostream & operator<<( std::basic_ostream & os, const safe_base & t ){ t.output(os); return os; } template void input(std::basic_istream & is); // see above template friend inline std::basic_istream & operator>>( std::basic_istream & is, safe_base & t ){ t.input(is); return is; } public: //////////////////////////////////////////////////////////// // constructors struct skip_validation{}; constexpr explicit safe_base(const Stored & rhs, skip_validation); constexpr safe_base(); // construct an instance of a safe type // from an instance of a convertible underlying type. template constexpr /*explicit*/ safe_base( const T & t, typename std::enable_if< is_safe::value, bool >::type = true ); template constexpr /*explicit*/ safe_base( const T & t, typename std::enable_if< std::is_integral::value, bool >::type = true ); template constexpr /*explicit*/ safe_base( const std::integral_constant & ); // note: Rule of Five. Supply all or none of the following // a) user-defined destructor ~safe_base() = default; // b) copy-constructor constexpr safe_base(const safe_base &) = default; // c) copy-assignment constexpr safe_base & operator=(const safe_base &) = default; // d) move constructor constexpr safe_base(safe_base &&) = default; // e) move assignment operator constexpr safe_base & operator=(safe_base &&) = default; ///////////////////////////////////////////////////////////////// // casting operators for intrinsic integers // convert to any type which is not safe. safe types need to be // excluded to prevent ambiguous function selection which // would otherwise occur. validity of safe types is checked in // the constructor of safe types template< class R, typename std::enable_if< ! boost::safe_numerics::is_safe::value, int >::type = 0 > constexpr /*explicit*/ operator R () const; constexpr /*explicit*/ operator Stored () const; ///////////////////////////////////////////////////////////////// // modification binary operators template constexpr safe_base & operator=(const T & rhs){ m_t = validated_cast(rhs); return *this; } // required to passify VS2017 constexpr safe_base & operator=(const Stored & rhs){ m_t = validated_cast(rhs); return *this; } // mutating unary operators safe_base & operator++(){ // pre increment return *this = *this + 1; } safe_base & operator--(){ // pre decrement return *this = *this - 1; } safe_base operator++(int){ // post increment safe_base old_t = *this; ++(*this); return old_t; } safe_base operator--(int){ // post decrement safe_base old_t = *this; --(*this); return old_t; } // non mutating unary operators constexpr auto operator+() const { // unary plus return *this; } // after much consideration, I've permited the resulting value of a unary // - to change the type. The C++ standard does invoke integral promotions // so it's changing the type as well. /* section 5.3.1 &8 of the C++ standard The operand of the unary - operator shall have arithmetic or unscoped enumeration type and the result is the negation of its operand. Integral promotion is performed on integral or enumeration operands. The negative of an unsigned quantity is computed by subtracting its value from 2n, where n is the number of bits in the promoted operand. The type of the result is the type of the promoted operand. */ constexpr auto operator-() const { // unary minus // if this is a unsigned type and the promotion policy is native // the result will be unsigned. But then the operation will fail // according to the requirements of arithmetic correctness. // if this is an unsigned type and the promotion policy is automatic. // the result will be signed. return 0 - *this; } /* section 5.3.1 &10 of the C++ standard The operand of ~ shall have integral or unscoped enumeration type; the result is the ones’ complement of its operand. Integral promotions are performed. The type of the result is the type of the promoted operand. */ constexpr auto operator~() const { // complement return ~Stored(0u) ^ *this; } }; } // safe_numerics } // boost ///////////////////////////////////////////////////////////////// // numeric limits for safe etc. #include namespace std { template< class T, T Min, T Max, class P, class E > class numeric_limits > : public std::numeric_limits { using SB = boost::safe_numerics::safe_base; public: constexpr static SB lowest() noexcept { return SB(Min, typename SB::skip_validation()); } constexpr static SB min() noexcept { return SB(Min, typename SB::skip_validation()); } constexpr static SB max() noexcept { return SB(Max, typename SB::skip_validation()); } }; } // std #if BOOST_CLANG==1 #pragma GCC diagnostic pop #endif #endif // BOOST_NUMERIC_SAFE_BASE_HPP