|  | /* | 
|  | * Copyright (C) 2018 The Android Open Source Project | 
|  | * | 
|  | * 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 | 
|  | * | 
|  | *      http://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. | 
|  | */ | 
|  |  | 
|  | // Comparisions of floats is used extensively in this file. Ignore warnings | 
|  | // as we want to stay close to Chromium. | 
|  | #if defined(__GNUC__) || defined(__clang__) | 
|  | #pragma GCC diagnostic push | 
|  | #pragma GCC diagnostic ignored "-Wfloat-equal" | 
|  | #endif | 
|  |  | 
|  | #include <memory> | 
|  | #include <set> | 
|  | #include <string> | 
|  | #include <vector> | 
|  |  | 
|  | #include "perfetto/ext/base/optional.h" | 
|  | #include "perfetto/ext/base/utils.h" | 
|  | #include "test/gtest_and_gmock.h" | 
|  |  | 
|  | using ::testing::ElementsAre; | 
|  |  | 
|  | namespace perfetto { | 
|  | namespace base { | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | // Object used to test complex object with Optional<T> in addition of the move | 
|  | // semantics. | 
|  | class TestObject { | 
|  | public: | 
|  | enum class State { | 
|  | DEFAULT_CONSTRUCTED, | 
|  | VALUE_CONSTRUCTED, | 
|  | COPY_CONSTRUCTED, | 
|  | MOVE_CONSTRUCTED, | 
|  | MOVED_FROM, | 
|  | COPY_ASSIGNED, | 
|  | MOVE_ASSIGNED, | 
|  | SWAPPED, | 
|  | }; | 
|  |  | 
|  | TestObject() : foo_(0), bar_(0.0), state_(State::DEFAULT_CONSTRUCTED) {} | 
|  |  | 
|  | TestObject(int foo, double bar) | 
|  | : foo_(foo), bar_(bar), state_(State::VALUE_CONSTRUCTED) {} | 
|  |  | 
|  | TestObject(const TestObject& other) | 
|  | : foo_(other.foo_), | 
|  | bar_(other.bar_), | 
|  | state_(State::COPY_CONSTRUCTED), | 
|  | move_ctors_count_(other.move_ctors_count_) {} | 
|  |  | 
|  | TestObject(TestObject&& other) | 
|  | : foo_(std::move(other.foo_)), | 
|  | bar_(std::move(other.bar_)), | 
|  | state_(State::MOVE_CONSTRUCTED), | 
|  | move_ctors_count_(other.move_ctors_count_ + 1) { | 
|  | other.state_ = State::MOVED_FROM; | 
|  | } | 
|  |  | 
|  | TestObject& operator=(const TestObject& other) { | 
|  | foo_ = other.foo_; | 
|  | bar_ = other.bar_; | 
|  | state_ = State::COPY_ASSIGNED; | 
|  | move_ctors_count_ = other.move_ctors_count_; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | TestObject& operator=(TestObject&& other) { | 
|  | foo_ = other.foo_; | 
|  | bar_ = other.bar_; | 
|  | state_ = State::MOVE_ASSIGNED; | 
|  | move_ctors_count_ = other.move_ctors_count_; | 
|  | other.state_ = State::MOVED_FROM; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | void Swap(TestObject* other) { | 
|  | using std::swap; | 
|  | swap(foo_, other->foo_); | 
|  | swap(bar_, other->bar_); | 
|  | swap(move_ctors_count_, other->move_ctors_count_); | 
|  | state_ = State::SWAPPED; | 
|  | other->state_ = State::SWAPPED; | 
|  | } | 
|  |  | 
|  | bool operator==(const TestObject& other) const { | 
|  | return std::tie(foo_, bar_) == std::tie(other.foo_, other.bar_); | 
|  | } | 
|  |  | 
|  | bool operator!=(const TestObject& other) const { return !(*this == other); } | 
|  |  | 
|  | int foo() const { return foo_; } | 
|  | State state() const { return state_; } | 
|  | int move_ctors_count() const { return move_ctors_count_; } | 
|  |  | 
|  | private: | 
|  | int foo_; | 
|  | double bar_; | 
|  | State state_; | 
|  | int move_ctors_count_ = 0; | 
|  | }; | 
|  |  | 
|  | // Implementing Swappable concept. | 
|  | void swap(TestObject& lhs, TestObject& rhs) { | 
|  | lhs.Swap(&rhs); | 
|  | } | 
|  |  | 
|  | class NonTriviallyDestructible { | 
|  | ~NonTriviallyDestructible() {} | 
|  | }; | 
|  |  | 
|  | class DeletedDefaultConstructor { | 
|  | public: | 
|  | DeletedDefaultConstructor() = delete; | 
|  | DeletedDefaultConstructor(int foo) : foo_(foo) {} | 
|  |  | 
|  | int foo() const { return foo_; } | 
|  |  | 
|  | private: | 
|  | int foo_; | 
|  | }; | 
|  |  | 
|  | class DeletedCopy { | 
|  | public: | 
|  | explicit DeletedCopy(int foo) : foo_(foo) {} | 
|  | DeletedCopy(const DeletedCopy&) = delete; | 
|  | DeletedCopy(DeletedCopy&&) = default; | 
|  |  | 
|  | DeletedCopy& operator=(const DeletedCopy&) = delete; | 
|  | DeletedCopy& operator=(DeletedCopy&&) = default; | 
|  |  | 
|  | int foo() const { return foo_; } | 
|  |  | 
|  | private: | 
|  | int foo_; | 
|  | }; | 
|  |  | 
|  | class DeletedMove { | 
|  | public: | 
|  | explicit DeletedMove(int foo) : foo_(foo) {} | 
|  | DeletedMove(const DeletedMove&) = default; | 
|  | DeletedMove(DeletedMove&&) = delete; | 
|  |  | 
|  | DeletedMove& operator=(const DeletedMove&) = default; | 
|  | DeletedMove& operator=(DeletedMove&&) = delete; | 
|  |  | 
|  | int foo() const { return foo_; } | 
|  |  | 
|  | private: | 
|  | int foo_; | 
|  | }; | 
|  |  | 
|  | class NonTriviallyDestructibleDeletedCopyConstructor { | 
|  | public: | 
|  | explicit NonTriviallyDestructibleDeletedCopyConstructor(int foo) | 
|  | : foo_(foo) {} | 
|  | NonTriviallyDestructibleDeletedCopyConstructor( | 
|  | const NonTriviallyDestructibleDeletedCopyConstructor&) = delete; | 
|  | NonTriviallyDestructibleDeletedCopyConstructor( | 
|  | NonTriviallyDestructibleDeletedCopyConstructor&&) = default; | 
|  |  | 
|  | ~NonTriviallyDestructibleDeletedCopyConstructor() {} | 
|  |  | 
|  | int foo() const { return foo_; } | 
|  |  | 
|  | private: | 
|  | int foo_; | 
|  | }; | 
|  |  | 
|  | class DeleteNewOperators { | 
|  | public: | 
|  | void* operator new(size_t) = delete; | 
|  | void* operator new(size_t, void*) = delete; | 
|  | void* operator new[](size_t) = delete; | 
|  | void* operator new[](size_t, void*) = delete; | 
|  | }; | 
|  |  | 
|  | }  // anonymous namespace | 
|  |  | 
|  | static_assert(std::is_trivially_destructible<Optional<int>>::value, | 
|  | "OptionalIsTriviallyDestructible"); | 
|  |  | 
|  | static_assert( | 
|  | !std::is_trivially_destructible<Optional<NonTriviallyDestructible>>::value, | 
|  | "OptionalIsTriviallyDestructible"); | 
|  |  | 
|  | static_assert(sizeof(Optional<int>) == sizeof(internal::OptionalBase<int>), | 
|  | "internal::{Copy,Move}{Constructible,Assignable} structs " | 
|  | "should be 0-sized"); | 
|  |  | 
|  | TEST(OptionalTest, DefaultConstructor) { | 
|  | { | 
|  | constexpr Optional<float> o; | 
|  | EXPECT_FALSE(o); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<std::string> o; | 
|  | EXPECT_FALSE(o); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<TestObject> o; | 
|  | EXPECT_FALSE(o); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, CopyConstructor) { | 
|  | { | 
|  | constexpr Optional<float> first(0.1f); | 
|  | constexpr Optional<float> other(first); | 
|  |  | 
|  | EXPECT_TRUE(other); | 
|  | EXPECT_EQ(other.value(), 0.1f); | 
|  | EXPECT_EQ(first, other); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<std::string> first("foo"); | 
|  | Optional<std::string> other(first); | 
|  |  | 
|  | EXPECT_TRUE(other); | 
|  | EXPECT_EQ(other.value(), "foo"); | 
|  | EXPECT_EQ(first, other); | 
|  | } | 
|  |  | 
|  | { | 
|  | const Optional<std::string> first("foo"); | 
|  | Optional<std::string> other(first); | 
|  |  | 
|  | EXPECT_TRUE(other); | 
|  | EXPECT_EQ(other.value(), "foo"); | 
|  | EXPECT_EQ(first, other); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<TestObject> first(TestObject(3, 0.1)); | 
|  | Optional<TestObject> other(first); | 
|  |  | 
|  | EXPECT_TRUE(!!other); | 
|  | EXPECT_TRUE(other.value() == TestObject(3, 0.1)); | 
|  | EXPECT_TRUE(first == other); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, ValueConstructor) { | 
|  | { | 
|  | constexpr float value = 0.1f; | 
|  | constexpr Optional<float> o(value); | 
|  |  | 
|  | EXPECT_TRUE(o); | 
|  | EXPECT_EQ(value, o.value()); | 
|  | } | 
|  |  | 
|  | { | 
|  | std::string value("foo"); | 
|  | Optional<std::string> o(value); | 
|  |  | 
|  | EXPECT_TRUE(o); | 
|  | EXPECT_EQ(value, o.value()); | 
|  | } | 
|  |  | 
|  | { | 
|  | TestObject value(3, 0.1); | 
|  | Optional<TestObject> o(value); | 
|  |  | 
|  | EXPECT_TRUE(o); | 
|  | EXPECT_EQ(TestObject::State::COPY_CONSTRUCTED, o->state()); | 
|  | EXPECT_EQ(value, o.value()); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, MoveConstructor) { | 
|  | { | 
|  | constexpr Optional<float> first(0.1f); | 
|  | constexpr Optional<float> second(std::move(first)); | 
|  |  | 
|  | EXPECT_TRUE(second.has_value()); | 
|  | EXPECT_EQ(second.value(), 0.1f); | 
|  |  | 
|  | EXPECT_TRUE(first.has_value()); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<std::string> first("foo"); | 
|  | Optional<std::string> second(std::move(first)); | 
|  |  | 
|  | EXPECT_TRUE(second.has_value()); | 
|  | EXPECT_EQ("foo", second.value()); | 
|  |  | 
|  | EXPECT_TRUE(first.has_value()); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<TestObject> first(TestObject(3, 0.1)); | 
|  | Optional<TestObject> second(std::move(first)); | 
|  |  | 
|  | EXPECT_TRUE(second.has_value()); | 
|  | EXPECT_EQ(TestObject::State::MOVE_CONSTRUCTED, second->state()); | 
|  | EXPECT_TRUE(TestObject(3, 0.1) == second.value()); | 
|  |  | 
|  | EXPECT_TRUE(first.has_value()); | 
|  | EXPECT_EQ(TestObject::State::MOVED_FROM, first->state()); | 
|  | } | 
|  |  | 
|  | // Even if copy constructor is deleted, move constructor needs to work. | 
|  | // Note that it couldn't be constexpr. | 
|  | { | 
|  | Optional<DeletedCopy> first(in_place, 42); | 
|  | Optional<DeletedCopy> second(std::move(first)); | 
|  |  | 
|  | EXPECT_TRUE(second.has_value()); | 
|  | EXPECT_EQ(42, second->foo()); | 
|  |  | 
|  | EXPECT_TRUE(first.has_value()); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<DeletedMove> first(in_place, 42); | 
|  | Optional<DeletedMove> second(std::move(first)); | 
|  |  | 
|  | EXPECT_TRUE(second.has_value()); | 
|  | EXPECT_EQ(42, second->foo()); | 
|  |  | 
|  | EXPECT_TRUE(first.has_value()); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<NonTriviallyDestructibleDeletedCopyConstructor> first(in_place, | 
|  | 42); | 
|  | Optional<NonTriviallyDestructibleDeletedCopyConstructor> second( | 
|  | std::move(first)); | 
|  |  | 
|  | EXPECT_TRUE(second.has_value()); | 
|  | EXPECT_EQ(42, second->foo()); | 
|  |  | 
|  | EXPECT_TRUE(first.has_value()); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, MoveValueConstructor) { | 
|  | { | 
|  | constexpr float value = 0.1f; | 
|  | constexpr Optional<float> o(std::move(value)); | 
|  |  | 
|  | EXPECT_TRUE(o); | 
|  | EXPECT_EQ(0.1f, o.value()); | 
|  | } | 
|  |  | 
|  | { | 
|  | float value = 0.1f; | 
|  | Optional<float> o(std::move(value)); | 
|  |  | 
|  | EXPECT_TRUE(o); | 
|  | EXPECT_EQ(0.1f, o.value()); | 
|  | } | 
|  |  | 
|  | { | 
|  | std::string value("foo"); | 
|  | Optional<std::string> o(std::move(value)); | 
|  |  | 
|  | EXPECT_TRUE(o); | 
|  | EXPECT_EQ("foo", o.value()); | 
|  | } | 
|  |  | 
|  | { | 
|  | TestObject value(3, 0.1); | 
|  | Optional<TestObject> o(std::move(value)); | 
|  |  | 
|  | EXPECT_TRUE(o); | 
|  | EXPECT_EQ(TestObject::State::MOVE_CONSTRUCTED, o->state()); | 
|  | EXPECT_EQ(TestObject(3, 0.1), o.value()); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, ConvertingCopyConstructor) { | 
|  | { | 
|  | Optional<int> first(1); | 
|  | Optional<double> second(first); | 
|  | EXPECT_TRUE(second.has_value()); | 
|  | EXPECT_EQ(1.0, second.value()); | 
|  | } | 
|  |  | 
|  | // Make sure explicit is not marked for convertible case. | 
|  | { | 
|  | Optional<int> o(1); | 
|  | ignore_result<Optional<double>>(o); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, ConvertingMoveConstructor) { | 
|  | { | 
|  | Optional<int> first(1); | 
|  | Optional<double> second(std::move(first)); | 
|  | EXPECT_TRUE(second.has_value()); | 
|  | EXPECT_EQ(1.0, second.value()); | 
|  | } | 
|  |  | 
|  | // Make sure explicit is not marked for convertible case. | 
|  | { | 
|  | Optional<int> o(1); | 
|  | ignore_result<Optional<double>>(std::move(o)); | 
|  | } | 
|  |  | 
|  | { | 
|  | class Test1 { | 
|  | public: | 
|  | explicit Test1(int foo) : foo_(foo) {} | 
|  |  | 
|  | int foo() const { return foo_; } | 
|  |  | 
|  | private: | 
|  | int foo_; | 
|  | }; | 
|  |  | 
|  | // Not copyable but convertible from Test1. | 
|  | class Test2 { | 
|  | public: | 
|  | Test2(const Test2&) = delete; | 
|  | explicit Test2(Test1&& other) : bar_(other.foo()) {} | 
|  |  | 
|  | double bar() const { return bar_; } | 
|  |  | 
|  | private: | 
|  | double bar_; | 
|  | }; | 
|  |  | 
|  | Optional<Test1> first(in_place, 42); | 
|  | Optional<Test2> second(std::move(first)); | 
|  | EXPECT_TRUE(second.has_value()); | 
|  | EXPECT_EQ(42.0, second->bar()); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, ConstructorForwardArguments) { | 
|  | { | 
|  | constexpr Optional<float> a(base::in_place, 0.1f); | 
|  | EXPECT_TRUE(a); | 
|  | EXPECT_EQ(0.1f, a.value()); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<float> a(base::in_place, 0.1f); | 
|  | EXPECT_TRUE(a); | 
|  | EXPECT_EQ(0.1f, a.value()); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<std::string> a(base::in_place, "foo"); | 
|  | EXPECT_TRUE(a); | 
|  | EXPECT_EQ("foo", a.value()); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<TestObject> a(base::in_place, 0, 0.1); | 
|  | EXPECT_TRUE(!!a); | 
|  | EXPECT_TRUE(TestObject(0, 0.1) == a.value()); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, ConstructorForwardInitListAndArguments) { | 
|  | { | 
|  | Optional<std::vector<int>> opt(in_place, {3, 1}); | 
|  | EXPECT_TRUE(opt); | 
|  | EXPECT_THAT(*opt, ElementsAre(3, 1)); | 
|  | EXPECT_EQ(2u, opt->size()); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<std::vector<int>> opt(in_place, {3, 1}, std::allocator<int>()); | 
|  | EXPECT_TRUE(opt); | 
|  | EXPECT_THAT(*opt, ElementsAre(3, 1)); | 
|  | EXPECT_EQ(2u, opt->size()); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, ForwardConstructor) { | 
|  | { | 
|  | Optional<double> a(1); | 
|  | EXPECT_TRUE(a.has_value()); | 
|  | EXPECT_EQ(1.0, a.value()); | 
|  | } | 
|  |  | 
|  | // Test that default type of 'U' is value_type. | 
|  | { | 
|  | struct TestData { | 
|  | int a; | 
|  | double b; | 
|  | bool c; | 
|  | }; | 
|  |  | 
|  | Optional<TestData> a({1, 2.0, true}); | 
|  | EXPECT_TRUE(a.has_value()); | 
|  | EXPECT_EQ(1, a->a); | 
|  | EXPECT_EQ(2.0, a->b); | 
|  | EXPECT_TRUE(a->c); | 
|  | } | 
|  |  | 
|  | // If T has a constructor with a param Optional<U>, and another ctor with a | 
|  | // param U, then T(Optional<U>) should be used for Optional<T>(Optional<U>) | 
|  | // constructor. | 
|  | { | 
|  | enum class ParamType { | 
|  | DEFAULT_CONSTRUCTED, | 
|  | COPY_CONSTRUCTED, | 
|  | MOVE_CONSTRUCTED, | 
|  | INT, | 
|  | IN_PLACE, | 
|  | OPTIONAL_INT, | 
|  | }; | 
|  | struct Test { | 
|  | Test() : param_type(ParamType::DEFAULT_CONSTRUCTED) {} | 
|  | Test(const Test&) : param_type(ParamType::COPY_CONSTRUCTED) {} | 
|  | Test(Test&&) : param_type(ParamType::MOVE_CONSTRUCTED) {} | 
|  | explicit Test(int) : param_type(ParamType::INT) {} | 
|  | explicit Test(in_place_t) : param_type(ParamType::IN_PLACE) {} | 
|  | explicit Test(Optional<int>) : param_type(ParamType::OPTIONAL_INT) {} | 
|  |  | 
|  | ParamType param_type; | 
|  | }; | 
|  |  | 
|  | // Overload resolution with copy-conversion constructor. | 
|  | { | 
|  | const Optional<int> arg(in_place, 1); | 
|  | Optional<Test> testee(arg); | 
|  | EXPECT_EQ(ParamType::OPTIONAL_INT, testee->param_type); | 
|  | } | 
|  |  | 
|  | // Overload resolution with move conversion constructor. | 
|  | { | 
|  | Optional<Test> testee(Optional<int>(in_place, 1)); | 
|  | EXPECT_EQ(ParamType::OPTIONAL_INT, testee->param_type); | 
|  | } | 
|  |  | 
|  | // Default constructor should be used. | 
|  | { | 
|  | Optional<Test> testee(in_place); | 
|  | EXPECT_EQ(ParamType::DEFAULT_CONSTRUCTED, testee->param_type); | 
|  | } | 
|  | } | 
|  |  | 
|  | { | 
|  | struct Test { | 
|  | Test(int) {}  // NOLINT(runtime/explicit) | 
|  | }; | 
|  | // If T is convertible from U, it is not marked as explicit. | 
|  | static_assert(std::is_convertible<int, Test>::value, | 
|  | "Int should be convertible to Test."); | 
|  | ([](Optional<Test>) {})(1); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, NulloptConstructor) { | 
|  | constexpr Optional<int> a(base::nullopt); | 
|  | EXPECT_FALSE(a); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, AssignValue) { | 
|  | { | 
|  | Optional<float> a; | 
|  | EXPECT_FALSE(a); | 
|  | a = 0.1f; | 
|  | EXPECT_TRUE(a); | 
|  |  | 
|  | Optional<float> b(0.1f); | 
|  | EXPECT_TRUE(a == b); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<std::string> a; | 
|  | EXPECT_FALSE(a); | 
|  | a = std::string("foo"); | 
|  | EXPECT_TRUE(a); | 
|  |  | 
|  | Optional<std::string> b(std::string("foo")); | 
|  | EXPECT_EQ(a, b); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<TestObject> a; | 
|  | EXPECT_FALSE(!!a); | 
|  | a = TestObject(3, 0.1); | 
|  | EXPECT_TRUE(!!a); | 
|  |  | 
|  | Optional<TestObject> b(TestObject(3, 0.1)); | 
|  | EXPECT_TRUE(a == b); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<TestObject> a = TestObject(4, 1.0); | 
|  | EXPECT_TRUE(!!a); | 
|  | a = TestObject(3, 0.1); | 
|  | EXPECT_TRUE(!!a); | 
|  |  | 
|  | Optional<TestObject> b(TestObject(3, 0.1)); | 
|  | EXPECT_TRUE(a == b); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, AssignObject) { | 
|  | { | 
|  | Optional<float> a; | 
|  | Optional<float> b(0.1f); | 
|  | a = b; | 
|  |  | 
|  | EXPECT_TRUE(a); | 
|  | EXPECT_EQ(a.value(), 0.1f); | 
|  | EXPECT_EQ(a, b); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<std::string> a; | 
|  | Optional<std::string> b("foo"); | 
|  | a = b; | 
|  |  | 
|  | EXPECT_TRUE(a); | 
|  | EXPECT_EQ(a.value(), "foo"); | 
|  | EXPECT_EQ(a, b); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<TestObject> a; | 
|  | Optional<TestObject> b(TestObject(3, 0.1)); | 
|  | a = b; | 
|  |  | 
|  | EXPECT_TRUE(!!a); | 
|  | EXPECT_TRUE(a.value() == TestObject(3, 0.1)); | 
|  | EXPECT_TRUE(a == b); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<TestObject> a(TestObject(4, 1.0)); | 
|  | Optional<TestObject> b(TestObject(3, 0.1)); | 
|  | a = b; | 
|  |  | 
|  | EXPECT_TRUE(!!a); | 
|  | EXPECT_TRUE(a.value() == TestObject(3, 0.1)); | 
|  | EXPECT_TRUE(a == b); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<DeletedMove> a(in_place, 42); | 
|  | Optional<DeletedMove> b; | 
|  | b = a; | 
|  |  | 
|  | EXPECT_TRUE(!!a); | 
|  | EXPECT_TRUE(!!b); | 
|  | EXPECT_EQ(a->foo(), b->foo()); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<DeletedMove> a(in_place, 42); | 
|  | Optional<DeletedMove> b(in_place, 1); | 
|  | b = a; | 
|  |  | 
|  | EXPECT_TRUE(!!a); | 
|  | EXPECT_TRUE(!!b); | 
|  | EXPECT_EQ(a->foo(), b->foo()); | 
|  | } | 
|  |  | 
|  | // Converting assignment. | 
|  | { | 
|  | Optional<int> a(in_place, 1); | 
|  | Optional<double> b; | 
|  | b = a; | 
|  |  | 
|  | EXPECT_TRUE(!!a); | 
|  | EXPECT_TRUE(!!b); | 
|  | EXPECT_EQ(1, a.value()); | 
|  | EXPECT_EQ(1.0, b.value()); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<int> a(in_place, 42); | 
|  | Optional<double> b(in_place, 1); | 
|  | b = a; | 
|  |  | 
|  | EXPECT_TRUE(!!a); | 
|  | EXPECT_TRUE(!!b); | 
|  | EXPECT_EQ(42, a.value()); | 
|  | EXPECT_EQ(42.0, b.value()); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<int> a; | 
|  | Optional<double> b(in_place, 1); | 
|  | b = a; | 
|  | EXPECT_FALSE(!!a); | 
|  | EXPECT_FALSE(!!b); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, AssignObject_rvalue) { | 
|  | { | 
|  | Optional<float> a; | 
|  | Optional<float> b(0.1f); | 
|  | a = std::move(b); | 
|  |  | 
|  | EXPECT_TRUE(a); | 
|  | EXPECT_TRUE(b); | 
|  | EXPECT_EQ(0.1f, a.value()); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<std::string> a; | 
|  | Optional<std::string> b("foo"); | 
|  | a = std::move(b); | 
|  |  | 
|  | EXPECT_TRUE(a); | 
|  | EXPECT_TRUE(b); | 
|  | EXPECT_EQ("foo", a.value()); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<TestObject> a; | 
|  | Optional<TestObject> b(TestObject(3, 0.1)); | 
|  | a = std::move(b); | 
|  |  | 
|  | EXPECT_TRUE(!!a); | 
|  | EXPECT_TRUE(!!b); | 
|  | EXPECT_TRUE(TestObject(3, 0.1) == a.value()); | 
|  |  | 
|  | EXPECT_EQ(TestObject::State::MOVE_CONSTRUCTED, a->state()); | 
|  | EXPECT_EQ(TestObject::State::MOVED_FROM, b->state()); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<TestObject> a(TestObject(4, 1.0)); | 
|  | Optional<TestObject> b(TestObject(3, 0.1)); | 
|  | a = std::move(b); | 
|  |  | 
|  | EXPECT_TRUE(!!a); | 
|  | EXPECT_TRUE(!!b); | 
|  | EXPECT_TRUE(TestObject(3, 0.1) == a.value()); | 
|  |  | 
|  | EXPECT_EQ(TestObject::State::MOVE_ASSIGNED, a->state()); | 
|  | EXPECT_EQ(TestObject::State::MOVED_FROM, b->state()); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<DeletedMove> a(in_place, 42); | 
|  | Optional<DeletedMove> b; | 
|  | b = std::move(a); | 
|  |  | 
|  | EXPECT_TRUE(!!a); | 
|  | EXPECT_TRUE(!!b); | 
|  | EXPECT_EQ(42, b->foo()); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<DeletedMove> a(in_place, 42); | 
|  | Optional<DeletedMove> b(in_place, 1); | 
|  | b = std::move(a); | 
|  |  | 
|  | EXPECT_TRUE(!!a); | 
|  | EXPECT_TRUE(!!b); | 
|  | EXPECT_EQ(42, b->foo()); | 
|  | } | 
|  |  | 
|  | // Converting assignment. | 
|  | { | 
|  | Optional<int> a(in_place, 1); | 
|  | Optional<double> b; | 
|  | b = std::move(a); | 
|  |  | 
|  | EXPECT_TRUE(!!a); | 
|  | EXPECT_TRUE(!!b); | 
|  | EXPECT_EQ(1.0, b.value()); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<int> a(in_place, 42); | 
|  | Optional<double> b(in_place, 1); | 
|  | b = std::move(a); | 
|  |  | 
|  | EXPECT_TRUE(!!a); | 
|  | EXPECT_TRUE(!!b); | 
|  | EXPECT_EQ(42.0, b.value()); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<int> a; | 
|  | Optional<double> b(in_place, 1); | 
|  | b = std::move(a); | 
|  |  | 
|  | EXPECT_FALSE(!!a); | 
|  | EXPECT_FALSE(!!b); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, AssignNull) { | 
|  | { | 
|  | Optional<float> a(0.1f); | 
|  | Optional<float> b(0.2f); | 
|  | a = base::nullopt; | 
|  | b = base::nullopt; | 
|  | EXPECT_EQ(a, b); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<std::string> a("foo"); | 
|  | Optional<std::string> b("bar"); | 
|  | a = base::nullopt; | 
|  | b = base::nullopt; | 
|  | EXPECT_EQ(a, b); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<TestObject> a(TestObject(3, 0.1)); | 
|  | Optional<TestObject> b(TestObject(4, 1.0)); | 
|  | a = base::nullopt; | 
|  | b = base::nullopt; | 
|  | EXPECT_TRUE(a == b); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, AssignOverload) { | 
|  | struct Test1 { | 
|  | enum class State { | 
|  | CONSTRUCTED, | 
|  | MOVED, | 
|  | }; | 
|  | State state = State::CONSTRUCTED; | 
|  | }; | 
|  |  | 
|  | // Here, Optional<Test2> can be assigned from Optioanl<Test1>. | 
|  | // In case of move, marks MOVED to Test1 instance. | 
|  | struct Test2 { | 
|  | enum class State { | 
|  | DEFAULT_CONSTRUCTED, | 
|  | COPY_CONSTRUCTED_FROM_TEST1, | 
|  | MOVE_CONSTRUCTED_FROM_TEST1, | 
|  | COPY_ASSIGNED_FROM_TEST1, | 
|  | MOVE_ASSIGNED_FROM_TEST1, | 
|  | }; | 
|  |  | 
|  | Test2() = default; | 
|  | explicit Test2(const Test1&) : state(State::COPY_CONSTRUCTED_FROM_TEST1) {} | 
|  | explicit Test2(Test1&& test1) : state(State::MOVE_CONSTRUCTED_FROM_TEST1) { | 
|  | test1.state = Test1::State::MOVED; | 
|  | } | 
|  | Test2& operator=(const Test1&) { | 
|  | state = State::COPY_ASSIGNED_FROM_TEST1; | 
|  | return *this; | 
|  | } | 
|  | Test2& operator=(Test1&& test1) { | 
|  | state = State::MOVE_ASSIGNED_FROM_TEST1; | 
|  | test1.state = Test1::State::MOVED; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | State state = State::DEFAULT_CONSTRUCTED; | 
|  | }; | 
|  |  | 
|  | { | 
|  | Optional<Test1> a(in_place); | 
|  | Optional<Test2> b; | 
|  |  | 
|  | b = a; | 
|  | EXPECT_TRUE(!!a); | 
|  | EXPECT_TRUE(!!b); | 
|  | EXPECT_EQ(Test1::State::CONSTRUCTED, a->state); | 
|  | EXPECT_EQ(Test2::State::COPY_CONSTRUCTED_FROM_TEST1, b->state); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<Test1> a(in_place); | 
|  | Optional<Test2> b(in_place); | 
|  |  | 
|  | b = a; | 
|  | EXPECT_TRUE(!!a); | 
|  | EXPECT_TRUE(!!b); | 
|  | EXPECT_EQ(Test1::State::CONSTRUCTED, a->state); | 
|  | EXPECT_EQ(Test2::State::COPY_ASSIGNED_FROM_TEST1, b->state); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<Test1> a(in_place); | 
|  | Optional<Test2> b; | 
|  |  | 
|  | b = std::move(a); | 
|  | EXPECT_TRUE(!!a); | 
|  | EXPECT_TRUE(!!b); | 
|  | EXPECT_EQ(Test1::State::MOVED, a->state); | 
|  | EXPECT_EQ(Test2::State::MOVE_CONSTRUCTED_FROM_TEST1, b->state); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<Test1> a(in_place); | 
|  | Optional<Test2> b(in_place); | 
|  |  | 
|  | b = std::move(a); | 
|  | EXPECT_TRUE(!!a); | 
|  | EXPECT_TRUE(!!b); | 
|  | EXPECT_EQ(Test1::State::MOVED, a->state); | 
|  | EXPECT_EQ(Test2::State::MOVE_ASSIGNED_FROM_TEST1, b->state); | 
|  | } | 
|  |  | 
|  | // Similar to Test2, but Test3 also has copy/move ctor and assign operators | 
|  | // from Optional<Test1>, too. In this case, for a = b where a is | 
|  | // Optional<Test3> and b is Optional<Test1>, | 
|  | // Optional<T>::operator=(U&&) where U is Optional<Test1> should be used | 
|  | // rather than Optional<T>::operator=(Optional<U>&&) where U is Test1. | 
|  | struct Test3 { | 
|  | enum class State { | 
|  | DEFAULT_CONSTRUCTED, | 
|  | COPY_CONSTRUCTED_FROM_TEST1, | 
|  | MOVE_CONSTRUCTED_FROM_TEST1, | 
|  | COPY_CONSTRUCTED_FROM_OPTIONAL_TEST1, | 
|  | MOVE_CONSTRUCTED_FROM_OPTIONAL_TEST1, | 
|  | COPY_ASSIGNED_FROM_TEST1, | 
|  | MOVE_ASSIGNED_FROM_TEST1, | 
|  | COPY_ASSIGNED_FROM_OPTIONAL_TEST1, | 
|  | MOVE_ASSIGNED_FROM_OPTIONAL_TEST1, | 
|  | }; | 
|  |  | 
|  | Test3() = default; | 
|  | explicit Test3(const Test1&) : state(State::COPY_CONSTRUCTED_FROM_TEST1) {} | 
|  | explicit Test3(Test1&& test1) : state(State::MOVE_CONSTRUCTED_FROM_TEST1) { | 
|  | test1.state = Test1::State::MOVED; | 
|  | } | 
|  | explicit Test3(const Optional<Test1>&) | 
|  | : state(State::COPY_CONSTRUCTED_FROM_OPTIONAL_TEST1) {} | 
|  | explicit Test3(Optional<Test1>&& test1) | 
|  | : state(State::MOVE_CONSTRUCTED_FROM_OPTIONAL_TEST1) { | 
|  | // In the following senarios, given |test1| should always have value. | 
|  | PERFETTO_DCHECK(test1.has_value()); | 
|  | test1->state = Test1::State::MOVED; | 
|  | } | 
|  | Test3& operator=(const Test1&) { | 
|  | state = State::COPY_ASSIGNED_FROM_TEST1; | 
|  | return *this; | 
|  | } | 
|  | Test3& operator=(Test1&& test1) { | 
|  | state = State::MOVE_ASSIGNED_FROM_TEST1; | 
|  | test1.state = Test1::State::MOVED; | 
|  | return *this; | 
|  | } | 
|  | Test3& operator=(const Optional<Test1>&) { | 
|  | state = State::COPY_ASSIGNED_FROM_OPTIONAL_TEST1; | 
|  | return *this; | 
|  | } | 
|  | Test3& operator=(Optional<Test1>&& test1) { | 
|  | state = State::MOVE_ASSIGNED_FROM_OPTIONAL_TEST1; | 
|  | // In the following senarios, given |test1| should always have value. | 
|  | PERFETTO_DCHECK(test1.has_value()); | 
|  | test1->state = Test1::State::MOVED; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | State state = State::DEFAULT_CONSTRUCTED; | 
|  | }; | 
|  |  | 
|  | { | 
|  | Optional<Test1> a(in_place); | 
|  | Optional<Test3> b; | 
|  |  | 
|  | b = a; | 
|  | EXPECT_TRUE(!!a); | 
|  | EXPECT_TRUE(!!b); | 
|  | EXPECT_EQ(Test1::State::CONSTRUCTED, a->state); | 
|  | EXPECT_EQ(Test3::State::COPY_CONSTRUCTED_FROM_OPTIONAL_TEST1, b->state); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<Test1> a(in_place); | 
|  | Optional<Test3> b(in_place); | 
|  |  | 
|  | b = a; | 
|  | EXPECT_TRUE(!!a); | 
|  | EXPECT_TRUE(!!b); | 
|  | EXPECT_EQ(Test1::State::CONSTRUCTED, a->state); | 
|  | EXPECT_EQ(Test3::State::COPY_ASSIGNED_FROM_OPTIONAL_TEST1, b->state); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<Test1> a(in_place); | 
|  | Optional<Test3> b; | 
|  |  | 
|  | b = std::move(a); | 
|  | EXPECT_TRUE(!!a); | 
|  | EXPECT_TRUE(!!b); | 
|  | EXPECT_EQ(Test1::State::MOVED, a->state); | 
|  | EXPECT_EQ(Test3::State::MOVE_CONSTRUCTED_FROM_OPTIONAL_TEST1, b->state); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<Test1> a(in_place); | 
|  | Optional<Test3> b(in_place); | 
|  |  | 
|  | b = std::move(a); | 
|  | EXPECT_TRUE(!!a); | 
|  | EXPECT_TRUE(!!b); | 
|  | EXPECT_EQ(Test1::State::MOVED, a->state); | 
|  | EXPECT_EQ(Test3::State::MOVE_ASSIGNED_FROM_OPTIONAL_TEST1, b->state); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, OperatorStar) { | 
|  | { | 
|  | Optional<float> a(0.1f); | 
|  | EXPECT_EQ(a.value(), *a); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<std::string> a("foo"); | 
|  | EXPECT_EQ(a.value(), *a); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<TestObject> a(TestObject(3, 0.1)); | 
|  | EXPECT_EQ(a.value(), *a); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, OperatorStar_rvalue) { | 
|  | EXPECT_EQ(0.1f, *Optional<float>(0.1f)); | 
|  | EXPECT_EQ(std::string("foo"), *Optional<std::string>("foo")); | 
|  | EXPECT_TRUE(TestObject(3, 0.1) == *Optional<TestObject>(TestObject(3, 0.1))); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, OperatorArrow) { | 
|  | Optional<TestObject> a(TestObject(3, 0.1)); | 
|  | EXPECT_EQ(a->foo(), 3); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, Value_rvalue) { | 
|  | EXPECT_EQ(0.1f, Optional<float>(0.1f).value()); | 
|  | EXPECT_EQ(std::string("foo"), Optional<std::string>("foo").value()); | 
|  | EXPECT_TRUE(TestObject(3, 0.1) == | 
|  | Optional<TestObject>(TestObject(3, 0.1)).value()); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, ValueOr) { | 
|  | { | 
|  | Optional<float> a; | 
|  | EXPECT_EQ(0.0f, a.value_or(0.0f)); | 
|  |  | 
|  | a = 0.1f; | 
|  | EXPECT_EQ(0.1f, a.value_or(0.0f)); | 
|  |  | 
|  | a = base::nullopt; | 
|  | EXPECT_EQ(0.0f, a.value_or(0.0f)); | 
|  | } | 
|  |  | 
|  | // value_or() can be constexpr. | 
|  | { | 
|  | constexpr Optional<int> a(in_place, 1); | 
|  | constexpr int value = a.value_or(10); | 
|  | EXPECT_EQ(1, value); | 
|  | } | 
|  | { | 
|  | constexpr Optional<int> a; | 
|  | constexpr int value = a.value_or(10); | 
|  | EXPECT_EQ(10, value); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<std::string> a; | 
|  | EXPECT_EQ("bar", a.value_or("bar")); | 
|  |  | 
|  | a = std::string("foo"); | 
|  | EXPECT_EQ(std::string("foo"), a.value_or("bar")); | 
|  |  | 
|  | a = base::nullopt; | 
|  | EXPECT_EQ(std::string("bar"), a.value_or("bar")); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<TestObject> a; | 
|  | EXPECT_TRUE(a.value_or(TestObject(1, 0.3)) == TestObject(1, 0.3)); | 
|  |  | 
|  | a = TestObject(3, 0.1); | 
|  | EXPECT_TRUE(a.value_or(TestObject(1, 0.3)) == TestObject(3, 0.1)); | 
|  |  | 
|  | a = base::nullopt; | 
|  | EXPECT_TRUE(a.value_or(TestObject(1, 0.3)) == TestObject(1, 0.3)); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, Swap_bothNoValue) { | 
|  | Optional<TestObject> a, b; | 
|  | a.swap(b); | 
|  |  | 
|  | EXPECT_FALSE(a); | 
|  | EXPECT_FALSE(b); | 
|  | EXPECT_TRUE(TestObject(42, 0.42) == a.value_or(TestObject(42, 0.42))); | 
|  | EXPECT_TRUE(TestObject(42, 0.42) == b.value_or(TestObject(42, 0.42))); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, Swap_inHasValue) { | 
|  | Optional<TestObject> a(TestObject(1, 0.3)); | 
|  | Optional<TestObject> b; | 
|  | a.swap(b); | 
|  |  | 
|  | EXPECT_FALSE(a); | 
|  |  | 
|  | EXPECT_TRUE(!!b); | 
|  | EXPECT_TRUE(TestObject(42, 0.42) == a.value_or(TestObject(42, 0.42))); | 
|  | EXPECT_TRUE(TestObject(1, 0.3) == b.value_or(TestObject(42, 0.42))); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, Swap_outHasValue) { | 
|  | Optional<TestObject> a; | 
|  | Optional<TestObject> b(TestObject(1, 0.3)); | 
|  | a.swap(b); | 
|  |  | 
|  | EXPECT_TRUE(!!a); | 
|  | EXPECT_FALSE(!!b); | 
|  | EXPECT_TRUE(TestObject(1, 0.3) == a.value_or(TestObject(42, 0.42))); | 
|  | EXPECT_TRUE(TestObject(42, 0.42) == b.value_or(TestObject(42, 0.42))); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, Swap_bothValue) { | 
|  | Optional<TestObject> a(TestObject(0, 0.1)); | 
|  | Optional<TestObject> b(TestObject(1, 0.3)); | 
|  | a.swap(b); | 
|  |  | 
|  | EXPECT_TRUE(!!a); | 
|  | EXPECT_TRUE(!!b); | 
|  | EXPECT_TRUE(TestObject(1, 0.3) == a.value_or(TestObject(42, 0.42))); | 
|  | EXPECT_TRUE(TestObject(0, 0.1) == b.value_or(TestObject(42, 0.42))); | 
|  | EXPECT_EQ(TestObject::State::SWAPPED, a->state()); | 
|  | EXPECT_EQ(TestObject::State::SWAPPED, b->state()); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, Emplace) { | 
|  | { | 
|  | Optional<float> a(0.1f); | 
|  | EXPECT_EQ(0.3f, a.emplace(0.3f)); | 
|  |  | 
|  | EXPECT_TRUE(a); | 
|  | EXPECT_EQ(0.3f, a.value()); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<std::string> a("foo"); | 
|  | EXPECT_EQ("bar", a.emplace("bar")); | 
|  |  | 
|  | EXPECT_TRUE(a); | 
|  | EXPECT_EQ("bar", a.value()); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<TestObject> a(TestObject(0, 0.1)); | 
|  | EXPECT_EQ(TestObject(1, 0.2), a.emplace(TestObject(1, 0.2))); | 
|  |  | 
|  | EXPECT_TRUE(!!a); | 
|  | EXPECT_TRUE(TestObject(1, 0.2) == a.value()); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<std::vector<int>> a; | 
|  | auto& ref = a.emplace({2, 3}); | 
|  | static_assert(std::is_same<std::vector<int>&, decltype(ref)>::value, ""); | 
|  | EXPECT_TRUE(a); | 
|  | EXPECT_THAT(*a, ElementsAre(2, 3)); | 
|  | EXPECT_EQ(&ref, &*a); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<std::vector<int>> a; | 
|  | auto& ref = a.emplace({4, 5}, std::allocator<int>()); | 
|  | static_assert(std::is_same<std::vector<int>&, decltype(ref)>::value, ""); | 
|  | EXPECT_TRUE(a); | 
|  | EXPECT_THAT(*a, ElementsAre(4, 5)); | 
|  | EXPECT_EQ(&ref, &*a); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, Equals_TwoEmpty) { | 
|  | Optional<int> a; | 
|  | Optional<int> b; | 
|  |  | 
|  | EXPECT_TRUE(a == b); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, Equals_TwoEquals) { | 
|  | Optional<int> a(1); | 
|  | Optional<int> b(1); | 
|  |  | 
|  | EXPECT_TRUE(a == b); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, Equals_OneEmpty) { | 
|  | Optional<int> a; | 
|  | Optional<int> b(1); | 
|  |  | 
|  | EXPECT_FALSE(a == b); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, Equals_TwoDifferent) { | 
|  | Optional<int> a(0); | 
|  | Optional<int> b(1); | 
|  |  | 
|  | EXPECT_FALSE(a == b); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, Equals_DifferentType) { | 
|  | Optional<int> a(0); | 
|  | Optional<double> b(0); | 
|  |  | 
|  | EXPECT_TRUE(a == b); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, NotEquals_TwoEmpty) { | 
|  | Optional<int> a; | 
|  | Optional<int> b; | 
|  |  | 
|  | EXPECT_FALSE(a != b); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, NotEquals_TwoEquals) { | 
|  | Optional<int> a(1); | 
|  | Optional<int> b(1); | 
|  |  | 
|  | EXPECT_FALSE(a != b); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, NotEquals_OneEmpty) { | 
|  | Optional<int> a; | 
|  | Optional<int> b(1); | 
|  |  | 
|  | EXPECT_TRUE(a != b); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, NotEquals_TwoDifferent) { | 
|  | Optional<int> a(0); | 
|  | Optional<int> b(1); | 
|  |  | 
|  | EXPECT_TRUE(a != b); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, NotEquals_DifferentType) { | 
|  | Optional<int> a(0); | 
|  | Optional<double> b(0.0); | 
|  |  | 
|  | EXPECT_FALSE(a != b); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, Less_LeftEmpty) { | 
|  | Optional<int> l; | 
|  | Optional<int> r(1); | 
|  |  | 
|  | EXPECT_TRUE(l < r); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, Less_RightEmpty) { | 
|  | Optional<int> l(1); | 
|  | Optional<int> r; | 
|  |  | 
|  | EXPECT_FALSE(l < r); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, Less_BothEmpty) { | 
|  | Optional<int> l; | 
|  | Optional<int> r; | 
|  |  | 
|  | EXPECT_FALSE(l < r); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, Less_BothValues) { | 
|  | { | 
|  | Optional<int> l(1); | 
|  | Optional<int> r(2); | 
|  |  | 
|  | EXPECT_TRUE(l < r); | 
|  | } | 
|  | { | 
|  | Optional<int> l(2); | 
|  | Optional<int> r(1); | 
|  |  | 
|  | EXPECT_FALSE(l < r); | 
|  | } | 
|  | { | 
|  | Optional<int> l(1); | 
|  | Optional<int> r(1); | 
|  |  | 
|  | EXPECT_FALSE(l < r); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, Less_DifferentType) { | 
|  | Optional<int> l(1); | 
|  | Optional<double> r(2.0); | 
|  |  | 
|  | EXPECT_TRUE(l < r); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, LessEq_LeftEmpty) { | 
|  | Optional<int> l; | 
|  | Optional<int> r(1); | 
|  |  | 
|  | EXPECT_TRUE(l <= r); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, LessEq_RightEmpty) { | 
|  | Optional<int> l(1); | 
|  | Optional<int> r; | 
|  |  | 
|  | EXPECT_FALSE(l <= r); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, LessEq_BothEmpty) { | 
|  | Optional<int> l; | 
|  | Optional<int> r; | 
|  |  | 
|  | EXPECT_TRUE(l <= r); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, LessEq_BothValues) { | 
|  | { | 
|  | Optional<int> l(1); | 
|  | Optional<int> r(2); | 
|  |  | 
|  | EXPECT_TRUE(l <= r); | 
|  | } | 
|  | { | 
|  | Optional<int> l(2); | 
|  | Optional<int> r(1); | 
|  |  | 
|  | EXPECT_FALSE(l <= r); | 
|  | } | 
|  | { | 
|  | Optional<int> l(1); | 
|  | Optional<int> r(1); | 
|  |  | 
|  | EXPECT_TRUE(l <= r); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, LessEq_DifferentType) { | 
|  | Optional<int> l(1); | 
|  | Optional<double> r(2.0); | 
|  |  | 
|  | EXPECT_TRUE(l <= r); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, Greater_BothEmpty) { | 
|  | Optional<int> l; | 
|  | Optional<int> r; | 
|  |  | 
|  | EXPECT_FALSE(l > r); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, Greater_LeftEmpty) { | 
|  | Optional<int> l; | 
|  | Optional<int> r(1); | 
|  |  | 
|  | EXPECT_FALSE(l > r); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, Greater_RightEmpty) { | 
|  | Optional<int> l(1); | 
|  | Optional<int> r; | 
|  |  | 
|  | EXPECT_TRUE(l > r); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, Greater_BothValue) { | 
|  | { | 
|  | Optional<int> l(1); | 
|  | Optional<int> r(2); | 
|  |  | 
|  | EXPECT_FALSE(l > r); | 
|  | } | 
|  | { | 
|  | Optional<int> l(2); | 
|  | Optional<int> r(1); | 
|  |  | 
|  | EXPECT_TRUE(l > r); | 
|  | } | 
|  | { | 
|  | Optional<int> l(1); | 
|  | Optional<int> r(1); | 
|  |  | 
|  | EXPECT_FALSE(l > r); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, Greater_DifferentType) { | 
|  | Optional<int> l(1); | 
|  | Optional<double> r(2.0); | 
|  |  | 
|  | EXPECT_FALSE(l > r); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, GreaterEq_BothEmpty) { | 
|  | Optional<int> l; | 
|  | Optional<int> r; | 
|  |  | 
|  | EXPECT_TRUE(l >= r); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, GreaterEq_LeftEmpty) { | 
|  | Optional<int> l; | 
|  | Optional<int> r(1); | 
|  |  | 
|  | EXPECT_FALSE(l >= r); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, GreaterEq_RightEmpty) { | 
|  | Optional<int> l(1); | 
|  | Optional<int> r; | 
|  |  | 
|  | EXPECT_TRUE(l >= r); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, GreaterEq_BothValue) { | 
|  | { | 
|  | Optional<int> l(1); | 
|  | Optional<int> r(2); | 
|  |  | 
|  | EXPECT_FALSE(l >= r); | 
|  | } | 
|  | { | 
|  | Optional<int> l(2); | 
|  | Optional<int> r(1); | 
|  |  | 
|  | EXPECT_TRUE(l >= r); | 
|  | } | 
|  | { | 
|  | Optional<int> l(1); | 
|  | Optional<int> r(1); | 
|  |  | 
|  | EXPECT_TRUE(l >= r); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, GreaterEq_DifferentType) { | 
|  | Optional<int> l(1); | 
|  | Optional<double> r(2.0); | 
|  |  | 
|  | EXPECT_FALSE(l >= r); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, OptNullEq) { | 
|  | { | 
|  | Optional<int> opt; | 
|  | EXPECT_TRUE(opt == base::nullopt); | 
|  | } | 
|  | { | 
|  | Optional<int> opt(1); | 
|  | EXPECT_FALSE(opt == base::nullopt); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, NullOptEq) { | 
|  | { | 
|  | Optional<int> opt; | 
|  | EXPECT_TRUE(base::nullopt == opt); | 
|  | } | 
|  | { | 
|  | Optional<int> opt(1); | 
|  | EXPECT_FALSE(base::nullopt == opt); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, OptNullNotEq) { | 
|  | { | 
|  | Optional<int> opt; | 
|  | EXPECT_FALSE(opt != base::nullopt); | 
|  | } | 
|  | { | 
|  | Optional<int> opt(1); | 
|  | EXPECT_TRUE(opt != base::nullopt); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, NullOptNotEq) { | 
|  | { | 
|  | Optional<int> opt; | 
|  | EXPECT_FALSE(base::nullopt != opt); | 
|  | } | 
|  | { | 
|  | Optional<int> opt(1); | 
|  | EXPECT_TRUE(base::nullopt != opt); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, OptNullLower) { | 
|  | { | 
|  | Optional<int> opt; | 
|  | EXPECT_FALSE(opt < base::nullopt); | 
|  | } | 
|  | { | 
|  | Optional<int> opt(1); | 
|  | EXPECT_FALSE(opt < base::nullopt); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, NullOptLower) { | 
|  | { | 
|  | Optional<int> opt; | 
|  | EXPECT_FALSE(base::nullopt < opt); | 
|  | } | 
|  | { | 
|  | Optional<int> opt(1); | 
|  | EXPECT_TRUE(base::nullopt < opt); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, OptNullLowerEq) { | 
|  | { | 
|  | Optional<int> opt; | 
|  | EXPECT_TRUE(opt <= base::nullopt); | 
|  | } | 
|  | { | 
|  | Optional<int> opt(1); | 
|  | EXPECT_FALSE(opt <= base::nullopt); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, NullOptLowerEq) { | 
|  | { | 
|  | Optional<int> opt; | 
|  | EXPECT_TRUE(base::nullopt <= opt); | 
|  | } | 
|  | { | 
|  | Optional<int> opt(1); | 
|  | EXPECT_TRUE(base::nullopt <= opt); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, OptNullGreater) { | 
|  | { | 
|  | Optional<int> opt; | 
|  | EXPECT_FALSE(opt > base::nullopt); | 
|  | } | 
|  | { | 
|  | Optional<int> opt(1); | 
|  | EXPECT_TRUE(opt > base::nullopt); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, NullOptGreater) { | 
|  | { | 
|  | Optional<int> opt; | 
|  | EXPECT_FALSE(base::nullopt > opt); | 
|  | } | 
|  | { | 
|  | Optional<int> opt(1); | 
|  | EXPECT_FALSE(base::nullopt > opt); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, OptNullGreaterEq) { | 
|  | { | 
|  | Optional<int> opt; | 
|  | EXPECT_TRUE(opt >= base::nullopt); | 
|  | } | 
|  | { | 
|  | Optional<int> opt(1); | 
|  | EXPECT_TRUE(opt >= base::nullopt); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, NullOptGreaterEq) { | 
|  | { | 
|  | Optional<int> opt; | 
|  | EXPECT_TRUE(base::nullopt >= opt); | 
|  | } | 
|  | { | 
|  | Optional<int> opt(1); | 
|  | EXPECT_FALSE(base::nullopt >= opt); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, ValueEq_Empty) { | 
|  | Optional<int> opt; | 
|  | EXPECT_FALSE(opt == 1); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, ValueEq_NotEmpty) { | 
|  | { | 
|  | Optional<int> opt(0); | 
|  | EXPECT_FALSE(opt == 1); | 
|  | } | 
|  | { | 
|  | Optional<int> opt(1); | 
|  | EXPECT_TRUE(opt == 1); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, ValueEq_DifferentType) { | 
|  | Optional<int> opt(0); | 
|  | EXPECT_TRUE(opt == 0.0); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, EqValue_Empty) { | 
|  | Optional<int> opt; | 
|  | EXPECT_FALSE(1 == opt); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, EqValue_NotEmpty) { | 
|  | { | 
|  | Optional<int> opt(0); | 
|  | EXPECT_FALSE(1 == opt); | 
|  | } | 
|  | { | 
|  | Optional<int> opt(1); | 
|  | EXPECT_TRUE(1 == opt); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, EqValue_DifferentType) { | 
|  | Optional<int> opt(0); | 
|  | EXPECT_TRUE(0.0 == opt); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, ValueNotEq_Empty) { | 
|  | Optional<int> opt; | 
|  | EXPECT_TRUE(opt != 1); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, ValueNotEq_NotEmpty) { | 
|  | { | 
|  | Optional<int> opt(0); | 
|  | EXPECT_TRUE(opt != 1); | 
|  | } | 
|  | { | 
|  | Optional<int> opt(1); | 
|  | EXPECT_FALSE(opt != 1); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OPtionalTest, ValueNotEq_DifferentType) { | 
|  | Optional<int> opt(0); | 
|  | EXPECT_FALSE(opt != 0.0); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, NotEqValue_Empty) { | 
|  | Optional<int> opt; | 
|  | EXPECT_TRUE(1 != opt); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, NotEqValue_NotEmpty) { | 
|  | { | 
|  | Optional<int> opt(0); | 
|  | EXPECT_TRUE(1 != opt); | 
|  | } | 
|  | { | 
|  | Optional<int> opt(1); | 
|  | EXPECT_FALSE(1 != opt); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, NotEqValue_DifferentType) { | 
|  | Optional<int> opt(0); | 
|  | EXPECT_FALSE(0.0 != opt); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, ValueLess_Empty) { | 
|  | Optional<int> opt; | 
|  | EXPECT_TRUE(opt < 1); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, ValueLess_NotEmpty) { | 
|  | { | 
|  | Optional<int> opt(0); | 
|  | EXPECT_TRUE(opt < 1); | 
|  | } | 
|  | { | 
|  | Optional<int> opt(1); | 
|  | EXPECT_FALSE(opt < 1); | 
|  | } | 
|  | { | 
|  | Optional<int> opt(2); | 
|  | EXPECT_FALSE(opt < 1); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OPtionalTest, ValueLess_DifferentType) { | 
|  | Optional<int> opt(0); | 
|  | EXPECT_TRUE(opt < 1.0); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, LessValue_Empty) { | 
|  | Optional<int> opt; | 
|  | EXPECT_FALSE(1 < opt); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, LessValue_NotEmpty) { | 
|  | { | 
|  | Optional<int> opt(0); | 
|  | EXPECT_FALSE(1 < opt); | 
|  | } | 
|  | { | 
|  | Optional<int> opt(1); | 
|  | EXPECT_FALSE(1 < opt); | 
|  | } | 
|  | { | 
|  | Optional<int> opt(2); | 
|  | EXPECT_TRUE(1 < opt); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, LessValue_DifferentType) { | 
|  | Optional<int> opt(0); | 
|  | EXPECT_FALSE(0.0 < opt); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, ValueLessEq_Empty) { | 
|  | Optional<int> opt; | 
|  | EXPECT_TRUE(opt <= 1); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, ValueLessEq_NotEmpty) { | 
|  | { | 
|  | Optional<int> opt(0); | 
|  | EXPECT_TRUE(opt <= 1); | 
|  | } | 
|  | { | 
|  | Optional<int> opt(1); | 
|  | EXPECT_TRUE(opt <= 1); | 
|  | } | 
|  | { | 
|  | Optional<int> opt(2); | 
|  | EXPECT_FALSE(opt <= 1); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, ValueLessEq_DifferentType) { | 
|  | Optional<int> opt(0); | 
|  | EXPECT_TRUE(opt <= 0.0); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, LessEqValue_Empty) { | 
|  | Optional<int> opt; | 
|  | EXPECT_FALSE(1 <= opt); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, LessEqValue_NotEmpty) { | 
|  | { | 
|  | Optional<int> opt(0); | 
|  | EXPECT_FALSE(1 <= opt); | 
|  | } | 
|  | { | 
|  | Optional<int> opt(1); | 
|  | EXPECT_TRUE(1 <= opt); | 
|  | } | 
|  | { | 
|  | Optional<int> opt(2); | 
|  | EXPECT_TRUE(1 <= opt); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, LessEqValue_DifferentType) { | 
|  | Optional<int> opt(0); | 
|  | EXPECT_TRUE(0.0 <= opt); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, ValueGreater_Empty) { | 
|  | Optional<int> opt; | 
|  | EXPECT_FALSE(opt > 1); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, ValueGreater_NotEmpty) { | 
|  | { | 
|  | Optional<int> opt(0); | 
|  | EXPECT_FALSE(opt > 1); | 
|  | } | 
|  | { | 
|  | Optional<int> opt(1); | 
|  | EXPECT_FALSE(opt > 1); | 
|  | } | 
|  | { | 
|  | Optional<int> opt(2); | 
|  | EXPECT_TRUE(opt > 1); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, ValueGreater_DifferentType) { | 
|  | Optional<int> opt(0); | 
|  | EXPECT_FALSE(opt > 0.0); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, GreaterValue_Empty) { | 
|  | Optional<int> opt; | 
|  | EXPECT_TRUE(1 > opt); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, GreaterValue_NotEmpty) { | 
|  | { | 
|  | Optional<int> opt(0); | 
|  | EXPECT_TRUE(1 > opt); | 
|  | } | 
|  | { | 
|  | Optional<int> opt(1); | 
|  | EXPECT_FALSE(1 > opt); | 
|  | } | 
|  | { | 
|  | Optional<int> opt(2); | 
|  | EXPECT_FALSE(1 > opt); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, GreaterValue_DifferentType) { | 
|  | Optional<int> opt(0); | 
|  | EXPECT_FALSE(0.0 > opt); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, ValueGreaterEq_Empty) { | 
|  | Optional<int> opt; | 
|  | EXPECT_FALSE(opt >= 1); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, ValueGreaterEq_NotEmpty) { | 
|  | { | 
|  | Optional<int> opt(0); | 
|  | EXPECT_FALSE(opt >= 1); | 
|  | } | 
|  | { | 
|  | Optional<int> opt(1); | 
|  | EXPECT_TRUE(opt >= 1); | 
|  | } | 
|  | { | 
|  | Optional<int> opt(2); | 
|  | EXPECT_TRUE(opt >= 1); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, ValueGreaterEq_DifferentType) { | 
|  | Optional<int> opt(0); | 
|  | EXPECT_TRUE(opt <= 0.0); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, GreaterEqValue_Empty) { | 
|  | Optional<int> opt; | 
|  | EXPECT_TRUE(1 >= opt); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, GreaterEqValue_NotEmpty) { | 
|  | { | 
|  | Optional<int> opt(0); | 
|  | EXPECT_TRUE(1 >= opt); | 
|  | } | 
|  | { | 
|  | Optional<int> opt(1); | 
|  | EXPECT_TRUE(1 >= opt); | 
|  | } | 
|  | { | 
|  | Optional<int> opt(2); | 
|  | EXPECT_FALSE(1 >= opt); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, GreaterEqValue_DifferentType) { | 
|  | Optional<int> opt(0); | 
|  | EXPECT_TRUE(0.0 >= opt); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, NotEquals) { | 
|  | { | 
|  | Optional<float> a(0.1f); | 
|  | Optional<float> b(0.2f); | 
|  | EXPECT_NE(a, b); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<std::string> a("foo"); | 
|  | Optional<std::string> b("bar"); | 
|  | EXPECT_NE(a, b); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<int> a(1); | 
|  | Optional<double> b(2); | 
|  | EXPECT_NE(a, b); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<TestObject> a(TestObject(3, 0.1)); | 
|  | Optional<TestObject> b(TestObject(4, 1.0)); | 
|  | EXPECT_TRUE(a != b); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, NotEqualsNull) { | 
|  | { | 
|  | Optional<float> a(0.1f); | 
|  | Optional<float> b(0.1f); | 
|  | b = base::nullopt; | 
|  | EXPECT_NE(a, b); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<std::string> a("foo"); | 
|  | Optional<std::string> b("foo"); | 
|  | b = base::nullopt; | 
|  | EXPECT_NE(a, b); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<TestObject> a(TestObject(3, 0.1)); | 
|  | Optional<TestObject> b(TestObject(3, 0.1)); | 
|  | b = base::nullopt; | 
|  | EXPECT_TRUE(a != b); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, MakeOptional) { | 
|  | { | 
|  | // Use qualified base::make_optional here and elsewhere to avoid the name | 
|  | // confliction to std::make_optional. | 
|  | // The name conflict happens only for types in std namespace, such as | 
|  | // std::string. The other qualified base::make_optional usages are just for | 
|  | // consistency. | 
|  | Optional<float> o = base::make_optional(32.f); | 
|  | EXPECT_TRUE(o); | 
|  | EXPECT_EQ(32.f, *o); | 
|  |  | 
|  | float value = 3.f; | 
|  | o = base::make_optional(std::move(value)); | 
|  | EXPECT_TRUE(o); | 
|  | EXPECT_EQ(3.f, *o); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<std::string> o = base::make_optional(std::string("foo")); | 
|  | EXPECT_TRUE(o); | 
|  | EXPECT_EQ("foo", *o); | 
|  |  | 
|  | std::string value = "bar"; | 
|  | o = base::make_optional(std::move(value)); | 
|  | EXPECT_TRUE(o); | 
|  | EXPECT_EQ(std::string("bar"), *o); | 
|  | } | 
|  |  | 
|  | { | 
|  | Optional<TestObject> o = base::make_optional(TestObject(3, 0.1)); | 
|  | EXPECT_TRUE(!!o); | 
|  | EXPECT_TRUE(TestObject(3, 0.1) == *o); | 
|  |  | 
|  | TestObject value = TestObject(0, 0.42); | 
|  | o = base::make_optional(std::move(value)); | 
|  | EXPECT_TRUE(!!o); | 
|  | EXPECT_TRUE(TestObject(0, 0.42) == *o); | 
|  | EXPECT_EQ(TestObject::State::MOVED_FROM, value.state()); | 
|  | EXPECT_EQ(TestObject::State::MOVE_ASSIGNED, o->state()); | 
|  |  | 
|  | EXPECT_EQ(TestObject::State::MOVE_CONSTRUCTED, | 
|  | base::make_optional(std::move(value))->state()); | 
|  | } | 
|  |  | 
|  | { | 
|  | struct Test { | 
|  | Test(int x, double y, bool z) : a(x), b(y), c(z) {} | 
|  |  | 
|  | int a; | 
|  | double b; | 
|  | bool c; | 
|  | }; | 
|  |  | 
|  | Optional<Test> o = base::make_optional<Test>(1, 2.0, true); | 
|  | EXPECT_TRUE(!!o); | 
|  | EXPECT_EQ(1, o->a); | 
|  | EXPECT_EQ(2.0, o->b); | 
|  | EXPECT_TRUE(o->c); | 
|  | } | 
|  |  | 
|  | { | 
|  | auto str1 = base::make_optional<std::string>({'1', '2', '3'}); | 
|  | EXPECT_EQ("123", *str1); | 
|  |  | 
|  | auto str2 = base::make_optional<std::string>({'a', 'b', 'c'}, | 
|  | std::allocator<char>()); | 
|  | EXPECT_EQ("abc", *str2); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, NonMemberSwap_bothNoValue) { | 
|  | Optional<TestObject> a, b; | 
|  | base::swap(a, b); | 
|  |  | 
|  | EXPECT_FALSE(!!a); | 
|  | EXPECT_FALSE(!!b); | 
|  | EXPECT_TRUE(TestObject(42, 0.42) == a.value_or(TestObject(42, 0.42))); | 
|  | EXPECT_TRUE(TestObject(42, 0.42) == b.value_or(TestObject(42, 0.42))); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, NonMemberSwap_inHasValue) { | 
|  | Optional<TestObject> a(TestObject(1, 0.3)); | 
|  | Optional<TestObject> b; | 
|  | base::swap(a, b); | 
|  |  | 
|  | EXPECT_FALSE(!!a); | 
|  | EXPECT_TRUE(!!b); | 
|  | EXPECT_TRUE(TestObject(42, 0.42) == a.value_or(TestObject(42, 0.42))); | 
|  | EXPECT_TRUE(TestObject(1, 0.3) == b.value_or(TestObject(42, 0.42))); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, NonMemberSwap_outHasValue) { | 
|  | Optional<TestObject> a; | 
|  | Optional<TestObject> b(TestObject(1, 0.3)); | 
|  | base::swap(a, b); | 
|  |  | 
|  | EXPECT_TRUE(!!a); | 
|  | EXPECT_FALSE(!!b); | 
|  | EXPECT_TRUE(TestObject(1, 0.3) == a.value_or(TestObject(42, 0.42))); | 
|  | EXPECT_TRUE(TestObject(42, 0.42) == b.value_or(TestObject(42, 0.42))); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, NonMemberSwap_bothValue) { | 
|  | Optional<TestObject> a(TestObject(0, 0.1)); | 
|  | Optional<TestObject> b(TestObject(1, 0.3)); | 
|  | base::swap(a, b); | 
|  |  | 
|  | EXPECT_TRUE(!!a); | 
|  | EXPECT_TRUE(!!b); | 
|  | EXPECT_TRUE(TestObject(1, 0.3) == a.value_or(TestObject(42, 0.42))); | 
|  | EXPECT_TRUE(TestObject(0, 0.1) == b.value_or(TestObject(42, 0.42))); | 
|  | EXPECT_EQ(TestObject::State::SWAPPED, a->state()); | 
|  | EXPECT_EQ(TestObject::State::SWAPPED, b->state()); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, Hash_OptionalReflectsInternal) { | 
|  | { | 
|  | std::hash<int> int_hash; | 
|  | std::hash<Optional<int>> opt_int_hash; | 
|  |  | 
|  | EXPECT_EQ(int_hash(1), opt_int_hash(Optional<int>(1))); | 
|  | } | 
|  |  | 
|  | { | 
|  | std::hash<std::string> str_hash; | 
|  | std::hash<Optional<std::string>> opt_str_hash; | 
|  |  | 
|  | EXPECT_EQ(str_hash(std::string("foobar")), | 
|  | opt_str_hash(Optional<std::string>(std::string("foobar")))); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, Hash_NullOptEqualsNullOpt) { | 
|  | std::hash<Optional<int>> opt_int_hash; | 
|  | std::hash<Optional<std::string>> opt_str_hash; | 
|  |  | 
|  | EXPECT_EQ(opt_str_hash(Optional<std::string>()), | 
|  | opt_int_hash(Optional<int>())); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, Hash_UseInSet) { | 
|  | std::set<Optional<int>> setOptInt; | 
|  |  | 
|  | EXPECT_EQ(setOptInt.end(), setOptInt.find(42)); | 
|  |  | 
|  | setOptInt.insert(Optional<int>(3)); | 
|  | EXPECT_EQ(setOptInt.end(), setOptInt.find(42)); | 
|  | EXPECT_NE(setOptInt.end(), setOptInt.find(3)); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, HasValue) { | 
|  | Optional<int> a; | 
|  | EXPECT_FALSE(a.has_value()); | 
|  |  | 
|  | a = 42; | 
|  | EXPECT_TRUE(a.has_value()); | 
|  |  | 
|  | a = nullopt; | 
|  | EXPECT_FALSE(a.has_value()); | 
|  |  | 
|  | a = 0; | 
|  | EXPECT_TRUE(a.has_value()); | 
|  |  | 
|  | a = Optional<int>(); | 
|  | EXPECT_FALSE(a.has_value()); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, Reset_int) { | 
|  | Optional<int> a(0); | 
|  | EXPECT_TRUE(a.has_value()); | 
|  | EXPECT_EQ(0, a.value()); | 
|  |  | 
|  | a.reset(); | 
|  | EXPECT_FALSE(a.has_value()); | 
|  | EXPECT_EQ(-1, a.value_or(-1)); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, Reset_Object) { | 
|  | Optional<TestObject> a(TestObject(0, 0.1)); | 
|  | EXPECT_TRUE(a.has_value()); | 
|  | EXPECT_EQ(TestObject(0, 0.1), a.value()); | 
|  |  | 
|  | a.reset(); | 
|  | EXPECT_FALSE(a.has_value()); | 
|  | EXPECT_EQ(TestObject(42, 0.0), a.value_or(TestObject(42, 0.0))); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, Reset_NoOp) { | 
|  | Optional<int> a; | 
|  | EXPECT_FALSE(a.has_value()); | 
|  |  | 
|  | a.reset(); | 
|  | EXPECT_FALSE(a.has_value()); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, AssignFromRValue) { | 
|  | Optional<TestObject> a; | 
|  | EXPECT_FALSE(a.has_value()); | 
|  |  | 
|  | TestObject obj; | 
|  | a = std::move(obj); | 
|  | EXPECT_TRUE(a.has_value()); | 
|  | EXPECT_EQ(1, a->move_ctors_count()); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, DontCallDefaultCtor) { | 
|  | Optional<DeletedDefaultConstructor> a; | 
|  | EXPECT_FALSE(a.has_value()); | 
|  |  | 
|  | a = base::make_optional<DeletedDefaultConstructor>(42); | 
|  | EXPECT_TRUE(a.has_value()); | 
|  | EXPECT_EQ(42, a->foo()); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, DontCallNewMemberFunction) { | 
|  | Optional<DeleteNewOperators> a; | 
|  | EXPECT_FALSE(a.has_value()); | 
|  |  | 
|  | a = DeleteNewOperators(); | 
|  | EXPECT_TRUE(a.has_value()); | 
|  | } | 
|  |  | 
|  | TEST(OptionalTest, Noexcept) { | 
|  | // Trivial copy ctor, non-trivial move ctor, nothrow move assign. | 
|  | struct Test1 { | 
|  | Test1(const Test1&) = default; | 
|  | Test1(Test1&&) {} | 
|  | Test1& operator=(Test1&&) = default; | 
|  | }; | 
|  | // Non-trivial copy ctor, trivial move ctor, throw move assign. | 
|  | struct Test2 { | 
|  | Test2(const Test2&) {} | 
|  | Test2(Test2&&) = default; | 
|  | Test2& operator=(Test2&&) { return *this; } | 
|  | }; | 
|  | // Trivial copy ctor, non-trivial nothrow move ctor. | 
|  | struct Test3 { | 
|  | Test3(const Test3&) = default; | 
|  | Test3(Test3&&) noexcept {} | 
|  | }; | 
|  | // Non-trivial copy ctor, non-trivial nothrow move ctor. | 
|  | struct Test4 { | 
|  | Test4(const Test4&) {} | 
|  | Test4(Test4&&) noexcept {} | 
|  | }; | 
|  | // Non-trivial copy ctor, non-trivial move ctor. | 
|  | struct Test5 { | 
|  | Test5(const Test5&) {} | 
|  | Test5(Test5&&) {} | 
|  | }; | 
|  |  | 
|  | static_assert( | 
|  | noexcept(Optional<int>(std::declval<Optional<int>>())), | 
|  | "move constructor for noexcept move-constructible T must be noexcept " | 
|  | "(trivial copy, trivial move)"); | 
|  | static_assert( | 
|  | !noexcept(Optional<Test1>(std::declval<Optional<Test1>>())), | 
|  | "move constructor for non-noexcept move-constructible T must not be " | 
|  | "noexcept (trivial copy)"); | 
|  | static_assert( | 
|  | noexcept(Optional<Test2>(std::declval<Optional<Test2>>())), | 
|  | "move constructor for noexcept move-constructible T must be noexcept " | 
|  | "(non-trivial copy, trivial move)"); | 
|  | static_assert( | 
|  | noexcept(Optional<Test3>(std::declval<Optional<Test3>>())), | 
|  | "move constructor for noexcept move-constructible T must be noexcept " | 
|  | "(trivial copy, non-trivial move)"); | 
|  | static_assert( | 
|  | noexcept(Optional<Test4>(std::declval<Optional<Test4>>())), | 
|  | "move constructor for noexcept move-constructible T must be noexcept " | 
|  | "(non-trivial copy, non-trivial move)"); | 
|  | static_assert( | 
|  | !noexcept(Optional<Test5>(std::declval<Optional<Test5>>())), | 
|  | "move constructor for non-noexcept move-constructible T must not be " | 
|  | "noexcept (non-trivial copy)"); | 
|  |  | 
|  | static_assert( | 
|  | noexcept(std::declval<Optional<int>>() = std::declval<Optional<int>>()), | 
|  | "move assign for noexcept move-constructible/move-assignable T " | 
|  | "must be noexcept"); | 
|  | static_assert( | 
|  | !noexcept(std::declval<Optional<Test1>>() = | 
|  | std::declval<Optional<Test1>>()), | 
|  | "move assign for non-noexcept move-constructible T must not be noexcept"); | 
|  | static_assert( | 
|  | !noexcept(std::declval<Optional<Test2>>() = | 
|  | std::declval<Optional<Test2>>()), | 
|  | "move assign for non-noexcept move-assignable T must not be noexcept"); | 
|  | } | 
|  |  | 
|  | }  // namespace base | 
|  | }  // namespace perfetto | 
|  |  | 
|  | #if defined(__GNUC__) || defined(__clang__) | 
|  | #pragma GCC diagnostic pop | 
|  | #endif |