Export of internal Abseil changes

--
dcd4d95f6201dc5781a3a374be8eb10c812fd98a by Derek Mauro <dmauro@google.com>:

Add -Wundef to GCC warnings

PiperOrigin-RevId: 322388155

--
b030746368262aff6bc487f5525bcd9b32d18ebb by Abseil Team <absl-team@google.com>:

Google internal clean-up.

PiperOrigin-RevId: 322381901

--
18e4cfcd50730c493cfc0cf1e127e57c186ce90b by Evan Brown <ezb@google.com>:

Rollback b-tree erase simplification change.

PiperOrigin-RevId: 322368252

--
d15431c52fa7ccb25ffbd967fd11f8f58246d48a by Abseil Team <absl-team@google.com>:

Update MOCK_METHOD (new format) in memory/memory_test.cc

PiperOrigin-RevId: 322208282
GitOrigin-RevId: dcd4d95f6201dc5781a3a374be8eb10c812fd98a
Change-Id: I3a900b4993f86bdd1c9597819c7a0e6e1759eda3
pull/746/head
Abseil Team 5 years ago committed by Andy Getz
parent 3c2bed2e77
commit 41a6263fd0
  1. 2
      absl/base/policy_checks.h
  2. 203
      absl/container/internal/btree.h
  3. 7
      absl/container/internal/container_memory.h
  4. 1
      absl/copts/GENERATED_AbseilCopts.cmake
  5. 1
      absl/copts/GENERATED_copts.bzl
  6. 1
      absl/copts/copts.py
  7. 36
      absl/memory/memory_test.cc

@ -41,7 +41,7 @@
#endif #endif
// ----------------------------------------------------------------------------- // -----------------------------------------------------------------------------
// Compiler Check // Toolchain Check
// ----------------------------------------------------------------------------- // -----------------------------------------------------------------------------
// We support MSVC++ 14.0 update 2 and later. // We support MSVC++ 14.0 update 2 and later.

