224 lines
		
	
	
		
			6.3 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
			
		
		
	
	
			224 lines
		
	
	
		
			6.3 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
| /////////////////////////////////////////////////////////////////////////////
 | |
| //
 | |
| // (C) Copyright Ion Gaztanaga 2007-2013
 | |
| //
 | |
| // 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/intrusive for documentation.
 | |
| //
 | |
| /////////////////////////////////////////////////////////////////////////////
 | |
| 
 | |
| #ifndef BOOST_INTRUSIVE_GENERIC_HOOK_HPP
 | |
| #define BOOST_INTRUSIVE_GENERIC_HOOK_HPP
 | |
| 
 | |
| #ifndef BOOST_CONFIG_HPP
 | |
| #  include <boost/config.hpp>
 | |
| #endif
 | |
| 
 | |
| #if defined(BOOST_HAS_PRAGMA_ONCE)
 | |
| #  pragma once
 | |
| #endif
 | |
| 
 | |
| #include <boost/intrusive/pointer_traits.hpp>
 | |
| #include <boost/intrusive/link_mode.hpp>
 | |
| #include <boost/intrusive/detail/mpl.hpp>
 | |
| #include <boost/intrusive/detail/assert.hpp>
 | |
| #include <boost/intrusive/detail/node_holder.hpp>
 | |
| #include <boost/intrusive/detail/algo_type.hpp>
 | |
| #include <boost/static_assert.hpp>
 | |
| 
 | |
