// 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 #include #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