@ -255,6 +255,10 @@ struct common_params {
static void move(Alloc *alloc, slot_type *src, slot_type *dest) { static void move(Alloc *alloc, slot_type *src, slot_type *dest) {
slot_policy::move(alloc, src, dest); slot_policy::move(alloc, src, dest);
} }
static void move(Alloc *alloc, slot_type *first, slot_type *last,
slot_type *result) {
slot_policy::move(alloc, first, last, result);
}
}; };
// A parameters structure for holding the type parameters for a btree_map. // A parameters structure for holding the type parameters for a btree_map.
@ -332,6 +336,13 @@ struct set_slot_policy {
static void move(Alloc * /*alloc*/, slot_type *src, slot_type *dest) { static void move(Alloc * /*alloc*/, slot_type *src, slot_type *dest) {
*dest = std::move(*src); *dest = std::move(*src);
} }
template <typename Alloc>
static void move(Alloc *alloc, slot_type *first, slot_type *last,
slot_type *result) {
for (slot_type *src = first, *dest = result; src != last; ++src, ++dest)
move(alloc, src, dest);
}
}; };
// A parameters structure for holding the type parameters for a btree_set. // A parameters structure for holding the type parameters for a btree_set.
@ -748,10 +759,14 @@ class btree_node {
template <typename... Args> template <typename... Args>
void emplace_value(size_type i, allocator_type *alloc, Args &&... args); void emplace_value(size_type i, allocator_type *alloc, Args &&... args);
// Removes the values at positions [i, i + to_erase), shifting all existing // Removes the value at position i, shifting all existing values and children
// values and children after that range to the left by to_erase. Clears all // at positions > i to the left by 1.
// children between [i, i + to_erase). void remove_value(int i, allocator_type *alloc);
void remove_values(field_type i, field_type to_erase, allocator_type *alloc);
// Removes the values at positions [i, i + to_erase), shifting all values
// after that range to the left by to_erase. Does not change children at all.
void remove_values_ignore_children(int i, int to_erase,
allocator_type *alloc);
// Rebalances a node with its right sibling. // Rebalances a node with its right sibling.
void rebalance_right_to_left(int to_move, btree_node *right, void rebalance_right_to_left(int to_move, btree_node *right,
@ -763,7 +778,7 @@ class btree_node {
void split(int insert_position, btree_node *dest, allocator_type *alloc); void split(int insert_position, btree_node *dest, allocator_type *alloc);
// Merges a node with its right sibling, moving all of the values and the // Merges a node with its right sibling, moving all of the values and the
// delimiting key in the parent node onto itself, and deleting the src node. // delimiting key in the parent node onto itself.
void merge(btree_node *src, allocator_type *alloc); void merge(btree_node *src, allocator_type *alloc);
// Node allocation/deletion routines. // Node allocation/deletion routines.
@ -784,27 +799,9 @@ class btree_node {
absl::container_internal::SanitizerPoisonMemoryRegion( absl::container_internal::SanitizerPoisonMemoryRegion(
&mutable_child(start()), (kNodeValues + 1) * sizeof(btree_node *)); &mutable_child(start()), (kNodeValues + 1) * sizeof(btree_node *));
} }
void destroy(allocator_type *alloc) {
// Deletes a node and all of its children. for (int i = start(); i < finish(); ++i) {
// TODO(ezb): don't use recursion here to avoid potential stack overflows. value_destroy(i, alloc);
static void clear_and_delete(btree_node *node, allocator_type *alloc) {
const field_type start = node->start();
const field_type finish = node->finish();
for (field_type i = start; i < finish; ++i) {
node->value_destroy(i, alloc);
}
if (node->leaf()) {
absl::container_internal::Deallocate<Alignment()>(
alloc, node, LeafSize(node->max_count()));
} else {
// If the node is empty, then there are no children so don't try clearing.
if (start < finish) {
for (field_type i = start; i <= finish; ++i) {
clear_and_delete(node->child(i), alloc);
}
}
absl::container_internal::Deallocate<Alignment()>(alloc, node,
InternalSize());
} }
} }
@ -1426,8 +1423,25 @@ class btree {
} }
// Deletion helper routines. // Deletion helper routines.
void erase_same_node(iterator begin, iterator end);
iterator erase_from_leaf_node(iterator begin, size_type to_erase);
iterator rebalance_after_delete(iterator iter); iterator rebalance_after_delete(iterator iter);
// Deallocates a node of a certain size in bytes using the allocator.
void deallocate(const size_type size, node_type *node) {
absl::container_internal::Deallocate<node_type::Alignment()>(
mutable_allocator(), node, size);
}
void delete_internal_node(node_type *node) {
node->destroy(mutable_allocator());
deallocate(node_type::InternalSize(), node);
}
void delete_leaf_node(node_type *node) {
node->destroy(mutable_allocator());
deallocate(node_type::LeafSize(node->max_count()), node);
}
// Rebalances or splits the node iter points to. // Rebalances or splits the node iter points to.
void rebalance_or_split(iterator *iter); void rebalance_or_split(iterator *iter);
@ -1496,6 +1510,9 @@ class btree {
template <typename K> template <typename K>
iterator internal_find(const K &key) const; iterator internal_find(const K &key) const;
// Deletes a node and all of its children.
void internal_clear(node_type *node);
// Verifies the tree structure of node. // Verifies the tree structure of node.
int internal_verify(const node_type *node, const key_type *lo, int internal_verify(const node_type *node, const key_type *lo,
const key_type *hi) const; const key_type *hi) const;
@ -1563,29 +1580,26 @@ inline void btree_node<P>::emplace_value(const size_type i,
} }
template <typename P> template <typename P>
inline void btree_node<P>::remove_values(const field_type i, inline void btree_node<P>::remove_value(const int i, allocator_type *alloc) {
const field_type to_erase, if (!leaf() && finish() > i + 1) {
allocator_type *alloc) { assert(child(i + 1)->count() == 0);
// Transfer values after the removed range into their new places. for (size_type j = i + 1; j < finish(); ++j) {
const field_type orig_finish = finish(); set_child(j, child(j + 1));
const field_type src_i = i + to_erase; }
for (field_type j = i; j < src_i; ++j) { clear_child(finish());
value_destroy(j, alloc);
} }
transfer_n(orig_finish - src_i, i, src_i, this, alloc);
if (!leaf()) { remove_values_ignore_children(i, /*to_erase=*/1, alloc);
// Delete all children between begin and end. }
for (field_type j = 0; j < to_erase; ++j) {
clear_and_delete(child(i + j + 1), alloc); template <typename P>
} inline void btree_node<P>::remove_values_ignore_children(
// Rotate children after end into new positions. const int i, const int to_erase, allocator_type *alloc) {
for (field_type j = i + to_erase + 1; j <= orig_finish; ++j) { params_type::move(alloc, slot(i + to_erase), finish_slot(), slot(i));
set_child(j - to_erase, child(j)); for (int j = finish() - to_erase; j < finish(); ++j) {
clear_child(j); value_destroy(j, alloc);
}
} }
set_finish(orig_finish - to_erase); set_finish(finish() - to_erase);
} }
template <typename P> template <typename P>
@ -1737,8 +1751,8 @@ void btree_node<P>::merge(btree_node *src, allocator_type *alloc) {
set_finish(start() + 1 + count() + src->count()); set_finish(start() + 1 + count() + src->count());
src->set_finish(src->start()); src->set_finish(src->start());
// Remove the value on the parent node and delete the src node. // Remove the value on the parent node.
parent()->remove_values(position(), /*to_erase=*/1, alloc); parent()->remove_value(position(), alloc);
} }
//// ////
@ -2020,7 +2034,7 @@ auto btree<P>::erase(iterator iter) -> iterator {
bool internal_delete = false; bool internal_delete = false;
if (!iter.node->leaf()) { if (!iter.node->leaf()) {
// Deletion of a value on an internal node. First, move the largest value // Deletion of a value on an internal node. First, move the largest value
// from our left child here, then delete that position (in remove_values() // from our left child here, then delete that position (in remove_value()
// below). We can get to the largest value from our left child by // below). We can get to the largest value from our left child by
// decrementing iter. // decrementing iter.
iterator internal_iter(iter); iterator internal_iter(iter);
@ -2032,7 +2046,7 @@ auto btree<P>::erase(iterator iter) -> iterator {
} }
// Delete the key from the leaf. // Delete the key from the leaf.
iter.node->remove_values(iter.position, /*to_erase=*/1, mutable_allocator()); iter.node->remove_value(iter.position, mutable_allocator());
--size_; --size_;
// We want to return the next value after the one we just erased. If we // We want to return the next value after the one we just erased. If we
@ -2107,9 +2121,7 @@ auto btree<P>::erase_range(iterator begin, iterator end)
} }
if (begin.node == end.node) { if (begin.node == end.node) {
assert(end.position > begin.position); erase_same_node(begin, end);
begin.node->remove_values(begin.position, end.position - begin.position,
mutable_allocator());
size_ -= count; size_ -= count;
return {count, rebalance_after_delete(begin)}; return {count, rebalance_after_delete(begin)};
} }
@ -2119,11 +2131,8 @@ auto btree<P>::erase_range(iterator begin, iterator end)
if (begin.node->leaf()) { if (begin.node->leaf()) {
const size_type remaining_to_erase = size_ - target_size; const size_type remaining_to_erase = size_ - target_size;
const size_type remaining_in_node = begin.node->finish() - begin.position; const size_type remaining_in_node = begin.node->finish() - begin.position;
const size_type to_erase = begin = erase_from_leaf_node(
(std::min)(remaining_to_erase, remaining_in_node); begin, (std::min)(remaining_to_erase, remaining_in_node));
begin.node->remove_values(begin.position, to_erase, mutable_allocator());
size_ -= to_erase;
begin = rebalance_after_delete(begin);
} else { } else {
begin = erase(begin); begin = erase(begin);
} }
@ -2131,6 +2140,51 @@ auto btree<P>::erase_range(iterator begin, iterator end)
return {count, begin}; return {count, begin};
} }
template <typename P>
void btree<P>::erase_same_node(iterator begin, iterator end) {
assert(begin.node == end.node);
assert(end.position > begin.position);
node_type *node = begin.node;
size_type to_erase = end.position - begin.position;
if (!node->leaf()) {
// Delete all children between begin and end.
for (size_type i = 0; i < to_erase; ++i) {
internal_clear(node->child(begin.position + i + 1));
}
// Rotate children after end into new positions.
for (size_type i = begin.position + to_erase + 1; i <= node->finish();
++i) {
node->set_child(i - to_erase, node->child(i));
node->clear_child(i);
}
}
node->remove_values_ignore_children(begin.position, to_erase,
mutable_allocator());
// Do not need to update rightmost_, because
// * either end == this->end(), and therefore node == rightmost_, and still
// exists
// * or end != this->end(), and therefore rightmost_ hasn't been erased, since
// it wasn't covered in [begin, end)
}
template <typename P>
auto btree<P>::erase_from_leaf_node(iterator begin, size_type to_erase)
-> iterator {
node_type *node = begin.node;
assert(node->leaf());
assert(node->finish() > begin.position);
assert(begin.position + to_erase <= node->finish());
node->remove_values_ignore_children(begin.position, to_erase,
mutable_allocator());
size_ -= to_erase;
return rebalance_after_delete(begin);
}
template <typename P> template <typename P>
template <typename K> template <typename K>
auto btree<P>::erase_unique(const K &key) -> size_type { auto btree<P>::erase_unique(const K &key) -> size_type {
@ -2159,7 +2213,7 @@ auto btree<P>::erase_multi(const K &key) -> size_type {
template <typename P> template <typename P>
void btree<P>::clear() { void btree<P>::clear() {
if (!empty()) { if (!empty()) {
node_type::clear_and_delete(root(), mutable_allocator()); internal_clear(root());
} }
mutable_root() = EmptyNode(); mutable_root() = EmptyNode();
rightmost_ = EmptyNode(); rightmost_ = EmptyNode();
@ -2300,7 +2354,12 @@ void btree<P>::rebalance_or_split(iterator *iter) {
template <typename P> template <typename P>
void btree<P>::merge_nodes(node_type *left, node_type *right) { void btree<P>::merge_nodes(node_type *left, node_type *right) {
left->merge(right, mutable_allocator()); left->merge(right, mutable_allocator());
if (rightmost_ == right) rightmost_ = left; if (right->leaf()) {
if (rightmost_ == right) rightmost_ = left;
delete_leaf_node(right);
} else {
delete_internal_node(right);
}
} }
template <typename P> template <typename P>
@ -2357,20 +2416,20 @@ bool btree<P>::try_merge_or_rebalance(iterator *iter) {
template <typename P> template <typename P>
void btree<P>::try_shrink() { void btree<P>::try_shrink() {
node_type *orig_root = root(); if (root()->count() > 0) {
if (orig_root->count() > 0) {
return; return;
} }
// Deleted the last item on the root node, shrink the height of the tree. // Deleted the last item on the root node, shrink the height of the tree.
if (orig_root->leaf()) { if (root()->leaf()) {
assert(size() == 0); assert(size() == 0);
delete_leaf_node(root());
mutable_root() = rightmost_ = EmptyNode(); mutable_root() = rightmost_ = EmptyNode();
} else { } else {
node_type *child = orig_root->start_child(); node_type *child = root()->start_child();
child->make_root(); child->make_root();
delete_internal_node(root());
mutable_root() = child; mutable_root() = child;
} }
node_type::clear_and_delete(orig_root, mutable_allocator());
} }
template <typename P> template <typename P>
@ -2415,7 +2474,7 @@ inline auto btree<P>::internal_emplace(iterator iter, Args &&... args)
old_root->start(), old_root, alloc); old_root->start(), old_root, alloc);
new_root->set_finish(old_root->finish()); new_root->set_finish(old_root->finish());
old_root->set_finish(old_root->start()); old_root->set_finish(old_root->start());
node_type::clear_and_delete(old_root, alloc); delete_leaf_node(old_root);
mutable_root() = rightmost_ = new_root; mutable_root() = rightmost_ = new_root;
} else { } else {
rebalance_or_split(&iter); rebalance_or_split(&iter);
@ -2518,6 +2577,18 @@ auto btree<P>::internal_find(const K &key) const -> iterator {
return {nullptr, 0}; return {nullptr, 0};
} }
template <typename P>
void btree<P>::internal_clear(node_type *node) {
if (!node->leaf()) {
for (int i = node->start(); i <= node->finish(); ++i) {
internal_clear(node->child(i));
}
delete_internal_node(node);
} else {
delete_leaf_node(node);
}
}
template <typename P> template <typename P>
int btree<P>::internal_verify(const node_type *node, const key_type *lo, int btree<P>::internal_verify(const node_type *node, const key_type *lo,
const key_type *hi) const { const key_type *hi) const {

@ -429,6 +429,13 @@ struct map_slot_policy {
std::move(src->value)); std::move(src->value));
} }
} }
template <class Allocator>
static void move(Allocator* alloc, slot_type* first, slot_type* last,
slot_type* result) {
for (slot_type *src = first, *dest = result; src != last; ++src, ++dest)
move(alloc, src, dest);
}
}; };
} // namespace container_internal } // namespace container_internal

