summaryrefslogtreecommitdiff
path: root/absl/container
diff options
context:
space:
mode:
authorGravatar Evan Brown <ezb@google.com>2022-11-21 08:35:02 -0800
committerGravatar Copybara-Service <copybara-worker@google.com>2022-11-21 08:35:39 -0800
commitd081b629b7c201e701911cfb4e0c99735c21c331 (patch)
treeb4671ba8aa849b9a0b0573c36965fa1fe4c96552 /absl/container
parent76b7618369826ba175beed1735b6b072f5be15d8 (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.cc8
-rw-r--r--absl/container/internal/btree.h47
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_;