123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501 |
- /* Flyweight class.
- *
- * Copyright 2006-2023 Joaquin M Lopez Munoz.
- * 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 http://www.boost.org/libs/flyweight for library home page.
- */
- #ifndef BOOST_FLYWEIGHT_FLYWEIGHT_HPP
- #define BOOST_FLYWEIGHT_FLYWEIGHT_HPP
- #if defined(_MSC_VER)
- #pragma once
- #endif
- #include <boost/config.hpp> /* keep it first to prevent nasty warns in MSVC */
- #include <algorithm>
- #include <boost/core/addressof.hpp>
- #include <boost/core/invoke_swap.hpp>
- #include <boost/detail/workaround.hpp>
- #include <boost/flyweight/detail/default_value_policy.hpp>
- #include <boost/flyweight/detail/flyweight_core.hpp>
- #include <boost/flyweight/detail/perfect_fwd.hpp>
- #include <boost/flyweight/factory_tag.hpp>
- #include <boost/flyweight/flyweight_fwd.hpp>
- #include <boost/flyweight/locking_tag.hpp>
- #include <boost/flyweight/simple_locking_fwd.hpp>
- #include <boost/flyweight/static_holder_fwd.hpp>
- #include <boost/flyweight/hashed_factory_fwd.hpp>
- #include <boost/flyweight/holder_tag.hpp>
- #include <boost/flyweight/refcounted_fwd.hpp>
- #include <boost/flyweight/tag.hpp>
- #include <boost/flyweight/tracking_tag.hpp>
- #include <boost/mpl/assert.hpp>
- #include <boost/mpl/if.hpp>
- #include <boost/mpl/not.hpp>
- #include <boost/mpl/or.hpp>
- #include <boost/parameter/binding.hpp>
- #include <boost/type_traits/is_same.hpp>
- #if !defined(BOOST_NO_SFINAE)&&!defined(BOOST_NO_CXX11_HDR_INITIALIZER_LIST)
- #include <boost/core/enable_if.hpp>
- #include <boost/type_traits/is_convertible.hpp>
- #include <initializer_list>
- #endif
- #if BOOST_WORKAROUND(BOOST_MSVC,BOOST_TESTED_AT(1400))
- #pragma warning(push)
- #pragma warning(disable:4520) /* multiple default ctors */
- #pragma warning(disable:4521) /* multiple copy ctors */
- #endif
- namespace boost{
-
- namespace flyweights{
- namespace detail{
- /* Used for the detection of unmatched template args in a
- * flyweight instantiation.
- */
- struct unmatched_arg;
- /* Boost.Parameter structures for use in flyweight.
- * NB: these types are derived from instead of typedef'd to force their
- * instantiation, which solves http://bugs.sun.com/view_bug.do?bug_id=6782987
- * as found out by Simon Atanasyan.
- */
- struct flyweight_signature:
- parameter::parameters<
- parameter::optional<
- parameter::deduced<tag<> >,
- detail::is_tag<boost::mpl::_>
- >,
- parameter::optional<
- parameter::deduced<tracking<> >,
- is_tracking<boost::mpl::_>
- >,
- parameter::optional<
- parameter::deduced<factory<> >,
- is_factory<boost::mpl::_>
- >,
- parameter::optional<
- parameter::deduced<locking<> >,
- is_locking<boost::mpl::_>
- >,
- parameter::optional<
- parameter::deduced<holder<> >,
- is_holder<boost::mpl::_>
- >
- >
- {};
- struct flyweight_unmatched_signature:
- parameter::parameters<
- parameter::optional<
- parameter::deduced<
- detail::unmatched_arg
- >,
- mpl::not_<
- mpl::or_<
- detail::is_tag<boost::mpl::_>,
- is_tracking<boost::mpl::_>,
- is_factory<boost::mpl::_>,
- is_locking<boost::mpl::_>,
- is_holder<boost::mpl::_>
- >
- >
- >
- >
- {};
- } /* namespace flyweights::detail */
- template<
- typename T,
- typename Arg1,typename Arg2,typename Arg3,typename Arg4,typename Arg5
- >
- class flyweight
- {
- private:
- typedef typename mpl::if_<
- detail::is_value<T>,
- T,
- detail::default_value_policy<T>
- >::type value_policy;
- typedef typename detail::
- flyweight_signature::bind<
- Arg1,Arg2,Arg3,Arg4,Arg5
- >::type args;
- typedef typename parameter::binding<
- args,tag<>,mpl::na
- >::type tag_type;
- typedef typename parameter::binding<
- args,tracking<>,refcounted
- >::type tracking_policy;
- typedef typename parameter::binding<
- args,factory<>,hashed_factory<>
- >::type factory_specifier;
- typedef typename parameter::binding<
- args,locking<>,simple_locking
- >::type locking_policy;
- typedef typename parameter::binding<
- args,holder<>,static_holder
- >::type holder_specifier;
- typedef typename detail::
- flyweight_unmatched_signature::bind<
- Arg1,Arg2,Arg3,Arg4,Arg5
- >::type unmatched_args;
- typedef typename parameter::binding<
- unmatched_args,detail::unmatched_arg,
- detail::unmatched_arg
- >::type unmatched_arg_detected;
- /* You have passed a type in the specification of a flyweight type that
- * could not be interpreted as a valid argument.
- */
- BOOST_MPL_ASSERT_MSG(
- (is_same<unmatched_arg_detected,detail::unmatched_arg>::value),
- INVALID_ARGUMENT_TO_FLYWEIGHT,
- (flyweight));
- typedef detail::flyweight_core<
- value_policy,tag_type,tracking_policy,
- factory_specifier,locking_policy,
- holder_specifier
- > core;
- typedef typename core::handle_type handle_type;
- public:
- typedef typename value_policy::key_type key_type;
- typedef typename value_policy::value_type value_type;
- /* static data initialization */
- static bool init(){return core::init();}
- class initializer
- {
- public:
- initializer():b(init()){}
- private:
- bool b;
- };
- /* construct/copy/destroy */
- flyweight():h(core::insert()){}
-
- #define BOOST_FLYWEIGHT_PERFECT_FWD_CTR_BODY(args) \
- :h(core::insert(BOOST_FLYWEIGHT_FORWARD(args))){}
- BOOST_FLYWEIGHT_PERFECT_FWD_WITH_ARGS(
- explicit flyweight,
- BOOST_FLYWEIGHT_PERFECT_FWD_CTR_BODY)
- #undef BOOST_FLYWEIGHT_PERFECT_FWD_CTR_BODY
- #if !defined(BOOST_NO_SFINAE)&&!defined(BOOST_NO_CXX11_HDR_INITIALIZER_LIST)
- template<typename V>
- flyweight(
- std::initializer_list<V> list,
- typename boost::enable_if<
- boost::is_convertible<std::initializer_list<V>,key_type> >::type* =0):
- h(core::insert(list)){}
- #endif
- flyweight(const flyweight& x):h(x.h){}
- flyweight(flyweight& x):h(x.h){}
- #if !defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
- flyweight(const flyweight&& x):h(x.h){}
- flyweight(flyweight&& x):h(x.h){}
- #endif
- #if !defined(BOOST_NO_SFINAE)&&!defined(BOOST_NO_CXX11_HDR_INITIALIZER_LIST)
- template<typename V>
- typename boost::enable_if<
- boost::is_convertible<std::initializer_list<V>,key_type>,flyweight&>::type
- operator=(std::initializer_list<V> list)
- {
- return operator=(flyweight(list));
- }
- #endif
- flyweight& operator=(const flyweight& x){h=x.h;return *this;}
- flyweight& operator=(const value_type& x){return operator=(flyweight(x));}
- #if !defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
- flyweight& operator=(value_type&& x)
- {
- return operator=(flyweight(std::move(x)));
- }
- #endif
- /* convertibility to underlying type */
-
- const key_type& get_key()const{return core::key(h);}
- const value_type& get()const{return core::value(h);}
- const value_type& operator*()const{return get();}
- operator const value_type&()const{return get();}
- const value_type* operator->()const{return boost::addressof(get());}
-
- /* exact type equality */
-
- friend bool operator==(const flyweight& x,const flyweight& y)
- {
- return &x.get()==&y.get();
- }
- /* modifiers */
- void swap(flyweight& x){boost::core::invoke_swap(h,x.h);}
-
- private:
- handle_type h;
- };
- #define BOOST_FLYWEIGHT_TYPENAME_TEMPL_ARGS(n) \
- typename Arg##n##1,typename Arg##n##2,typename Arg##n##3, \
- typename Arg##n##4,typename Arg##n##5
- #define BOOST_FLYWEIGHT_TEMPL_ARGS(n) \
- Arg##n##1,Arg##n##2,Arg##n##3,Arg##n##4,Arg##n##5
- /* Comparison. Unlike exact type comparison defined above, intertype
- * comparison just forwards to the underlying objects.
- */
- template<
- typename T1,BOOST_FLYWEIGHT_TYPENAME_TEMPL_ARGS(1),
- typename T2,BOOST_FLYWEIGHT_TYPENAME_TEMPL_ARGS(2)
- >
- bool operator==(
- const flyweight<T1,BOOST_FLYWEIGHT_TEMPL_ARGS(1)>& x,
- const flyweight<T2,BOOST_FLYWEIGHT_TEMPL_ARGS(2)>& y)
- {
- return x.get()==y.get();
- }
- template<
- typename T1,BOOST_FLYWEIGHT_TYPENAME_TEMPL_ARGS(1),
- typename T2,BOOST_FLYWEIGHT_TYPENAME_TEMPL_ARGS(2)
- >
- bool operator<(
- const flyweight<T1,BOOST_FLYWEIGHT_TEMPL_ARGS(1)>& x,
- const flyweight<T2,BOOST_FLYWEIGHT_TEMPL_ARGS(2)>& y)
- {
- return x.get()<y.get();
- }
- #if !defined(BOOST_NO_FUNCTION_TEMPLATE_ORDERING)
- template<
- typename T1,BOOST_FLYWEIGHT_TYPENAME_TEMPL_ARGS(1),
- typename T2
- >
- bool operator==(
- const flyweight<T1,BOOST_FLYWEIGHT_TEMPL_ARGS(1)>& x,const T2& y)
- {
- return x.get()==y;
- }
- template<
- typename T1,
- typename T2,BOOST_FLYWEIGHT_TYPENAME_TEMPL_ARGS(2)
- >
- bool operator==(
- const T1& x,const flyweight<T2,BOOST_FLYWEIGHT_TEMPL_ARGS(2)>& y)
- {
- return x==y.get();
- }
- template<
- typename T1,BOOST_FLYWEIGHT_TYPENAME_TEMPL_ARGS(1),
- typename T2
- >
- bool operator<(
- const flyweight<T1,BOOST_FLYWEIGHT_TEMPL_ARGS(1)>& x,const T2& y)
- {
- return x.get()<y;
- }
- template<
- typename T1,
- typename T2,BOOST_FLYWEIGHT_TYPENAME_TEMPL_ARGS(2)
- >
- bool operator<(
- const T1& x,const flyweight<T2,BOOST_FLYWEIGHT_TEMPL_ARGS(2)>& y)
- {
- return x<y.get();
- }
- #endif /* !defined(BOOST_NO_FUNCTION_TEMPLATE_ORDERING) */
- /* rest of comparison operators */
- #define BOOST_FLYWEIGHT_COMPLETE_COMP_OPS(t,a1,a2) \
- template<t> \
- inline bool operator!=(const a1& x,const a2& y) \
- { \
- return !(x==y); \
- } \
- \
- template<t> \
- inline bool operator>(const a1& x,const a2& y) \
- { \
- return y<x; \
- } \
- \
- template<t> \
- inline bool operator>=(const a1& x,const a2& y) \
- { \
- return !(x<y); \
- } \
- \
- template<t> \
- inline bool operator<=(const a1& x,const a2& y) \
- { \
- return !(y<x); \
- }
- BOOST_FLYWEIGHT_COMPLETE_COMP_OPS(
- typename T1 BOOST_PP_COMMA()
- BOOST_FLYWEIGHT_TYPENAME_TEMPL_ARGS(1) BOOST_PP_COMMA()
- typename T2 BOOST_PP_COMMA()
- BOOST_FLYWEIGHT_TYPENAME_TEMPL_ARGS(2),
- flyweight<
- T1 BOOST_PP_COMMA() BOOST_FLYWEIGHT_TEMPL_ARGS(1)
- >,
- flyweight<
- T2 BOOST_PP_COMMA() BOOST_FLYWEIGHT_TEMPL_ARGS(2)
- >)
- #if !defined(BOOST_NO_FUNCTION_TEMPLATE_ORDERING)
- BOOST_FLYWEIGHT_COMPLETE_COMP_OPS(
- typename T1 BOOST_PP_COMMA()
- BOOST_FLYWEIGHT_TYPENAME_TEMPL_ARGS(1) BOOST_PP_COMMA()
- typename T2,
- flyweight<
- T1 BOOST_PP_COMMA() BOOST_FLYWEIGHT_TEMPL_ARGS(1)
- >,
- T2)
-
- BOOST_FLYWEIGHT_COMPLETE_COMP_OPS(
- typename T1 BOOST_PP_COMMA()
- typename T2 BOOST_PP_COMMA()
- BOOST_FLYWEIGHT_TYPENAME_TEMPL_ARGS(2),
- T1,
- flyweight<
- T2 BOOST_PP_COMMA() BOOST_FLYWEIGHT_TEMPL_ARGS(2)
- >)
- #endif /* !defined(BOOST_NO_FUNCTION_TEMPLATE_ORDERING) */
- /* specialized algorithms */
- template<typename T,BOOST_FLYWEIGHT_TYPENAME_TEMPL_ARGS(_)>
- void swap(
- flyweight<T,BOOST_FLYWEIGHT_TEMPL_ARGS(_)>& x,
- flyweight<T,BOOST_FLYWEIGHT_TEMPL_ARGS(_)>& y)
- {
- x.swap(y);
- }
- template<
- BOOST_TEMPLATED_STREAM_ARGS(ElemType,Traits)
- BOOST_TEMPLATED_STREAM_COMMA
- typename T,BOOST_FLYWEIGHT_TYPENAME_TEMPL_ARGS(_)
- >
- BOOST_TEMPLATED_STREAM(ostream,ElemType,Traits)& operator<<(
- BOOST_TEMPLATED_STREAM(ostream,ElemType,Traits)& out,
- const flyweight<T,BOOST_FLYWEIGHT_TEMPL_ARGS(_)>& x)
- {
- return out<<x.get();
- }
- template<
- BOOST_TEMPLATED_STREAM_ARGS(ElemType,Traits)
- BOOST_TEMPLATED_STREAM_COMMA
- typename T,BOOST_FLYWEIGHT_TYPENAME_TEMPL_ARGS(_)
- >
- BOOST_TEMPLATED_STREAM(istream,ElemType,Traits)& operator>>(
- BOOST_TEMPLATED_STREAM(istream,ElemType,Traits)& in,
- flyweight<T,BOOST_FLYWEIGHT_TEMPL_ARGS(_)>& x)
- {
- typedef typename flyweight<
- T,BOOST_FLYWEIGHT_TEMPL_ARGS(_)
- >::value_type value_type;
- /* value_type need not be default ctble but must be copy ctble */
- value_type t(x.get());
- in>>t;
- x=t;
- return in;
- }
- } /* namespace flyweights */
- } /* namespace boost */
- #if !defined(BOOST_FLYWEIGHT_DISABLE_HASH_SUPPORT)
- /* hash support */
- #if !defined(BOOST_NO_CXX11_HDR_FUNCTIONAL)
- namespace std{
- template<typename T,BOOST_FLYWEIGHT_TYPENAME_TEMPL_ARGS(_)>
- BOOST_FLYWEIGHT_STD_HASH_STRUCT_KEYWORD
- hash<boost::flyweight<T,BOOST_FLYWEIGHT_TEMPL_ARGS(_)> >
- {
- public:
- typedef std::size_t result_type;
- typedef boost::flyweight<
- T,BOOST_FLYWEIGHT_TEMPL_ARGS(_)> argument_type;
- result_type operator()(const argument_type& x)const
- {
- typedef typename argument_type::value_type value_type;
- std::hash<const value_type*> h;
- return h(&x.get());
- }
- };
- } /* namespace std */
- #endif /* !defined(BOOST_NO_CXX11_HDR_FUNCTIONAL) */
- namespace boost{
- #if !defined(BOOST_NO_ARGUMENT_DEPENDENT_LOOKUP)
- namespace flyweights{
- #endif
- template<typename T,BOOST_FLYWEIGHT_TYPENAME_TEMPL_ARGS(_)>
- std::size_t hash_value(const flyweight<T,BOOST_FLYWEIGHT_TEMPL_ARGS(_)>& x)
- {
- typedef typename flyweight<
- T,BOOST_FLYWEIGHT_TEMPL_ARGS(_)
- >::value_type value_type;
- boost::hash<const value_type*> h;
- return h(&x.get());
- }
- #if !defined(BOOST_NO_ARGUMENT_DEPENDENT_LOOKUP)
- } /* namespace flyweights */
- #endif
- } /* namespace boost */
- #endif /* !defined(BOOST_FLYWEIGHT_DISABLE_HASH_SUPPORT) */
- #undef BOOST_FLYWEIGHT_COMPLETE_COMP_OPS
- #undef BOOST_FLYWEIGHT_TEMPL_ARGS
- #undef BOOST_FLYWEIGHT_TYPENAME_TEMPL_ARGS
- #if BOOST_WORKAROUND(BOOST_MSVC,BOOST_TESTED_AT(1400))
- #pragma warning(pop)
- #endif
- #endif
|