From 3a1c47566f2a1aa1a47e1cef9c213c915d35ae03 Mon Sep 17 00:00:00 2001 From: Christian Mazakas Date: Tue, 16 Aug 2022 14:34:58 -0700 Subject: [PATCH] Update code to be valid when the internal `buckets_` data member is moved-from --- include/boost/unordered/detail/fca.hpp | 18 +++++-- .../boost/unordered/detail/implementation.hpp | 49 +++++++++++-------- include/boost/unordered/unordered_map.hpp | 8 +++ include/boost/unordered/unordered_set.hpp | 8 +++ 4 files changed, 58 insertions(+), 25 deletions(-) diff --git a/include/boost/unordered/detail/fca.hpp b/include/boost/unordered/detail/fca.hpp index 19fafe7398..a1d14d9571 100644 --- a/include/boost/unordered/detail/fca.hpp +++ b/include/boost/unordered/detail/fca.hpp @@ -646,7 +646,7 @@ namespace boost { size_type bucket_count() const { return size_; } - iterator begin() const { return ++at(size_); } + iterator begin() const { return size_ == 0 ? end() : ++at(size_); } iterator end() const { @@ -660,6 +660,10 @@ namespace boost { local_iterator begin(size_type n) const { + if (size_ == 0) { + return this->end(n); + } + return local_iterator( (buckets + static_cast(n))->next); } @@ -670,12 +674,16 @@ namespace boost { iterator at(size_type n) const { - std::size_t const N = group::N; + if (size_ > 0) { + std::size_t const N = group::N; - iterator pbg(buckets + static_cast(n), - groups + static_cast(n / N)); + iterator pbg(buckets + static_cast(n), + groups + static_cast(n / N)); - return pbg; + return pbg; + } else { + return this->end(); + } } span raw() diff --git a/include/boost/unordered/detail/implementation.hpp b/include/boost/unordered/detail/implementation.hpp index 2cc27c5d4a..373236754f 100644 --- a/include/boost/unordered/detail/implementation.hpp +++ b/include/boost/unordered/detail/implementation.hpp @@ -2054,12 +2054,14 @@ namespace boost { std::size_t bucket_size(std::size_t index) const { - bucket_iterator itb = buckets_.at(index); - node_pointer n = itb->next; std::size_t count = 0; - while (n) { - ++count; - n = n->next; + if (size_ > 0) { + bucket_iterator itb = buckets_.at(index); + node_pointer n = itb->next; + while (n) { + ++count; + n = n->next; + } } return count; } @@ -2420,11 +2422,14 @@ namespace boost { node_pointer find_node_impl( Key const& x, bucket_iterator itb) const { - key_equal const& pred = this->key_eq(); - node_pointer p = itb->next; - for (; p; p = p->next) { - if (pred(x, extractor::extract(p->value()))) { - break; + node_pointer p = node_pointer(); + if (itb != buckets_.end()) { + key_equal const& pred = this->key_eq(); + p = itb->next; + for (; p; p = p->next) { + if (pred(x, extractor::extract(p->value()))) { + break; + } } } return p; @@ -2453,11 +2458,13 @@ namespace boost { inline iterator transparent_find( Key const& k, Hash const& h, Pred const& pred) const { - std::size_t const key_hash = h(k); - bucket_iterator itb = buckets_.at(buckets_.position(key_hash)); - for (node_pointer p = itb->next; p; p = p->next) { - if (BOOST_LIKELY(pred(k, extractor::extract(p->value())))) { - return iterator(p, itb); + if (size_ > 0) { + std::size_t const key_hash = h(k); + bucket_iterator itb = buckets_.at(buckets_.position(key_hash)); + for (node_pointer p = itb->next; p; p = p->next) { + if (BOOST_LIKELY(pred(k, extractor::extract(p->value())))) { + return iterator(p, itb); + } } } @@ -2467,11 +2474,13 @@ namespace boost { template node_pointer* find_prev(Key const& key, bucket_iterator itb) { - key_equal pred = this->key_eq(); - for (node_pointer* pp = boost::addressof(itb->next); *pp; - pp = boost::addressof((*pp)->next)) { - if (pred(key, extractor::extract((*pp)->value()))) { - return pp; + if (size_ > 0) { + key_equal pred = this->key_eq(); + for (node_pointer* pp = boost::addressof(itb->next); *pp; + pp = boost::addressof((*pp)->next)) { + if (pred(key, extractor::extract((*pp)->value()))) { + return pp; + } } } typedef node_pointer* node_pointer_pointer; diff --git a/include/boost/unordered/unordered_map.hpp b/include/boost/unordered/unordered_map.hpp index 97908fb65b..66ba20e597 100644 --- a/include/boost/unordered/unordered_map.hpp +++ b/include/boost/unordered/unordered_map.hpp @@ -2069,6 +2069,10 @@ namespace boost { template float unordered_map::load_factor() const BOOST_NOEXCEPT { + if (table_.size_ == 0) { + return 0.0; + } + BOOST_ASSERT(table_.bucket_count() != 0); return static_cast(table_.size_) / static_cast(table_.bucket_count()); @@ -2506,6 +2510,10 @@ namespace boost { template float unordered_multimap::load_factor() const BOOST_NOEXCEPT { + if (table_.size_ == 0) { + return 0.0; + } + BOOST_ASSERT(table_.bucket_count() != 0); return static_cast(table_.size_) / static_cast(table_.bucket_count()); diff --git a/include/boost/unordered/unordered_set.hpp b/include/boost/unordered/unordered_set.hpp index 8721a68a08..d921eec712 100644 --- a/include/boost/unordered/unordered_set.hpp +++ b/include/boost/unordered/unordered_set.hpp @@ -1586,6 +1586,10 @@ namespace boost { template float unordered_set::load_factor() const BOOST_NOEXCEPT { + if (table_.size_ == 0) { + return 0.0; + } + BOOST_ASSERT(table_.bucket_count() != 0); return static_cast(table_.size_) / static_cast(table_.bucket_count()); @@ -1986,6 +1990,10 @@ namespace boost { template float unordered_multiset::load_factor() const BOOST_NOEXCEPT { + if (table_.size_ == 0) { + return 0.0; + } + BOOST_ASSERT(table_.bucket_count() != 0); return static_cast(table_.size_) / static_cast(table_.bucket_count());