////////////////////////////////////////////////////////////////////////////// // // (C) Copyright Ion Gaztanaga 2005-2012. 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/interprocess for documentation. // ////////////////////////////////////////////////////////////////////////////// #ifndef BOOST_INTERPROCESS_IUNORDERED_SET_INDEX_HPP #define BOOST_INTERPROCESS_IUNORDERED_SET_INDEX_HPP #ifndef BOOST_CONFIG_HPP # include #endif # #if defined(BOOST_HAS_PRAGMA_ONCE) # pragma once #endif #include #include #include #include #include #include #include #include //std::less #include //std::char_traits #include #include //!\file //!Describes index adaptor of boost::intrusive::unordered_set container, to use it //!as name/shared memory index namespace boost { namespace interprocess { #if !defined(BOOST_INTERPROCESS_DOXYGEN_INVOKED) //!Helper class to define typedefs //!from IndexTraits template struct iunordered_set_index_aux { typedef typename MapConfig::segment_manager_base segment_manager_base; typedef typename segment_manager_base::void_pointer void_pointer; typedef typename bi::make_unordered_set_base_hook < bi::void_pointer >::type derivation_hook; typedef typename MapConfig::template intrusive_value_type::type value_type; typedef typename MapConfig:: intrusive_compare_key_type intrusive_compare_key_type; typedef std::equal_to value_equal; typedef typename MapConfig::char_type char_type; struct equal_function { bool operator()(const intrusive_compare_key_type &i, const value_type &b) const { return (i.m_len == b.name_length()) && (std::char_traits::compare (i.mp_str, b.name(), i.m_len) == 0); } bool operator()(const value_type &b, const intrusive_compare_key_type &i) const { return (i.m_len == b.name_length()) && (std::char_traits::compare (i.mp_str, b.name(), i.m_len) == 0); } bool operator()(const value_type &b1, const value_type &b2) const { return (b1.name_length() == b2.name_length()) && (std::char_traits::compare (b1.name(), b2.name(), b1.name_length()) == 0); } }; struct hash_function { typedef value_type argument_type; typedef std::size_t result_type; std::size_t hash_char_range(const char_type* beg, const char_type* end) const { std::size_t seed = 0; while (beg != end) { boost::intrusive::detail::hash_combine_size_t(seed, boost::intrusive::detail::internal_hash_functor()(*beg)); ++beg; } return seed; } std::size_t operator()(const value_type &val) const { const char_type* beg = ipcdetail::to_raw_pointer(val.name()), * end = beg + val.name_length(); return hash_char_range(beg, end); } std::size_t operator()(const intrusive_compare_key_type &i) const { const char_type *beg = i.mp_str, *end = beg + i.m_len; return hash_char_range(beg, end); } }; typedef typename bi::make_unordered_set < value_type , bi::hash , bi::equal , bi::size_type >::type index_t; typedef typename index_t::bucket_type bucket_type; typedef allocator allocator_type; struct allocator_holder { allocator_holder(segment_manager_base *mngr) : alloc(mngr) {} allocator_type alloc; bucket_type init_bucket; }; }; #endif //#ifndef BOOST_INTERPROCESS_DOXYGEN_INVOKED //!Index type based in boost::intrusive::set. //!Just derives from boost::intrusive::set //!and defines the interface needed by managed memory segments template class iunordered_set_index //Derive class from map specialization : private iunordered_set_index_aux::allocator_holder , public iunordered_set_index_aux::index_t { #if !defined(BOOST_INTERPROCESS_DOXYGEN_INVOKED) typedef iunordered_set_index_aux index_aux; typedef typename index_aux::index_t index_type; typedef typename MapConfig:: intrusive_compare_key_type intrusive_compare_key_type; typedef typename index_aux::equal_function equal_function; typedef typename index_aux::hash_function hash_function; typedef typename MapConfig::char_type char_type; typedef typename iunordered_set_index_aux::allocator_type allocator_type; typedef typename iunordered_set_index_aux::allocator_holder allocator_holder; #endif //#ifndef BOOST_INTERPROCESS_DOXYGEN_INVOKED public: typedef typename index_type::iterator iterator; typedef typename index_type::const_iterator const_iterator; typedef typename index_type::insert_commit_data insert_commit_data; typedef typename index_type::value_type value_type; typedef typename index_type::bucket_ptr bucket_ptr; typedef typename index_type::bucket_type bucket_type; typedef typename index_type::bucket_traits bucket_traits; typedef typename index_type::size_type size_type; typedef typename index_type::difference_type difference_type; #if !defined(BOOST_INTERPROCESS_DOXYGEN_INVOKED) private: typedef typename index_aux:: segment_manager_base segment_manager_base; static const std::size_t InitBufferSize = 64; static bucket_ptr create_buckets(allocator_type &alloc, size_type num) { num = index_type::suggested_upper_bucket_count(num); bucket_ptr buckets = alloc.allocate(num); bucket_ptr buckets_init = buckets; for(size_type i = 0; i < num; ++i){ ::new(to_raw_pointer(buckets_init++), boost_container_new_t())bucket_type(); } return buckets; } static size_type shrink_buckets ( bucket_ptr buckets, size_type old_size , allocator_type &alloc, size_type new_size) { if(old_size <= new_size ) return old_size; size_type received_size = new_size; if(!alloc.allocation_command (boost::interprocess::try_shrink_in_place | boost::interprocess::nothrow_allocation, old_size, received_size, buckets)){ return old_size; } for( bucket_type *p = ipcdetail::to_raw_pointer(buckets) + received_size , *pend = ipcdetail::to_raw_pointer(buckets) + old_size ; p != pend ; ++p){ p->~bucket_type(); } bucket_ptr shunk_p = alloc.allocation_command (boost::interprocess::shrink_in_place | boost::interprocess::nothrow_allocation, received_size, received_size, buckets); BOOST_ASSERT(buckets == shunk_p); (void)shunk_p; bucket_ptr buckets_init = buckets + difference_type(received_size); for(size_type i = 0; i < (old_size - received_size); ++i){ to_raw_pointer(buckets_init++)->~bucket_type(); } return received_size; } static bucket_ptr expand_or_create_buckets ( bucket_ptr old_buckets, const size_type old_num , allocator_type &alloc, const size_type new_num) { size_type received_size = new_num; bucket_ptr reuse(old_buckets); bucket_ptr ret = alloc.allocation_command (boost::interprocess::expand_fwd | boost::interprocess::allocate_new, new_num, received_size, reuse); if(ret == old_buckets){ bucket_ptr buckets_init = old_buckets + difference_type(old_num); for(size_type i = 0; i < (new_num - old_num); ++i){ ::new(to_raw_pointer(buckets_init++), boost_container_new_t())bucket_type(); } } else{ bucket_ptr buckets_init = ret; for(size_type i = 0; i < new_num; ++i){ ::new(to_raw_pointer(buckets_init++), boost_container_new_t())bucket_type(); } } return ret; } static void destroy_buckets (allocator_type &alloc, bucket_ptr buckets, size_type num) { bucket_ptr buckets_destroy = buckets; for(size_type i = 0; i < num; ++i){ to_raw_pointer(buckets_destroy++)->~bucket_type(); } alloc.deallocate(buckets, num); } iunordered_set_index* get_this_pointer() { return this; } #endif //#ifndef BOOST_INTERPROCESS_DOXYGEN_INVOKED public: //!Constructor. Takes a pointer to the //!segment manager. Can throw iunordered_set_index(segment_manager_base *mngr) : allocator_holder(mngr) , index_type(bucket_traits(&get_this_pointer()->init_bucket, 1)) {} ~iunordered_set_index() { index_type::clear(); if(index_type::bucket_pointer() != bucket_ptr(&this->init_bucket)){ destroy_buckets(this->alloc, index_type::bucket_pointer(), index_type::bucket_count()); } } //!This reserves memory to optimize the insertion of n //!elements in the index void reserve(size_type new_n) { //Let's maintain a 1.0f load factor size_type old_n = this->bucket_count(); if(new_n <= old_n) return; bucket_ptr old_p = this->bucket_pointer(); new_n = index_type::suggested_upper_bucket_count(new_n); bucket_ptr new_p; //This can throw BOOST_TRY{ if(old_p != bucket_ptr(&this->init_bucket)) new_p = expand_or_create_buckets(old_p, old_n, this->alloc, new_n); else new_p = create_buckets(this->alloc, new_n); } BOOST_CATCH(...){ return; } BOOST_CATCH_END //Rehashing does not throw, since neither the hash nor the //comparison function can throw this->rehash(bucket_traits(new_p, new_n)); if(new_p != old_p && old_p != bucket_ptr(&this->init_bucket)){ destroy_buckets(this->alloc, old_p, old_n); } } //!This tries to free unused memory //!previously allocated. void shrink_to_fit() { size_type cur_size = this->size(); size_type cur_count = this->bucket_count(); bucket_ptr old_p = this->bucket_pointer(); if(!this->size() && old_p != bucket_ptr(&this->init_bucket)){ this->rehash(bucket_traits(bucket_ptr(&this->init_bucket), 1)); destroy_buckets(this->alloc, old_p, cur_count); } else{ size_type sug_count = 0; //gcc warning sug_count = index_type::suggested_upper_bucket_count(cur_size); if(sug_count >= cur_count) return; BOOST_TRY{ shrink_buckets(old_p, cur_count, this->alloc, sug_count); } BOOST_CATCH(...){ return; } BOOST_CATCH_END //Rehashing does not throw, since neither the hash nor the //comparison function can throw this->rehash(bucket_traits(old_p, sug_count)); } } iterator find(const intrusive_compare_key_type &key) { return index_type::find(key, hash_function(), equal_function()); } const_iterator find(const intrusive_compare_key_type &key) const { return index_type::find(key, hash_function(), equal_function()); } std::pairinsert_check (const intrusive_compare_key_type &key, insert_commit_data &commit_data) { return index_type::insert_check(key, hash_function(), equal_function(), commit_data); } iterator insert_commit(value_type &val, insert_commit_data &commit_data) { iterator it = index_type::insert_commit(val, commit_data); size_type cur_size = this->size(); if(cur_size > this->bucket_count()){ BOOST_TRY{ this->reserve(cur_size); } BOOST_CATCH(...){ //Strong guarantee: if something goes wrong //we should remove the insertion. // //We can use the iterator because the hash function //can't throw and this means that "reserve" will //throw only because of the memory allocation: //the iterator has not been invalidated. index_type::erase(it); BOOST_RETHROW } BOOST_CATCH_END } return it; } }; #if !defined(BOOST_INTERPROCESS_DOXYGEN_INVOKED) //!Trait class to detect if an index is an intrusive //!index template struct is_intrusive_index > { static const bool value = true; }; #endif //#ifndef BOOST_INTERPROCESS_DOXYGEN_INVOKED }} //namespace boost { namespace interprocess { #include #endif //#ifndef BOOST_INTERPROCESS_IUNORDERED_SET_INDEX_HPP