Abseil Common Libraries (C++) (grcp 依赖)
https://abseil.io/
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
202 lines
6.9 KiB
202 lines
6.9 KiB
// Copyright 2019 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 "absl/base/config.h" |
|
#include "absl/container/fixed_array.h" |
|
|
|
#ifdef ABSL_HAVE_EXCEPTIONS |
|
|
|
#include <initializer_list> |
|
|
|
#include "gtest/gtest.h" |
|
#include "absl/base/internal/exception_safety_testing.h" |
|
|
|
namespace absl { |
|
ABSL_NAMESPACE_BEGIN |
|
|
|
namespace { |
|
|
|
constexpr size_t kInlined = 25; |
|
constexpr size_t kSmallSize = kInlined / 2; |
|
constexpr size_t kLargeSize = kInlined * 2; |
|
|
|
constexpr int kInitialValue = 5; |
|
constexpr int kUpdatedValue = 10; |
|
|
|
using ::testing::TestThrowingCtor; |
|
|
|
using Thrower = testing::ThrowingValue<testing::TypeSpec::kEverythingThrows>; |
|
using ThrowAlloc = |
|
testing::ThrowingAllocator<Thrower, testing::AllocSpec::kEverythingThrows>; |
|
using MoveThrower = testing::ThrowingValue<testing::TypeSpec::kNoThrowMove>; |
|
using MoveThrowAlloc = |
|
testing::ThrowingAllocator<MoveThrower, |
|
testing::AllocSpec::kEverythingThrows>; |
|
|
|
using FixedArr = absl::FixedArray<Thrower, kInlined>; |
|
using FixedArrWithAlloc = absl::FixedArray<Thrower, kInlined, ThrowAlloc>; |
|
|
|
using MoveFixedArr = absl::FixedArray<MoveThrower, kInlined>; |
|
using MoveFixedArrWithAlloc = |
|
absl::FixedArray<MoveThrower, kInlined, MoveThrowAlloc>; |
|
|
|
TEST(FixedArrayExceptionSafety, CopyConstructor) { |
|
auto small = FixedArr(kSmallSize); |
|
TestThrowingCtor<FixedArr>(small); |
|
|
|
auto large = FixedArr(kLargeSize); |
|
TestThrowingCtor<FixedArr>(large); |
|
} |
|
|
|
TEST(FixedArrayExceptionSafety, CopyConstructorWithAlloc) { |
|
auto small = FixedArrWithAlloc(kSmallSize); |
|
TestThrowingCtor<FixedArrWithAlloc>(small); |
|
|
|
auto large = FixedArrWithAlloc(kLargeSize); |
|
TestThrowingCtor<FixedArrWithAlloc>(large); |
|
} |
|
|
|
TEST(FixedArrayExceptionSafety, MoveConstructor) { |
|
TestThrowingCtor<FixedArr>(FixedArr(kSmallSize)); |
|
TestThrowingCtor<FixedArr>(FixedArr(kLargeSize)); |
|
|
|
// TypeSpec::kNoThrowMove |
|
TestThrowingCtor<MoveFixedArr>(MoveFixedArr(kSmallSize)); |
|
TestThrowingCtor<MoveFixedArr>(MoveFixedArr(kLargeSize)); |
|
} |
|
|
|
TEST(FixedArrayExceptionSafety, MoveConstructorWithAlloc) { |
|
TestThrowingCtor<FixedArrWithAlloc>(FixedArrWithAlloc(kSmallSize)); |
|
TestThrowingCtor<FixedArrWithAlloc>(FixedArrWithAlloc(kLargeSize)); |
|
|
|
// TypeSpec::kNoThrowMove |
|
TestThrowingCtor<MoveFixedArrWithAlloc>(MoveFixedArrWithAlloc(kSmallSize)); |
|
TestThrowingCtor<MoveFixedArrWithAlloc>(MoveFixedArrWithAlloc(kLargeSize)); |
|
} |
|
|
|
TEST(FixedArrayExceptionSafety, SizeConstructor) { |
|
TestThrowingCtor<FixedArr>(kSmallSize); |
|
TestThrowingCtor<FixedArr>(kLargeSize); |
|
} |
|
|
|
TEST(FixedArrayExceptionSafety, SizeConstructorWithAlloc) { |
|
TestThrowingCtor<FixedArrWithAlloc>(kSmallSize); |
|
TestThrowingCtor<FixedArrWithAlloc>(kLargeSize); |
|
} |
|
|
|
TEST(FixedArrayExceptionSafety, SizeValueConstructor) { |
|
TestThrowingCtor<FixedArr>(kSmallSize, Thrower()); |
|
TestThrowingCtor<FixedArr>(kLargeSize, Thrower()); |
|
} |
|
|
|
TEST(FixedArrayExceptionSafety, SizeValueConstructorWithAlloc) { |
|
TestThrowingCtor<FixedArrWithAlloc>(kSmallSize, Thrower()); |
|
TestThrowingCtor<FixedArrWithAlloc>(kLargeSize, Thrower()); |
|
} |
|
|
|
TEST(FixedArrayExceptionSafety, IteratorConstructor) { |
|
auto small = FixedArr(kSmallSize); |
|
TestThrowingCtor<FixedArr>(small.begin(), small.end()); |
|
|
|
auto large = FixedArr(kLargeSize); |
|
TestThrowingCtor<FixedArr>(large.begin(), large.end()); |
|
} |
|
|
|
TEST(FixedArrayExceptionSafety, IteratorConstructorWithAlloc) { |
|
auto small = FixedArrWithAlloc(kSmallSize); |
|
TestThrowingCtor<FixedArrWithAlloc>(small.begin(), small.end()); |
|
|
|
auto large = FixedArrWithAlloc(kLargeSize); |
|
TestThrowingCtor<FixedArrWithAlloc>(large.begin(), large.end()); |
|
} |
|
|
|
TEST(FixedArrayExceptionSafety, InitListConstructor) { |
|
constexpr int small_inlined = 3; |
|
using SmallFixedArr = absl::FixedArray<Thrower, small_inlined>; |
|
|
|
TestThrowingCtor<SmallFixedArr>(std::initializer_list<Thrower>{}); |
|
// Test inlined allocation |
|
TestThrowingCtor<SmallFixedArr>( |
|
std::initializer_list<Thrower>{Thrower{}, Thrower{}}); |
|
// Test out of line allocation |
|
TestThrowingCtor<SmallFixedArr>(std::initializer_list<Thrower>{ |
|
Thrower{}, Thrower{}, Thrower{}, Thrower{}, Thrower{}}); |
|
} |
|
|
|
TEST(FixedArrayExceptionSafety, InitListConstructorWithAlloc) { |
|
constexpr int small_inlined = 3; |
|
using SmallFixedArrWithAlloc = |
|
absl::FixedArray<Thrower, small_inlined, ThrowAlloc>; |
|
|
|
TestThrowingCtor<SmallFixedArrWithAlloc>(std::initializer_list<Thrower>{}); |
|
// Test inlined allocation |
|
TestThrowingCtor<SmallFixedArrWithAlloc>( |
|
std::initializer_list<Thrower>{Thrower{}, Thrower{}}); |
|
// Test out of line allocation |
|
TestThrowingCtor<SmallFixedArrWithAlloc>(std::initializer_list<Thrower>{ |
|
Thrower{}, Thrower{}, Thrower{}, Thrower{}, Thrower{}}); |
|
} |
|
|
|
template <typename FixedArrT> |
|
testing::AssertionResult ReadMemory(FixedArrT* fixed_arr) { |
|
// Marked volatile to prevent optimization. Used for running asan tests. |
|
volatile int sum = 0; |
|
for (const auto& thrower : *fixed_arr) { |
|
sum += thrower.Get(); |
|
} |
|
return testing::AssertionSuccess() << "Values sum to [" << sum << "]"; |
|
} |
|
|
|
TEST(FixedArrayExceptionSafety, Fill) { |
|
auto test_fill = testing::MakeExceptionSafetyTester() |
|
.WithContracts(ReadMemory<FixedArr>) |
|
.WithOperation([&](FixedArr* fixed_arr_ptr) { |
|
auto thrower = |
|
Thrower(kUpdatedValue, testing::nothrow_ctor); |
|
fixed_arr_ptr->fill(thrower); |
|
}); |
|
|
|
EXPECT_TRUE( |
|
test_fill.WithInitialValue(FixedArr(kSmallSize, Thrower(kInitialValue))) |
|
.Test()); |
|
EXPECT_TRUE( |
|
test_fill.WithInitialValue(FixedArr(kLargeSize, Thrower(kInitialValue))) |
|
.Test()); |
|
} |
|
|
|
TEST(FixedArrayExceptionSafety, FillWithAlloc) { |
|
auto test_fill = testing::MakeExceptionSafetyTester() |
|
.WithContracts(ReadMemory<FixedArrWithAlloc>) |
|
.WithOperation([&](FixedArrWithAlloc* fixed_arr_ptr) { |
|
auto thrower = |
|
Thrower(kUpdatedValue, testing::nothrow_ctor); |
|
fixed_arr_ptr->fill(thrower); |
|
}); |
|
|
|
EXPECT_TRUE(test_fill |
|
.WithInitialValue( |
|
FixedArrWithAlloc(kSmallSize, Thrower(kInitialValue))) |
|
.Test()); |
|
EXPECT_TRUE(test_fill |
|
.WithInitialValue( |
|
FixedArrWithAlloc(kLargeSize, Thrower(kInitialValue))) |
|
.Test()); |
|
} |
|
|
|
} // namespace |
|
|
|
ABSL_NAMESPACE_END |
|
} // namespace absl |
|
|
|
#endif // ABSL_HAVE_EXCEPTIONS
|
|
|