diff options
author | Evan Brown <ezb@google.com> | 2022-11-21 08:35:02 -0800 |
---|---|---|
committer | Copybara-Service <copybara-worker@google.com> | 2022-11-21 08:35:39 -0800 |
commit | d081b629b7c201e701911cfb4e0c99735c21c331 (patch) | |
tree | b4671ba8aa849b9a0b0573c36965fa1fe4c96552 /absl/container | |
parent | 76b7618369826ba175beed1735b6b072f5be15d8 (diff) |
Improve error messages when comparing btree iterators when generations are enabled.
- Add assertions that the iterators haven't been invalidated.
- Also refactor some generation-related code to define the functions inside ABSL_BTREE_ENABLE_GENERATIONS ifdef/else branches.
PiperOrigin-RevId: 489988637
Change-Id: I34d32ed2e27cf89f7f8bb5d9c1f6770bb40b8794
Diffstat (limited to 'absl/container')
-rw-r--r-- | absl/container/btree_test.cc | 8 | ||||
-rw-r--r-- | absl/container/internal/btree.h | 47 |
2 files changed, 35 insertions, 20 deletions
diff --git a/absl/container/btree_test.cc b/absl/container/btree_test.cc index 28dda8a6..cc763b29 100644 --- a/absl/container/btree_test.cc +++ b/absl/container/btree_test.cc @@ -3167,6 +3167,14 @@ TEST(Btree, InvalidIteratorUse) { set.erase(1); EXPECT_DEATH(*it, "invalidated iterator"); } + { + absl::btree_set<int> set; + for (int i = 0; i < 10; ++i) set.insert(i); + auto it = set.insert(20).first; + set.insert(30); + EXPECT_DEATH(void(it == set.begin()), "invalidated iterator"); + EXPECT_DEATH(void(set.begin() == it), "invalidated iterator"); + } } #endif diff --git a/absl/container/internal/btree.h b/absl/container/internal/btree.h index ab75afb4..ef4f5dc3 100644 --- a/absl/container/internal/btree.h +++ b/absl/container/internal/btree.h @@ -1084,16 +1084,16 @@ class btree_iterator { } bool operator==(const iterator &other) const { - return Equals(other.node_, other.position_); + return Equals(other); } bool operator==(const const_iterator &other) const { - return Equals(other.node_, other.position_); + return Equals(other); } bool operator!=(const iterator &other) const { - return !Equals(other.node_, other.position_); + return !Equals(other); } bool operator!=(const const_iterator &other) const { - return !Equals(other.node_, other.position_); + return !Equals(other); } // Returns n such that n calls to ++other yields *this. @@ -1172,17 +1172,19 @@ class btree_iterator { #endif } - bool Equals(const node_type *other_node, int other_position) const { - ABSL_HARDENING_ASSERT(((node_ == nullptr && other_node == nullptr) || - (node_ != nullptr && other_node != nullptr)) && + bool Equals(const const_iterator other) const { + ABSL_HARDENING_ASSERT(((node_ == nullptr && other.node_ == nullptr) || + (node_ != nullptr && other.node_ != nullptr)) && "Comparing default-constructed iterator with " "non-default-constructed iterator."); // Note: we use assert instead of ABSL_HARDENING_ASSERT here because this // changes the complexity of Equals from O(1) to O(log(N) + log(M)) where - // N/M are sizes of the containers containing node_/other_node. - assert(AreNodesFromSameContainer(node_, other_node) && + // N/M are sizes of the containers containing node_/other.node_. + assert(AreNodesFromSameContainer(node_, other.node_) && "Comparing iterators from different containers."); - return node_ == other_node && position_ == other_position; + assert_valid_generation(); + other.assert_valid_generation(); + return node_ == other.node_ && position_ == other.position_; } bool IsEndIterator() const { @@ -1219,14 +1221,6 @@ class btree_iterator { } void decrement_slow(); - // Updates the generation. For use internally right before we return an - // iterator to the user. - void update_generation() { -#ifdef ABSL_BTREE_ENABLE_GENERATIONS - if (node_ != nullptr) generation_ = node_->generation(); -#endif - } - const key_type &key() const { return node_->key(static_cast<size_type>(position_)); } @@ -1234,16 +1228,29 @@ class btree_iterator { return node_->slot(static_cast<size_type>(position_)); } - void assert_valid_generation() const { +// TODO(b/207380122): use an ifdef to select a base class instead of ifdefs +// inside the class. #ifdef ABSL_BTREE_ENABLE_GENERATIONS + // Updates the generation. For use internally right before we return an + // iterator to the user. + void update_generation() { + if (node_ != nullptr) generation_ = node_->generation(); + } + uint32_t generation() const { return generation_; } + + void assert_valid_generation() const { if (node_ != nullptr && node_->generation() != generation_) { ABSL_INTERNAL_LOG( FATAL, "Attempting to use an invalidated iterator. The corresponding b-tree " "container has been mutated since this iterator was constructed."); } -#endif } +#else + void update_generation() {} + uint32_t generation() const { return 0; } + void assert_valid_generation() const {} +#endif // The node in the tree the iterator is pointing at. Node *node_; |