@ -81,6 +81,7 @@ list(APPEND ABSL_GCC_FLAGS
"-Wmissing-declarations" "-Wmissing-declarations"
"-Woverlength-strings" "-Woverlength-strings"
"-Wpointer-arith" "-Wpointer-arith"
"-Wundef"
"-Wunused-local-typedefs" "-Wunused-local-typedefs"
"-Wunused-result" "-Wunused-result"
"-Wvarargs" "-Wvarargs"

@ -82,6 +82,7 @@ ABSL_GCC_FLAGS = [
"-Wmissing-declarations", "-Wmissing-declarations",
"-Woverlength-strings", "-Woverlength-strings",
"-Wpointer-arith", "-Wpointer-arith",
"-Wundef",
"-Wunused-local-typedefs", "-Wunused-local-typedefs",
"-Wunused-result", "-Wunused-result",
"-Wvarargs", "-Wvarargs",

@ -128,6 +128,7 @@ COPT_VARS = {
"-Wmissing-declarations", "-Wmissing-declarations",
"-Woverlength-strings", "-Woverlength-strings",
"-Wpointer-arith", "-Wpointer-arith",
"-Wundef",
"-Wunused-local-typedefs", "-Wunused-local-typedefs",
"-Wunused-result", "-Wunused-result",
"-Wvarargs", "-Wvarargs",

@ -17,6 +17,7 @@
#include "absl/memory/memory.h" #include "absl/memory/memory.h"
#include <sys/types.h> #include <sys/types.h>
#include <cstddef> #include <cstddef>
#include <memory> #include <memory>
#include <string> #include <string>
@ -36,10 +37,10 @@ using ::testing::Return;
// been called, via the instance_count variable. // been called, via the instance_count variable.
class DestructorVerifier { class DestructorVerifier {
public: public:
DestructorVerifier() { ++instance_count_; } DestructorVerifier() { ++instance_count_; }
DestructorVerifier(const DestructorVerifier&) = delete; DestructorVerifier(const DestructorVerifier&) = delete;
DestructorVerifier& operator=(const DestructorVerifier&) = delete; DestructorVerifier& operator=(const DestructorVerifier&) = delete;
~DestructorVerifier() { --instance_count_; } ~DestructorVerifier() { --instance_count_; }
// The number of instances of this class currently active. // The number of instances of this class currently active.
static int instance_count() { return instance_count_; } static int instance_count() { return instance_count_; }
@ -156,9 +157,7 @@ struct ArrayWatch {
allocs().push_back(n); allocs().push_back(n);
return ::operator new[](n); return ::operator new[](n);
} }
void operator delete[](void* p) { void operator delete[](void* p) { return ::operator delete[](p); }
return ::operator delete[](p);
}
static std::vector<size_t>& allocs() { static std::vector<size_t>& allocs() {
static auto& v = *new std::vector<size_t>; static auto& v = *new std::vector<size_t>;
return v; return v;
@ -171,8 +170,7 @@ TEST(Make_UniqueTest, Array) {
ArrayWatch::allocs().clear(); ArrayWatch::allocs().clear();
auto p = absl::make_unique<ArrayWatch[]>(5); auto p = absl::make_unique<ArrayWatch[]>(5);
static_assert(std::is_same<decltype(p), static_assert(std::is_same<decltype(p), std::unique_ptr<ArrayWatch[]>>::value,
std::unique_ptr<ArrayWatch[]>>::value,
"unexpected return type"); "unexpected return type");
EXPECT_THAT(ArrayWatch::allocs(), ElementsAre(5 * sizeof(ArrayWatch))); EXPECT_THAT(ArrayWatch::allocs(), ElementsAre(5 * sizeof(ArrayWatch)));
} }
@ -181,7 +179,7 @@ TEST(Make_UniqueTest, NotAmbiguousWithStdMakeUnique) {
// Ensure that absl::make_unique is not ambiguous with std::make_unique. // Ensure that absl::make_unique is not ambiguous with std::make_unique.
// In C++14 mode, the below call to make_unique has both types as candidates. // In C++14 mode, the below call to make_unique has both types as candidates.
struct TakesStdType { struct TakesStdType {
explicit TakesStdType(const std::vector<int> &vec) {} explicit TakesStdType(const std::vector<int>& vec) {}
}; };
using absl::make_unique; using absl::make_unique;
(void)make_unique<TakesStdType>(std::vector<int>()); (void)make_unique<TakesStdType>(std::vector<int>());
@ -541,8 +539,8 @@ struct MinimalMockAllocator {
MinimalMockAllocator(const MinimalMockAllocator& other) MinimalMockAllocator(const MinimalMockAllocator& other)
: value(other.value) {} : value(other.value) {}
using value_type = TestValue; using value_type = TestValue;
MOCK_METHOD1(allocate, value_type*(size_t)); MOCK_METHOD(value_type*, allocate, (size_t));
MOCK_METHOD2(deallocate, void(value_type*, size_t)); MOCK_METHOD(void, deallocate, (value_type*, size_t));
int value; int value;
}; };
@ -579,13 +577,14 @@ struct FullMockAllocator {
explicit FullMockAllocator(int value) : value(value) {} explicit FullMockAllocator(int value) : value(value) {}
FullMockAllocator(const FullMockAllocator& other) : value(other.value) {} FullMockAllocator(const FullMockAllocator& other) : value(other.value) {}
using value_type = TestValue; using value_type = TestValue;
MOCK_METHOD1(allocate, value_type*(size_t)); MOCK_METHOD(value_type*, allocate, (size_t));
MOCK_METHOD2(allocate, value_type*(size_t, const void*)); MOCK_METHOD(value_type*, allocate, (size_t, const void*));
MOCK_METHOD2(construct, void(value_type*, int*)); MOCK_METHOD(void, construct, (value_type*, int*));
MOCK_METHOD1(destroy, void(value_type*)); MOCK_METHOD(void, destroy, (value_type*));
MOCK_CONST_METHOD0(max_size, size_t()); MOCK_METHOD(size_t, max_size, (),
MOCK_CONST_METHOD0(select_on_container_copy_construction, (const));
FullMockAllocator()); MOCK_METHOD(FullMockAllocator, select_on_container_copy_construction, (),
(const));
int value; int value;
}; };
@ -642,8 +641,7 @@ TEST(AllocatorNoThrowTest, CustomAllocator) {
struct CanThrowAllocator { struct CanThrowAllocator {
using is_nothrow = std::false_type; using is_nothrow = std::false_type;
}; };
struct UnspecifiedAllocator { struct UnspecifiedAllocator {};
};
EXPECT_TRUE(absl::allocator_is_nothrow<NoThrowAllocator>::value); EXPECT_TRUE(absl::allocator_is_nothrow<NoThrowAllocator>::value);
EXPECT_FALSE(absl::allocator_is_nothrow<CanThrowAllocator>::value); EXPECT_FALSE(absl::allocator_is_nothrow<CanThrowAllocator>::value);
EXPECT_FALSE(absl::allocator_is_nothrow<UnspecifiedAllocator>::value); EXPECT_FALSE(absl::allocator_is_nothrow<UnspecifiedAllocator>::value);

Loading…
Cancel
Save