mirror of
https://github.com/abseil/abseil-cpp.git
synced 2026-06-04 12:07:05 +08:00
-- 9fc37c11b9e46287acef00ee06ed9adcba54dd13 by Greg Falcon <gfalcon@google.com>: Rename absl::hash_internal::HashState to absl::hash_internal::MixingHashState. Before this change, we had two classes named HashState: absl::HashState, the public API used for type erasure, and absl::hash_internal::HashState, the internal concrete implementation ordinarily used. The internal class used to be named `CityHashState`, but we renamed it to `HashState` it when we changed underlying hash implementation to wyhash. This inadvertent naming conflict made the code much harder to read, and this change intends to undo that. PiperOrigin-RevId: 373481959 -- 4aec55ffddebd085c239352a2e20721091f719a1 by Greg Falcon <gfalcon@google.com>: Introduce absl::HashOf(), a convenience wrapper around absl::Hash that calculates hashes from the values of its arguments. PiperOrigin-RevId: 373461406 -- 86b5fd8db50bbc8bd0aa9258523527381fe0445d by Abseil Team <absl-team@google.com>: Improve speed of BlockingCounter by making its most common path lock free. With the new implementation, the fast path of BlockingCounter::DecrementCount() is only a fetch_sub operation. This is most times much more efficient than the previous implementation (full mutex lock/unlock). As a matter of fact, in most actual usecases in practice, the waiter thread is already waiting on the Wait() call when DecrementCount() is called, which makes Mutex::Unlock() take the slow path as there's a waiter thread that it might need to wake up. PiperOrigin-RevId: 373394164 -- 65c876be5eac0cd32583ff8535ede4109d39cf3f by Martijn Vels <mvels@google.com>: Move the 'sample copied cord' logic into MaybeTrackCord(), This changes move the logic for selecting if a cord should remain being sampled from Cord to CordzInfo::MaybeTrackCord, and updates the documentation for the latter method. PiperOrigin-RevId: 373363168 -- e84410bd0aada293a81dfb82656c952e209e21fb by Martijn Vels <mvels@google.com>: Add check for the first call to cordz_should_profile() for each thread. This prevents the first cord of a newly created thread to be always sampled, which is a 'bad' kind of determinism for sampling. PiperOrigin-RevId: 373229768 -- bf09c589dc099ac8f4af780bf7e609c53c27574c by Samuel Benzaquen <sbenza@google.com>: Refactor the Flags structure into an enum. This gives us more control over the representation and allows for easier merging during parsing. PiperOrigin-RevId: 373163038 -- b947b0c51083b7b6508284b5d31819596c91729e by Derek Mauro <dmauro@google.com>: Fixes warnings about shadowed variables Fixes #956 PiperOrigin-RevId: 373158133 GitOrigin-RevId: 9fc37c11b9e46287acef00ee06ed9adcba54dd13 Change-Id: I91f35699f9bf439d1a870c6493946a310afe088c
467 lines
17 KiB
C++
467 lines
17 KiB
C++
// Copyright 2021 The Abseil Authors
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// https://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
#include <cstdint>
|
|
#include <string>
|
|
|
|
#include "gmock/gmock.h"
|
|
#include "gtest/gtest.h"
|
|
#include "absl/base/config.h"
|
|
#include "absl/base/internal/raw_logging.h"
|
|
#include "absl/base/macros.h"
|
|
#include "absl/strings/cord.h"
|
|
#include "absl/strings/cord_test_helpers.h"
|
|
#include "absl/strings/cordz_test_helpers.h"
|
|
#include "absl/strings/internal/cordz_functions.h"
|
|
#include "absl/strings/internal/cordz_info.h"
|
|
#include "absl/strings/internal/cordz_sample_token.h"
|
|
#include "absl/strings/internal/cordz_statistics.h"
|
|
#include "absl/strings/internal/cordz_update_tracker.h"
|
|
#include "absl/strings/str_cat.h"
|
|
#include "absl/strings/string_view.h"
|
|
|
|
#ifdef ABSL_INTERNAL_CORDZ_ENABLED
|
|
|
|
using testing::Eq;
|
|
using testing::AnyOf;
|
|
|
|
namespace absl {
|
|
ABSL_NAMESPACE_BEGIN
|
|
|
|
using cord_internal::CordzInfo;
|
|
using cord_internal::CordzSampleToken;
|
|
using cord_internal::CordzStatistics;
|
|
using cord_internal::CordzUpdateTracker;
|
|
using Method = CordzUpdateTracker::MethodIdentifier;
|
|
|
|
// Do not print cord contents, we only care about 'size' perhaps.
|
|
// Note that this method must be inside the named namespace.
|
|
inline void PrintTo(const Cord& cord, std::ostream* s) {
|
|
if (s) *s << "Cord[" << cord.size() << "]";
|
|
}
|
|
|
|
namespace {
|
|
|
|
auto constexpr kMaxInline = cord_internal::kMaxInline;
|
|
|
|
// Returns a string_view value of the specified length
|
|
// We do this to avoid 'consuming' large strings in Cord by default.
|
|
absl::string_view MakeString(size_t size) {
|
|
thread_local std::string str;
|
|
str = std::string(size, '.');
|
|
return str;
|
|
}
|
|
|
|
absl::string_view MakeString(TestCordSize size) {
|
|
return MakeString(Length(size));
|
|
}
|
|
|
|
// Returns a cord with a sampled method of kAppendString.
|
|
absl::Cord MakeAppendStringCord(TestCordSize size) {
|
|
CordzSamplingIntervalHelper always(1);
|
|
absl::Cord cord;
|
|
cord.Append(MakeString(size));
|
|
return cord;
|
|
}
|
|
|
|
std::string TestParamToString(::testing::TestParamInfo<TestCordSize> size) {
|
|
return absl::StrCat("On", ToString(size.param), "Cord");
|
|
}
|
|
|
|
class CordzUpdateTest : public testing::TestWithParam<TestCordSize> {
|
|
public:
|
|
Cord& cord() { return cord_; }
|
|
|
|
Method InitialOr(Method method) const {
|
|
return (GetParam() > TestCordSize::kInlined) ? Method::kConstructorString
|
|
: method;
|
|
}
|
|
|
|
private:
|
|
CordzSamplingIntervalHelper sample_every_{1};
|
|
Cord cord_{MakeString(GetParam())};
|
|
};
|
|
|
|
template <typename T>
|
|
std::string ParamToString(::testing::TestParamInfo<T> param) {
|
|
return std::string(ToString(param.param));
|
|
}
|
|
|
|
INSTANTIATE_TEST_SUITE_P(WithParam, CordzUpdateTest,
|
|
testing::Values(TestCordSize::kEmpty,
|
|
TestCordSize::kInlined,
|
|
TestCordSize::kLarge),
|
|
TestParamToString);
|
|
|
|
class CordzStringTest : public testing::TestWithParam<TestCordSize> {
|
|
private:
|
|
CordzSamplingIntervalHelper sample_every_{1};
|
|
};
|
|
|
|
INSTANTIATE_TEST_SUITE_P(WithParam, CordzStringTest,
|
|
testing::Values(TestCordSize::kInlined,
|
|
TestCordSize::kStringSso1,
|
|
TestCordSize::kStringSso2,
|
|
TestCordSize::kSmall,
|
|
TestCordSize::kLarge),
|
|
ParamToString<TestCordSize>);
|
|
|
|
TEST(CordzTest, ConstructSmallArray) {
|
|
CordzSamplingIntervalHelper sample_every{1};
|
|
Cord cord(MakeString(TestCordSize::kSmall));
|
|
EXPECT_THAT(cord, HasValidCordzInfoOf(Method::kConstructorString));
|
|
}
|
|
|
|
TEST(CordzTest, ConstructLargeArray) {
|
|
CordzSamplingIntervalHelper sample_every{1};
|
|
Cord cord(MakeString(TestCordSize::kLarge));
|
|
EXPECT_THAT(cord, HasValidCordzInfoOf(Method::kConstructorString));
|
|
}
|
|
|
|
TEST_P(CordzStringTest, ConstructString) {
|
|
CordzSamplingIntervalHelper sample_every{1};
|
|
Cord cord(std::string(Length(GetParam()), '.'));
|
|
if (Length(GetParam()) > kMaxInline) {
|
|
EXPECT_THAT(cord, HasValidCordzInfoOf(Method::kConstructorString));
|
|
}
|
|
}
|
|
|
|
TEST(CordzTest, CopyConstructFromUnsampled) {
|
|
CordzSamplingIntervalHelper sample_every{1};
|
|
Cord src = UnsampledCord(MakeString(TestCordSize::kLarge));
|
|
Cord cord(src);
|
|
EXPECT_THAT(GetCordzInfoForTesting(cord), Eq(nullptr));
|
|
}
|
|
|
|
TEST(CordzTest, CopyConstructFromSampled) {
|
|
CordzSamplingIntervalHelper sample_never{99999};
|
|
Cord src = MakeAppendStringCord(TestCordSize::kLarge);
|
|
Cord cord(src);
|
|
ASSERT_THAT(cord, HasValidCordzInfoOf(Method::kConstructorCord));
|
|
CordzStatistics stats = GetCordzInfoForTesting(cord)->GetCordzStatistics();
|
|
EXPECT_THAT(stats.parent_method, Eq(Method::kAppendString));
|
|
EXPECT_THAT(stats.update_tracker.Value(Method::kAppendString), Eq(1));
|
|
}
|
|
|
|
TEST(CordzTest, MoveConstruct) {
|
|
CordzSamplingIntervalHelper sample_every{1};
|
|
Cord src(MakeString(TestCordSize::kLarge));
|
|
Cord cord(std::move(src));
|
|
EXPECT_THAT(cord, HasValidCordzInfoOf(Method::kConstructorString));
|
|
}
|
|
|
|
TEST_P(CordzUpdateTest, AssignUnsampledCord) {
|
|
Cord src = UnsampledCord(MakeString(TestCordSize::kLarge));
|
|
const CordzInfo* info = GetCordzInfoForTesting(cord());
|
|
cord() = src;
|
|
EXPECT_THAT(GetCordzInfoForTesting(cord()), Eq(nullptr));
|
|
EXPECT_FALSE(CordzInfoIsListed(info));
|
|
}
|
|
|
|
TEST_P(CordzUpdateTest, AssignSampledCord) {
|
|
Cord src = MakeAppendStringCord(TestCordSize::kLarge);
|
|
cord() = src;
|
|
ASSERT_THAT(cord(), HasValidCordzInfoOf(Method::kAssignCord));
|
|
CordzStatistics stats = GetCordzInfoForTesting(cord())->GetCordzStatistics();
|
|
EXPECT_THAT(stats.parent_method, Eq(Method::kAppendString));
|
|
EXPECT_THAT(stats.update_tracker.Value(Method::kAppendString), Eq(1));
|
|
EXPECT_THAT(stats.update_tracker.Value(Method::kConstructorString), Eq(0));
|
|
}
|
|
|
|
TEST(CordzUpdateTest, AssignSampledCordToInlined) {
|
|
CordzSamplingIntervalHelper sample_never{99999};
|
|
Cord cord;
|
|
Cord src = MakeAppendStringCord(TestCordSize::kLarge);
|
|
cord = src;
|
|
ASSERT_THAT(cord, HasValidCordzInfoOf(Method::kAssignCord));
|
|
CordzStatistics stats = GetCordzInfoForTesting(cord)->GetCordzStatistics();
|
|
EXPECT_THAT(stats.parent_method, Eq(Method::kAppendString));
|
|
EXPECT_THAT(stats.update_tracker.Value(Method::kAppendString), Eq(1));
|
|
EXPECT_THAT(stats.update_tracker.Value(Method::kConstructorString), Eq(0));
|
|
}
|
|
|
|
TEST(CordzUpdateTest, AssignSampledCordToUnsampledCord) {
|
|
CordzSamplingIntervalHelper sample_never{99999};
|
|
Cord cord = UnsampledCord(MakeString(TestCordSize::kLarge));
|
|
Cord src = MakeAppendStringCord(TestCordSize::kLarge);
|
|
cord = src;
|
|
ASSERT_THAT(cord, HasValidCordzInfoOf(Method::kAssignCord));
|
|
CordzStatistics stats = GetCordzInfoForTesting(cord)->GetCordzStatistics();
|
|
EXPECT_THAT(stats.parent_method, Eq(Method::kAppendString));
|
|
EXPECT_THAT(stats.update_tracker.Value(Method::kAppendString), Eq(1));
|
|
EXPECT_THAT(stats.update_tracker.Value(Method::kConstructorString), Eq(0));
|
|
}
|
|
|
|
TEST(CordzUpdateTest, AssignUnsampledCordToSampledCordWithoutSampling) {
|
|
CordzSamplingIntervalHelper sample_never{99999};
|
|
Cord cord = MakeAppendStringCord(TestCordSize::kLarge);
|
|
const CordzInfo* info = GetCordzInfoForTesting(cord);
|
|
Cord src = UnsampledCord(MakeString(TestCordSize::kLarge));
|
|
cord = src;
|
|
EXPECT_THAT(GetCordzInfoForTesting(cord), Eq(nullptr));
|
|
EXPECT_FALSE(CordzInfoIsListed(info));
|
|
}
|
|
|
|
TEST(CordzUpdateTest, AssignUnsampledCordToSampledCordWithSampling) {
|
|
CordzSamplingIntervalHelper sample_every{1};
|
|
Cord cord = MakeAppendStringCord(TestCordSize::kLarge);
|
|
const CordzInfo* info = GetCordzInfoForTesting(cord);
|
|
Cord src = UnsampledCord(MakeString(TestCordSize::kLarge));
|
|
cord = src;
|
|
EXPECT_THAT(GetCordzInfoForTesting(cord), Eq(nullptr));
|
|
EXPECT_FALSE(CordzInfoIsListed(info));
|
|
}
|
|
|
|
TEST(CordzUpdateTest, AssignSampledCordToSampledCord) {
|
|
CordzSamplingIntervalHelper sample_every{1};
|
|
Cord src = MakeAppendStringCord(TestCordSize::kLarge);
|
|
Cord cord(MakeString(TestCordSize::kLarge));
|
|
cord = src;
|
|
ASSERT_THAT(cord, HasValidCordzInfoOf(Method::kAssignCord));
|
|
CordzStatistics stats = GetCordzInfoForTesting(cord)->GetCordzStatistics();
|
|
EXPECT_THAT(stats.parent_method, Eq(Method::kAppendString));
|
|
EXPECT_THAT(stats.update_tracker.Value(Method::kAppendString), Eq(1));
|
|
EXPECT_THAT(stats.update_tracker.Value(Method::kConstructorString), Eq(0));
|
|
}
|
|
|
|
TEST(CordzUpdateTest, AssignUnsampledCordToSampledCord) {
|
|
CordzSamplingIntervalHelper sample_every{1};
|
|
Cord src = MakeAppendStringCord(TestCordSize::kLarge);
|
|
Cord cord(MakeString(TestCordSize::kLarge));
|
|
cord = src;
|
|
ASSERT_THAT(cord, HasValidCordzInfoOf(Method::kAssignCord));
|
|
CordzStatistics stats = GetCordzInfoForTesting(cord)->GetCordzStatistics();
|
|
EXPECT_THAT(stats.parent_method, Eq(Method::kAppendString));
|
|
EXPECT_THAT(stats.update_tracker.Value(Method::kAppendString), Eq(1));
|
|
EXPECT_THAT(stats.update_tracker.Value(Method::kConstructorString), Eq(0));
|
|
}
|
|
|
|
TEST(CordzTest, AssignInlinedCordToSampledCord) {
|
|
CordzSampleToken token;
|
|
CordzSamplingIntervalHelper sample_every{1};
|
|
Cord cord(MakeString(TestCordSize::kLarge));
|
|
const CordzInfo* info = GetCordzInfoForTesting(cord);
|
|
Cord src = UnsampledCord(MakeString(TestCordSize::kInlined));
|
|
cord = src;
|
|
EXPECT_THAT(GetCordzInfoForTesting(cord), Eq(nullptr));
|
|
EXPECT_FALSE(CordzInfoIsListed(info));
|
|
}
|
|
|
|
TEST(CordzUpdateTest, MoveAssignCord) {
|
|
CordzSamplingIntervalHelper sample_every{1};
|
|
Cord cord;
|
|
Cord src(MakeString(TestCordSize::kLarge));
|
|
cord = std::move(src);
|
|
EXPECT_THAT(cord, HasValidCordzInfoOf(Method::kConstructorString));
|
|
}
|
|
|
|
TEST_P(CordzUpdateTest, AssignLargeArray) {
|
|
cord() = MakeString(TestCordSize::kSmall);
|
|
EXPECT_THAT(cord(), HasValidCordzInfoOf(Method::kAssignString));
|
|
}
|
|
|
|
TEST_P(CordzUpdateTest, AssignSmallArray) {
|
|
cord() = MakeString(TestCordSize::kSmall);
|
|
EXPECT_THAT(cord(), HasValidCordzInfoOf(Method::kAssignString));
|
|
}
|
|
|
|
TEST_P(CordzUpdateTest, AssignInlinedArray) {
|
|
cord() = MakeString(TestCordSize::kInlined);
|
|
EXPECT_THAT(GetCordzInfoForTesting(cord()), Eq(nullptr));
|
|
}
|
|
|
|
TEST_P(CordzStringTest, AssignStringToInlined) {
|
|
Cord cord;
|
|
cord = std::string(Length(GetParam()), '.');
|
|
if (Length(GetParam()) > kMaxInline) {
|
|
EXPECT_THAT(cord, HasValidCordzInfoOf(Method::kAssignString));
|
|
}
|
|
}
|
|
|
|
TEST_P(CordzStringTest, AssignStringToCord) {
|
|
Cord cord(MakeString(TestCordSize::kLarge));
|
|
cord = std::string(Length(GetParam()), '.');
|
|
if (Length(GetParam()) > kMaxInline) {
|
|
EXPECT_THAT(cord, HasValidCordzInfoOf(Method::kConstructorString));
|
|
EXPECT_THAT(cord, CordzMethodCountEq(Method::kAssignString, 1));
|
|
}
|
|
}
|
|
|
|
TEST_P(CordzUpdateTest, AssignInlinedString) {
|
|
cord() = std::string(Length(TestCordSize::kInlined), '.');
|
|
EXPECT_THAT(GetCordzInfoForTesting(cord()), Eq(nullptr));
|
|
}
|
|
|
|
TEST_P(CordzUpdateTest, AppendCord) {
|
|
Cord src = UnsampledCord(MakeString(TestCordSize::kLarge));
|
|
cord().Append(src);
|
|
EXPECT_THAT(cord(), HasValidCordzInfoOf(InitialOr(Method::kAppendCord)));
|
|
}
|
|
|
|
TEST_P(CordzUpdateTest, MoveAppendCord) {
|
|
cord().Append(UnsampledCord(MakeString(TestCordSize::kLarge)));
|
|
EXPECT_THAT(cord(), HasValidCordzInfoOf(InitialOr(Method::kAppendCord)));
|
|
}
|
|
|
|
TEST_P(CordzUpdateTest, AppendSmallArray) {
|
|
cord().Append(MakeString(TestCordSize::kSmall));
|
|
EXPECT_THAT(cord(), HasValidCordzInfoOf(InitialOr(Method::kAppendString)));
|
|
}
|
|
|
|
TEST_P(CordzUpdateTest, AppendLargeArray) {
|
|
cord().Append(MakeString(TestCordSize::kLarge));
|
|
EXPECT_THAT(cord(), HasValidCordzInfoOf(InitialOr(Method::kAppendString)));
|
|
}
|
|
|
|
TEST_P(CordzStringTest, AppendStringToEmpty) {
|
|
Cord cord;
|
|
cord.Append(std::string(Length(GetParam()), '.'));
|
|
if (Length(GetParam()) > kMaxInline) {
|
|
EXPECT_THAT(cord, HasValidCordzInfoOf(Method::kAppendString));
|
|
}
|
|
}
|
|
|
|
TEST_P(CordzStringTest, AppendStringToInlined) {
|
|
Cord cord(MakeString(TestCordSize::kInlined));
|
|
cord.Append(std::string(Length(GetParam()), '.'));
|
|
if (Length(TestCordSize::kInlined) + Length(GetParam()) > kMaxInline) {
|
|
EXPECT_THAT(cord, HasValidCordzInfoOf(Method::kAppendString));
|
|
}
|
|
}
|
|
|
|
TEST_P(CordzStringTest, AppendStringToCord) {
|
|
Cord cord(MakeString(TestCordSize::kLarge));
|
|
cord.Append(std::string(Length(GetParam()), '.'));
|
|
EXPECT_THAT(cord, HasValidCordzInfoOf(Method::kConstructorString));
|
|
EXPECT_THAT(cord, CordzMethodCountEq(Method::kAppendString, 1));
|
|
}
|
|
|
|
TEST(CordzTest, MakeCordFromExternal) {
|
|
CordzSamplingIntervalHelper sample_every{1};
|
|
Cord cord = MakeCordFromExternal("Hello world", [](absl::string_view) {});
|
|
EXPECT_THAT(cord, HasValidCordzInfoOf(Method::kMakeCordFromExternal));
|
|
}
|
|
|
|
TEST(CordzTest, MakeCordFromEmptyExternal) {
|
|
CordzSamplingIntervalHelper sample_every{1};
|
|
Cord cord = MakeCordFromExternal({}, [](absl::string_view) {});
|
|
EXPECT_THAT(GetCordzInfoForTesting(cord), Eq(nullptr));
|
|
}
|
|
|
|
TEST_P(CordzUpdateTest, PrependCord) {
|
|
Cord src = UnsampledCord(MakeString(TestCordSize::kLarge));
|
|
cord().Prepend(src);
|
|
EXPECT_THAT(cord(), HasValidCordzInfoOf(InitialOr(Method::kPrependCord)));
|
|
}
|
|
|
|
TEST_P(CordzUpdateTest, PrependSmallArray) {
|
|
cord().Prepend(MakeString(TestCordSize::kSmall));
|
|
EXPECT_THAT(cord(), HasValidCordzInfoOf(InitialOr(Method::kPrependString)));
|
|
}
|
|
|
|
TEST_P(CordzUpdateTest, PrependLargeArray) {
|
|
cord().Prepend(MakeString(TestCordSize::kLarge));
|
|
EXPECT_THAT(cord(), HasValidCordzInfoOf(InitialOr(Method::kPrependString)));
|
|
}
|
|
|
|
TEST_P(CordzStringTest, PrependStringToEmpty) {
|
|
Cord cord;
|
|
cord.Prepend(std::string(Length(GetParam()), '.'));
|
|
if (Length(GetParam()) > kMaxInline) {
|
|
EXPECT_THAT(cord, HasValidCordzInfoOf(Method::kPrependString));
|
|
}
|
|
}
|
|
|
|
TEST_P(CordzStringTest, PrependStringToInlined) {
|
|
Cord cord(MakeString(TestCordSize::kInlined));
|
|
cord.Prepend(std::string(Length(GetParam()), '.'));
|
|
if (Length(TestCordSize::kInlined) + Length(GetParam()) > kMaxInline) {
|
|
EXPECT_THAT(cord, HasValidCordzInfoOf(Method::kPrependString));
|
|
}
|
|
}
|
|
|
|
TEST_P(CordzStringTest, PrependStringToCord) {
|
|
Cord cord(MakeString(TestCordSize::kLarge));
|
|
cord.Prepend(std::string(Length(GetParam()), '.'));
|
|
EXPECT_THAT(cord, HasValidCordzInfoOf(Method::kConstructorString));
|
|
EXPECT_THAT(cord, CordzMethodCountEq(Method::kPrependString, 1));
|
|
}
|
|
|
|
TEST(CordzTest, RemovePrefix) {
|
|
CordzSamplingIntervalHelper sample_every(1);
|
|
Cord cord(MakeString(TestCordSize::kLarge));
|
|
|
|
// Half the cord
|
|
cord.RemovePrefix(cord.size() / 2);
|
|
EXPECT_THAT(cord, HasValidCordzInfoOf(Method::kConstructorString));
|
|
EXPECT_THAT(cord, CordzMethodCountEq(Method::kRemovePrefix, 1));
|
|
|
|
// TODO(mvels): RemovePrefix does not reset to inlined, except if empty?
|
|
cord.RemovePrefix(cord.size() - kMaxInline);
|
|
EXPECT_THAT(cord, HasValidCordzInfoOf(Method::kConstructorString));
|
|
EXPECT_THAT(cord, CordzMethodCountEq(Method::kRemovePrefix, 2));
|
|
|
|
cord.RemovePrefix(cord.size());
|
|
EXPECT_THAT(GetCordzInfoForTesting(cord), Eq(nullptr));
|
|
}
|
|
|
|
TEST(CordzTest, RemoveSuffix) {
|
|
CordzSamplingIntervalHelper sample_every(1);
|
|
Cord cord(MakeString(TestCordSize::kLarge));
|
|
|
|
// Half the cord
|
|
cord.RemoveSuffix(cord.size() / 2);
|
|
EXPECT_THAT(cord, HasValidCordzInfoOf(Method::kConstructorString));
|
|
EXPECT_THAT(cord, CordzMethodCountEq(Method::kRemoveSuffix, 1));
|
|
|
|
// TODO(mvels): RemoveSuffix does not reset to inlined, except if empty?
|
|
cord.RemoveSuffix(cord.size() - kMaxInline);
|
|
EXPECT_THAT(cord, HasValidCordzInfoOf(Method::kConstructorString));
|
|
EXPECT_THAT(cord, CordzMethodCountEq(Method::kRemoveSuffix, 2));
|
|
|
|
cord.RemoveSuffix(cord.size());
|
|
EXPECT_THAT(GetCordzInfoForTesting(cord), Eq(nullptr));
|
|
}
|
|
|
|
TEST(CordzTest, SubCordFromUnsampledCord) {
|
|
CordzSamplingIntervalHelper sample_every{1};
|
|
Cord src = UnsampledCord(MakeString(TestCordSize::kLarge));
|
|
Cord cord = src.Subcord(10, src.size() / 2);
|
|
EXPECT_THAT(GetCordzInfoForTesting(cord), Eq(nullptr));
|
|
}
|
|
|
|
TEST(CordzTest, SubCordFromSampledCord) {
|
|
CordzSamplingIntervalHelper sample_never{99999};
|
|
Cord src = MakeAppendStringCord(TestCordSize::kLarge);
|
|
Cord cord = src.Subcord(10, src.size() / 2);
|
|
ASSERT_THAT(cord, HasValidCordzInfoOf(Method::kSubCord));
|
|
CordzStatistics stats = GetCordzInfoForTesting(cord)->GetCordzStatistics();
|
|
EXPECT_THAT(stats.parent_method, Eq(Method::kAppendString));
|
|
EXPECT_THAT(stats.update_tracker.Value(Method::kAppendString), Eq(1));
|
|
}
|
|
|
|
TEST(CordzTest, SmallSubCord) {
|
|
CordzSamplingIntervalHelper sample_never{99999};
|
|
Cord src = MakeAppendStringCord(TestCordSize::kLarge);
|
|
Cord cord = src.Subcord(10, kMaxInline + 1);
|
|
EXPECT_THAT(cord, HasValidCordzInfoOf(Method::kSubCord));
|
|
}
|
|
|
|
} // namespace
|
|
|
|
ABSL_NAMESPACE_END
|
|
} // namespace absl
|
|
|
|
#endif // ABSL_INTERNAL_CORDZ_ENABLED
|