///////////////////////////////////////////////////////////////////////////// // // (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 BHO_INTRUSIVE_GENERIC_HOOK_HPP #define BHO_INTRUSIVE_GENERIC_HOOK_HPP #ifndef BHO_CONFIG_HPP # include #endif #if defined(BHO_HAS_PRAGMA_ONCE) # pragma once #endif #include #include #include #include #include #include #include namespace bho { namespace intrusive { /// @cond namespace detail { template struct link_dispatch {}; template BHO_INTRUSIVE_FORCEINLINE void destructor_impl(Hook &hook, detail::link_dispatch) { //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; BHO_INTRUSIVE_SAFE_HOOK_DESTRUCTOR_ASSERT(!hook.is_linked()); } template BHO_INTRUSIVE_FORCEINLINE void destructor_impl(Hook &hook, detail::link_dispatch) { hook.unlink(); } template BHO_INTRUSIVE_FORCEINLINE void destructor_impl(Hook &, detail::link_dispatch) {} } //namespace detail { enum base_hook_type { NoBaseHookId , ListBaseHookId , SlistBaseHookId , RbTreeBaseHookId , HashBaseHookId , AvlTreeBaseHookId , BsTreeBaseHookId , TreapTreeBaseHookId , AnyBaseHookId }; template struct hook_tags_definer{}; template struct hook_tags_definer { typedef HookTags default_list_hook; }; template struct hook_tags_definer { typedef HookTags default_slist_hook; }; template struct hook_tags_definer { typedef HookTags default_rbtree_hook; }; template struct hook_tags_definer { typedef HookTags default_hashtable_hook; }; template struct hook_tags_definer { typedef HookTags default_avltree_hook; }; template struct hook_tags_definer { typedef HookTags default_bstree_hook; }; template struct hook_tags_definer { 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::value; static const bool safemode_or_autounlink = is_safe_autounlink::value; static const unsigned int type = BaseHookType; }; /// @endcond template < bho::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::value , typename NodeTraits::node , node_holder >::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 , detail::is_same::value ? BaseHookType : NoBaseHookId> /// @endcond { /// @cond typedef typename get_algo::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; BHO_INTRUSIVE_FORCEINLINE node_ptr this_ptr() BHO_NOEXCEPT { return pointer_traits::pointer_to(static_cast(*this)); } BHO_INTRUSIVE_FORCEINLINE const_node_ptr this_ptr() const BHO_NOEXCEPT { return pointer_traits::pointer_to(static_cast(*this)); } public: /// @endcond BHO_INTRUSIVE_FORCEINLINE generic_hook() BHO_NOEXCEPT { if(hooktags::safemode_or_autounlink){ node_algorithms::init(this->this_ptr()); } } BHO_INTRUSIVE_FORCEINLINE generic_hook(const generic_hook& ) BHO_NOEXCEPT { if(hooktags::safemode_or_autounlink){ node_algorithms::init(this->this_ptr()); } } BHO_INTRUSIVE_FORCEINLINE generic_hook& operator=(const generic_hook& ) BHO_NOEXCEPT { return *this; } BHO_INTRUSIVE_FORCEINLINE ~generic_hook() { destructor_impl (*this, detail::link_dispatch()); } BHO_INTRUSIVE_FORCEINLINE void swap_nodes(generic_hook &other) BHO_NOEXCEPT { node_algorithms::swap_nodes (this->this_ptr(), other.this_ptr()); } BHO_INTRUSIVE_FORCEINLINE bool is_linked() const BHO_NOEXCEPT { //is_linked() can be only used in safe-mode or auto-unlink BHO_STATIC_ASSERT(( hooktags::safemode_or_autounlink )); return !node_algorithms::unique(this->this_ptr()); } BHO_INTRUSIVE_FORCEINLINE void unlink() BHO_NOEXCEPT { BHO_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 bho #endif //BHO_INTRUSIVE_GENERIC_HOOK_HPP