Skip to content

Commit

Permalink
Make PolicyTraits::transfer_uses_memcpy() true for node_hash_* tables.
Browse files Browse the repository at this point in the history
This should enable binary size savings for now and more efficiency improvements with small buffer optimization.

PiperOrigin-RevId: 564741270
Change-Id: Icf204d88256243eb60464439a52dd589d7a559cb
  • Loading branch information
ezbr authored and copybara-github committed Sep 12, 2023
1 parent f01b1b5 commit 5ae23ed
Show file tree
Hide file tree
Showing 6 changed files with 30 additions and 5 deletions.
2 changes: 2 additions & 0 deletions absl/container/BUILD.bazel
Original file line number Diff line number Diff line change
Expand Up @@ -499,6 +499,7 @@ cc_test(
linkopts = ABSL_DEFAULT_LINKOPTS,
deps = [
":common_policy_traits",
"//absl/base:config",
"@com_google_googletest//:gtest_main",
],
)
Expand Down Expand Up @@ -582,6 +583,7 @@ cc_test(
deps = [
":hash_policy_traits",
":node_slot_policy",
"//absl/base:config",
"@com_google_googletest//:gtest_main",
],
)
Expand Down
2 changes: 2 additions & 0 deletions absl/container/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -577,6 +577,7 @@ absl_cc_test(
${ABSL_TEST_COPTS}
DEPS
absl::common_policy_traits
absl::config
GTest::gmock_main
)

Expand Down Expand Up @@ -660,6 +661,7 @@ absl_cc_test(
COPTS
${ABSL_TEST_COPTS}
DEPS
absl::config
absl::hash_policy_traits
absl::node_slot_policy
GTest::gmock_main
Expand Down
6 changes: 4 additions & 2 deletions absl/container/internal/common_policy_traits.h
Original file line number Diff line number Diff line change
Expand Up @@ -93,11 +93,13 @@ struct common_policy_traits {
struct Rank0 : Rank1 {};

// Use auto -> decltype as an enabler.
// P::transfer returns std::true_type if transfer uses memcpy (e.g. in
// node_slot_policy).
template <class Alloc, class P = Policy>
static auto transfer_impl(Alloc* alloc, slot_type* new_slot,
slot_type* old_slot, Rank0)
-> decltype((void)P::transfer(alloc, new_slot, old_slot)) {
P::transfer(alloc, new_slot, old_slot);
-> decltype(P::transfer(alloc, new_slot, old_slot)) {
return P::transfer(alloc, new_slot, old_slot);
}
#if defined(__cpp_lib_launder) && __cpp_lib_launder >= 201606
// This overload returns true_type for the trait below.
Expand Down
18 changes: 16 additions & 2 deletions absl/container/internal/common_policy_traits_test.cc
Original file line number Diff line number Diff line change
Expand Up @@ -16,10 +16,12 @@

#include <functional>
#include <memory>
#include <new>
#include <type_traits>
#include <utility>

#include "gmock/gmock.h"
#include "gtest/gtest.h"
#include "absl/base/config.h"

namespace absl {
ABSL_NAMESPACE_BEGIN
Expand Down Expand Up @@ -51,9 +53,14 @@ std::function<Slot&(Slot*)> PolicyWithoutOptionalOps::element;
struct PolicyWithOptionalOps : PolicyWithoutOptionalOps {
static std::function<void(void*, Slot*, Slot*)> transfer;
};

std::function<void(void*, Slot*, Slot*)> PolicyWithOptionalOps::transfer;

struct PolicyWithMemcpyTransfer : PolicyWithoutOptionalOps {
static std::function<std::true_type(void*, Slot*, Slot*)> transfer;
};
std::function<std::true_type(void*, Slot*, Slot*)>
PolicyWithMemcpyTransfer::transfer;

struct Test : ::testing::Test {
Test() {
PolicyWithoutOptionalOps::construct = [&](void* a1, Slot* a2, Slot a3) {
Expand Down Expand Up @@ -114,6 +121,13 @@ TEST_F(Test, with_transfer) {
common_policy_traits<PolicyWithOptionalOps>::transfer(&alloc, &a, &b);
}

TEST(TransferUsesMemcpy, Basic) {
EXPECT_FALSE(
common_policy_traits<PolicyWithOptionalOps>::transfer_uses_memcpy());
EXPECT_TRUE(
common_policy_traits<PolicyWithMemcpyTransfer>::transfer_uses_memcpy());
}

} // namespace
} // namespace container_internal
ABSL_NAMESPACE_END
Expand Down
5 changes: 4 additions & 1 deletion absl/container/internal/node_slot_policy.h
Original file line number Diff line number Diff line change
Expand Up @@ -62,9 +62,12 @@ struct node_slot_policy {
Policy::delete_element(alloc, *slot);
}

// Returns true_type to indicate that transfer can use memcpy.
template <class Alloc>
static void transfer(Alloc*, slot_type* new_slot, slot_type* old_slot) {
static std::true_type transfer(Alloc*, slot_type* new_slot,
slot_type* old_slot) {
*new_slot = *old_slot;
return {};
}

static size_t space_used(const slot_type* slot) {
Expand Down
2 changes: 2 additions & 0 deletions absl/container/internal/node_slot_policy_test.cc
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@

#include "gmock/gmock.h"
#include "gtest/gtest.h"
#include "absl/base/config.h"
#include "absl/container/internal/hash_policy_traits.h"

namespace absl {
Expand Down Expand Up @@ -61,6 +62,7 @@ TEST_F(NodeTest, transfer) {
int* b = &s;
NodePolicy::transfer(&alloc, &a, &b);
EXPECT_EQ(&s, a);
EXPECT_TRUE(NodePolicy::transfer_uses_memcpy());
}

} // namespace
Expand Down

0 comments on commit 5ae23ed

Please sign in to comment.