| namespace boost {
 | |
| namespace intrusive {
 | |
| 
 | |
| /// @cond
 | |
| 
 | |
| namespace detail {
 | |
| 
 | |
| template <link_mode_type LinkMode>
 | |
| struct link_dispatch
 | |
| {};
 | |
| 
 | |
| template<class Hook>
 | |
| void destructor_impl(Hook &hook, detail::link_dispatch<safe_link>)
 | |
| {  //If this assertion raises, you might have destroyed an object
 | |
|    //while it was still inserted in a container that is alive.
 | |
|    //If so, remove the object from the container before destroying it.
 | |
|    (void)hook; BOOST_INTRUSIVE_SAFE_HOOK_DESTRUCTOR_ASSERT(!hook.is_linked());
 | |
| }
 | |
| 
 | |
| template<class Hook>
 | |
| void destructor_impl(Hook &hook, detail::link_dispatch<auto_unlink>)
 | |
| {  hook.unlink();  }
 | |
| 
 | |
| template<class Hook>
 | |
| void destructor_impl(Hook &, detail::link_dispatch<normal_link>)
 | |
| {}
 | |
| 
 | |
| }  //namespace detail {
 | |
| 
 | |
| enum base_hook_type
 | |
| {  NoBaseHookId
 | |
| ,  ListBaseHookId
 | |
| ,  SlistBaseHookId
 | |
| ,  RbTreeBaseHookId
 | |
| ,  HashBaseHookId
 | |
| ,  AvlTreeBaseHookId
 | |
| ,  BsTreeBaseHookId
 | |
| ,  TreapTreeBaseHookId
 | |
| ,  AnyBaseHookId
 | |
| };
 | |
| 
 | |
| 
 | |
| template <class HookTags, unsigned int>
 | |
| struct hook_tags_definer{};
 | |
| 
 | |
| template <class HookTags>
 | |
| struct hook_tags_definer<HookTags, ListBaseHookId>
 | |
| {  typedef HookTags default_list_hook;  };
 | |
| 
 | |
| template <class HookTags>
 | |
| struct hook_tags_definer<HookTags, SlistBaseHookId>
 | |
| {  typedef HookTags default_slist_hook;  };
 | |
| 
 | |
| template <class HookTags>
 | |
| struct hook_tags_definer<HookTags, RbTreeBaseHookId>
 | |
| {  typedef HookTags default_rbtree_hook;  };
 | |
| 
 | |
| template <class HookTags>
 | |
| struct hook_tags_definer<HookTags, HashBaseHookId>
 | |
| {  typedef HookTags default_hashtable_hook;  };
 | |
| 
 | |
| template <class HookTags>
 | |
| struct hook_tags_definer<HookTags, AvlTreeBaseHookId>
 | |
| {  typedef HookTags default_avltree_hook;  };
 | |
| 
 | |
| template <class HookTags>
 | |
| struct hook_tags_definer<HookTags, BsTreeBaseHookId>
 | |
| {  typedef HookTags default_bstree_hook;  };
 | |
| 
 | |
| template <class HookTags>
 | |
| struct hook_tags_definer<HookTags, AnyBaseHookId>
 | |
| {  typedef HookTags default_any_hook;  };
 | |
| 
 | |
| template
 | |
|    < class NodeTraits
 | |
|    , class Tag
 | |
|    , link_mode_type LinkMode
 | |
|    , base_hook_type BaseHookType
 | |
|    >
 | |
| struct hooktags_impl
 | |
| {
 | |
|    static const link_mode_type link_mode = LinkMode;
 | |
|    typedef Tag tag;
 | |
|    typedef NodeTraits node_traits;
 | |
|    static const bool is_base_hook = !detail::is_same<Tag, member_tag>::value;
 | |
|    static const bool safemode_or_autounlink = is_safe_autounlink<link_mode>::value;
 | |
|    static const unsigned int type = BaseHookType;
 | |
| };
 | |
| 
 | |
| /// @endcond
 | |
| 
 | |
| template
 | |
|    < boost::intrusive::algo_types Algo
 | |
|    , class NodeTraits
 | |
|    , class Tag
 | |
|    , link_mode_type LinkMode
 | |
|    , base_hook_type BaseHookType
 | |
|    >
 | |
| class generic_hook
 | |
|    /// @cond
 | |
|    //If the hook is a base hook, derive generic hook from node_holder
 | |
|    //so that a unique base class is created to convert from the node
 | |
|    //to the type. This mechanism will be used by bhtraits.
 | |
|    //
 | |
|    //If the hook is a member hook, generic hook will directly derive
 | |
|    //from the hook.
 | |
|    : public detail::if_c
 | |
|       < detail::is_same<Tag, member_tag>::value
 | |
|       , typename NodeTraits::node
 | |
|       , node_holder<typename NodeTraits::node, Tag, BaseHookType>
 | |
|       >::type
 | |
|    //If this is the a default-tagged base hook derive from a class that
 | |
|    //will define an special internal typedef. Containers will be able to detect this
 | |
|    //special typedef and obtain generic_hook's internal types in order to deduce
 | |
|    //value_traits for this hook.
 | |
|    , public hook_tags_definer
 | |
|       < generic_hook<Algo, NodeTraits, Tag, LinkMode, BaseHookType>
 | |
|       , detail::is_same<Tag, dft_tag>::value ? BaseHookType : NoBaseHookId>
 | |
|    /// @endcond
 | |
| {
 | |
|    /// @cond
 | |
|    typedef typename get_algo<Algo, NodeTraits>::type  node_algorithms;
 | |
|    typedef typename node_algorithms::node             node;
 | |
|    typedef typename node_algorithms::node_ptr         node_ptr;
 | |
|    typedef typename node_algorithms::const_node_ptr   const_node_ptr;
 | |
| 
 | |
|    public:
 | |
| 
 | |
|    typedef hooktags_impl
 | |
|       < NodeTraits
 | |
|       , Tag, LinkMode, BaseHookType>                  hooktags;
 | |
| 
 | |
|    node_ptr this_ptr()
 | |
|    {  return pointer_traits<node_ptr>::pointer_to(static_cast<node&>(*this)); }
 | |
| 
 | |
|    const_node_ptr this_ptr() const
 | |
|    {  return pointer_traits<const_node_ptr>::pointer_to(static_cast<const node&>(*this)); }
 | |
| 
 | |
|    public:
 | |
|    /// @endcond
 | |
| 
 | |
|    generic_hook()
 | |
|    {
 | |
|       if(hooktags::safemode_or_autounlink){
 | |
|          node_algorithms::init(this->this_ptr());
 | |
|       }
 | |
|    }
 | |
| 
 | |
|    generic_hook(const generic_hook& )
 | |
|    {
 | |
|       if(hooktags::safemode_or_autounlink){
 | |
|          node_algorithms::init(this->this_ptr());
 | |
|       }
 | |
|    }
 | |
| 
 | |
|    generic_hook& operator=(const generic_hook& )
 | |
|    {  return *this;  }
 | |
| 
 | |
|    ~generic_hook()
 | |
|    {
 | |
|       destructor_impl
 | |
|          (*this, detail::link_dispatch<hooktags::link_mode>());
 | |
|    }
 | |
| 
 | |
|    void swap_nodes(generic_hook &other)
 | |
|    {
 | |
|       node_algorithms::swap_nodes
 | |
|          (this->this_ptr(), other.this_ptr());
 | |
|    }
 | |
| 
 | |
|    bool is_linked() const
 | |
|    {
 | |
|       //is_linked() can be only used in safe-mode or auto-unlink
 | |
|       BOOST_STATIC_ASSERT(( hooktags::safemode_or_autounlink ));
 | |
|       return !node_algorithms::unique(this->this_ptr());
 | |
|    }
 | |
| 
 | |
|    void unlink()
 | |
|    {
 | |
|       BOOST_STATIC_ASSERT(( (int)hooktags::link_mode == (int)auto_unlink ));
 | |
|       node_ptr n(this->this_ptr());
 | |
|       if(!node_algorithms::inited(n)){
 | |
|          node_algorithms::unlink(n);
 | |
|          node_algorithms::init(n);
 | |
|       }
 | |
|    }
 | |
| };
 | |
| 
 | |
| } //namespace intrusive
 | |
| } //namespace boost
 | |
| 
 | |
| #endif //BOOST_INTRUSIVE_GENERIC_HOOK_HPP
 | 
