| // Copyright 2017 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/synchronization/mutex.h" |
| |
| #include <cstdlib> |
| #include <string> |
| |
| #include "gtest/gtest.h" |
| #include "absl/base/config.h" |
| |
| namespace { |
| |
| class IncompleteClass; |
| |
| #ifdef _MSC_VER |
| // These tests verify expectations about sizes of MSVC pointers to methods. |
| // Pointers to methods are distinguished by whether their class hierachies |
| // contain single inheritance, multiple inheritance, or virtual inheritence. |
| |
| // Declare classes of the various MSVC inheritance types. |
| class __single_inheritance SingleInheritance{}; |
| class __multiple_inheritance MultipleInheritance; |
| class __virtual_inheritance VirtualInheritance; |
| |
| TEST(MutexMethodPointerTest, MicrosoftMethodPointerSize) { |
| void (SingleInheritance::*single_inheritance_method_pointer)(); |
| void (MultipleInheritance::*multiple_inheritance_method_pointer)(); |
| void (VirtualInheritance::*virtual_inheritance_method_pointer)(); |
| |
| #if defined(_M_IX86) || defined(_M_ARM) |
| static_assert(sizeof(single_inheritance_method_pointer) == 4, |
| "Unexpected sizeof(single_inheritance_method_pointer)."); |
| static_assert(sizeof(multiple_inheritance_method_pointer) == 8, |
| "Unexpected sizeof(multiple_inheritance_method_pointer)."); |
| static_assert(sizeof(virtual_inheritance_method_pointer) == 12, |
| "Unexpected sizeof(virtual_inheritance_method_pointer)."); |
| #elif defined(_M_X64) || defined(__aarch64__) |
| static_assert(sizeof(single_inheritance_method_pointer) == 8, |
| "Unexpected sizeof(single_inheritance_method_pointer)."); |
| static_assert(sizeof(multiple_inheritance_method_pointer) == 16, |
| "Unexpected sizeof(multiple_inheritance_method_pointer)."); |
| static_assert(sizeof(virtual_inheritance_method_pointer) == 16, |
| "Unexpected sizeof(virtual_inheritance_method_pointer)."); |
| #endif |
| void (IncompleteClass::*incomplete_class_method_pointer)(); |
| static_assert(sizeof(incomplete_class_method_pointer) >= |
| sizeof(virtual_inheritance_method_pointer), |
| "Failed invariant: sizeof(incomplete_class_method_pointer) >= " |
| "sizeof(virtual_inheritance_method_pointer)!"); |
| } |
| |
| class Callback { |
| bool x = true; |
| |
| public: |
| Callback() {} |
| bool method() { |
| x = !x; |
| return x; |
| } |
| }; |
| |
| class M2 { |
| bool x = true; |
| |
| public: |
| M2() {} |
| bool method2() { |
| x = !x; |
| return x; |
| } |
| }; |
| |
| class MultipleInheritance : public Callback, public M2 {}; |
| |
| TEST(MutexMethodPointerTest, ConditionWithMultipleInheritanceMethod) { |
| // This test ensures that Condition can deal with method pointers from classes |
| // with multiple inheritance. |
| MultipleInheritance object = MultipleInheritance(); |
| absl::Condition condition(&object, &MultipleInheritance::method); |
| EXPECT_FALSE(condition.Eval()); |
| EXPECT_TRUE(condition.Eval()); |
| } |
| |
| class __virtual_inheritance VirtualInheritance : virtual public Callback { |
| bool x = false; |
| |
| public: |
| VirtualInheritance() {} |
| bool method() { |
| x = !x; |
| return x; |
| } |
| }; |
| |
| TEST(MutexMethodPointerTest, ConditionWithVirtualInheritanceMethod) { |
| // This test ensures that Condition can deal with method pointers from classes |
| // with virtual inheritance. |
| VirtualInheritance object = VirtualInheritance(); |
| absl::Condition condition(&object, &VirtualInheritance::method); |
| EXPECT_TRUE(condition.Eval()); |
| EXPECT_FALSE(condition.Eval()); |
| } |
| #endif // #ifdef _MSC_VER |
| |
| TEST(MutexMethodPointerTest, ConditionWithIncompleteClassMethod) { |
| using IncompleteClassMethodPointer = void (IncompleteClass::*)(); |
| |
| union CallbackSlot { |
| void (*anonymous_function_pointer)(); |
| IncompleteClassMethodPointer incomplete_class_method_pointer; |
| }; |
| |
| static_assert(sizeof(CallbackSlot) >= sizeof(IncompleteClassMethodPointer), |
| "The callback slot is not big enough for method pointers."); |
| static_assert( |
| sizeof(CallbackSlot) == sizeof(IncompleteClassMethodPointer), |
| "The callback slot is not big enough for anonymous function pointers."); |
| |
| #if defined(_MSC_VER) |
| static_assert(sizeof(IncompleteClassMethodPointer) <= 24, |
| "The pointer to a method of an incomplete class is too big."); |
| #endif |
| } |
| |
| } // namespace |