123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182 |
- // Support for interoperability between Boost.System and <system_error>
- //
- // Copyright 2018 Peter Dimov
- //
- // 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)
- //
- // See library home page at http://www.boost.org/libs/system
- #include <system_error>
- #include <map>
- #include <memory>
- #include <mutex>
- //
- namespace boost
- {
- namespace system
- {
- namespace detail
- {
- class BOOST_SYMBOL_VISIBLE std_category: public std::error_category
- {
- private:
- boost::system::error_category const * pc_;
- public:
- explicit std_category( boost::system::error_category const * pc, unsigned id ): pc_( pc )
- {
- if( id != 0 )
- {
- #if defined(_MSC_VER) && defined(_CPPLIB_VER) && _MSC_VER >= 1900 && _MSC_VER < 2000
- // Poking into the protected _Addr member of std::error_category
- // is not a particularly good programming practice, but what can
- // you do
- _Addr = id;
- #endif
- }
- }
- virtual const char * name() const BOOST_NOEXCEPT
- {
- return pc_->name();
- }
- virtual std::string message( int ev ) const
- {
- return pc_->message( ev );
- }
- virtual std::error_condition default_error_condition( int ev ) const BOOST_NOEXCEPT
- {
- return pc_->default_error_condition( ev );
- }
- virtual bool equivalent( int code, const std::error_condition & condition ) const BOOST_NOEXCEPT;
- virtual bool equivalent( const std::error_code & code, int condition ) const BOOST_NOEXCEPT;
- };
- #if !defined(__SUNPRO_CC) // trailing __global is not supported
- inline std::error_category const & to_std_category( boost::system::error_category const & cat ) BOOST_SYMBOL_VISIBLE;
- #endif
- struct cat_ptr_less
- {
- bool operator()( boost::system::error_category const * p1, boost::system::error_category const * p2 ) const BOOST_NOEXCEPT
- {
- return *p1 < *p2;
- }
- };
- inline std::error_category const & to_std_category( boost::system::error_category const & cat )
- {
- if( cat == boost::system::system_category() )
- {
- static const std_category system_instance( &cat, 0x1F4D7 );
- return system_instance;
- }
- else if( cat == boost::system::generic_category() )
- {
- static const std_category generic_instance( &cat, 0x1F4D3 );
- return generic_instance;
- }
- else
- {
- typedef std::map< boost::system::error_category const *, std::unique_ptr<std_category>, cat_ptr_less > map_type;
- static map_type map_;
- static std::mutex map_mx_;
- std::lock_guard<std::mutex> guard( map_mx_ );
- map_type::iterator i = map_.find( &cat );
- if( i == map_.end() )
- {
- std::unique_ptr<std_category> p( new std_category( &cat, 0 ) );
- std::pair<map_type::iterator, bool> r = map_.insert( map_type::value_type( &cat, std::move( p ) ) );
- i = r.first;
- }
- return *i->second;
- }
- }
- inline bool std_category::equivalent( int code, const std::error_condition & condition ) const BOOST_NOEXCEPT
- {
- if( condition.category() == *this )
- {
- boost::system::error_condition bn( condition.value(), *pc_ );
- return pc_->equivalent( code, bn );
- }
- else if( condition.category() == std::generic_category() || condition.category() == boost::system::generic_category() )
- {
- boost::system::error_condition bn( condition.value(), boost::system::generic_category() );
- return pc_->equivalent( code, bn );
- }
- #ifndef BOOST_NO_RTTI
- else if( std_category const* pc2 = dynamic_cast< std_category const* >( &condition.category() ) )
- {
- boost::system::error_condition bn( condition.value(), *pc2->pc_ );
- return pc_->equivalent( code, bn );
- }
- #endif
- else
- {
- return default_error_condition( code ) == condition;
- }
- }
- inline bool std_category::equivalent( const std::error_code & code, int condition ) const BOOST_NOEXCEPT
- {
- if( code.category() == *this )
- {
- boost::system::error_code bc( code.value(), *pc_ );
- return pc_->equivalent( bc, condition );
- }
- else if( code.category() == std::generic_category() || code.category() == boost::system::generic_category() )
- {
- boost::system::error_code bc( code.value(), boost::system::generic_category() );
- return pc_->equivalent( bc, condition );
- }
- #ifndef BOOST_NO_RTTI
- else if( std_category const* pc2 = dynamic_cast< std_category const* >( &code.category() ) )
- {
- boost::system::error_code bc( code.value(), *pc2->pc_ );
- return pc_->equivalent( bc, condition );
- }
- #endif
- else if( *pc_ == boost::system::generic_category() )
- {
- return std::generic_category().equivalent( code, condition );
- }
- else
- {
- return false;
- }
- }
- } // namespace detail
- } // namespace system
- } // namespace boost
|