// Copyright 2020 the V8 project authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #ifndef INCLUDE_CPPGC_INTERNAL_FINALIZER_TRAIT_H_ #define INCLUDE_CPPGC_INTERNAL_FINALIZER_TRAIT_H_ #include #include "cppgc/type-traits.h" namespace cppgc { namespace internal { using FinalizationCallback = void (*)(void*); template struct HasFinalizeGarbageCollectedObject : std::false_type {}; template struct HasFinalizeGarbageCollectedObject< T, void_t().FinalizeGarbageCollectedObject())>> : std::true_type {}; // The FinalizerTraitImpl specifies how to finalize objects. template struct FinalizerTraitImpl; template struct FinalizerTraitImpl { private: // Dispatch to custom FinalizeGarbageCollectedObject(). struct Custom { static void Call(void* obj) { static_cast(obj)->FinalizeGarbageCollectedObject(); } }; // Dispatch to regular destructor. struct Destructor { static void Call(void* obj) { static_cast(obj)->~T(); } }; using FinalizeImpl = std::conditional_t::value, Custom, Destructor>; public: static void Finalize(void* obj) { static_assert(sizeof(T), "T must be fully defined"); FinalizeImpl::Call(obj); } }; template struct FinalizerTraitImpl { static void Finalize(void* obj) { static_assert(sizeof(T), "T must be fully defined"); } }; // The FinalizerTrait is used to determine if a type requires finalization and // what finalization means. template struct FinalizerTrait { private: // Object has a finalizer if it has // - a custom FinalizeGarbageCollectedObject method, or // - a destructor. static constexpr bool kNonTrivialFinalizer = internal::HasFinalizeGarbageCollectedObject::value || !std::is_trivially_destructible::type>::value; static void Finalize(void* obj) { internal::FinalizerTraitImpl::Finalize(obj); } public: // The callback used to finalize an object of type T. static constexpr FinalizationCallback kCallback = kNonTrivialFinalizer ? Finalize : nullptr; }; template constexpr FinalizationCallback FinalizerTrait::kCallback; } // namespace internal } // namespace cppgc #endif // INCLUDE_CPPGC_INTERNAL_FINALIZER_TRAIT_H_