/* * Copyright (C) 2011-2024 Apple Inc. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS'' * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF * THE POSSIBILITY OF SUCH DAMAGE. */ #pragma once #include WTF_ALLOW_UNSAFE_BUFFER_USAGE_BEGIN #include #include #include #include #include #include #include namespace JSC { class VM; REFTRACKER_DECL(StrongRefTracker); // A strongly referenced handle that prevents the object it points to from being garbage collected. template class Strong final : public Handle { using Handle::slot; using Handle::setSlot; template friend class Strong; public: using ExternalType = typename Handle::ExternalType; using ValueType = std::remove_pointer_t; using PtrType = ExternalType; Strong() : Handle() { } inline Strong(VM&, ExternalType = ExternalType()); inline Strong(VM&, Handle); Strong(const Strong& other) : Handle() { if (!other.slot()) return; setSlot(HandleSet::heapFor(other.slot())->allocate()); set(other.get()); } template Strong(const Strong& other) : Handle() { if (!other.slot()) return; setSlot(HandleSet::heapFor(other.slot())->allocate()); set(other.get()); } enum HashTableDeletedValueTag { HashTableDeletedValue }; bool isHashTableDeletedValue() const { return slot() == hashTableDeletedValue(); } Strong(HashTableDeletedValueTag) : Handle(hashTableDeletedValue()) { } enum HashTableEmptyValueTag { HashTableEmptyValue }; bool isHashTableEmptyValue() const { return slot() == hashTableEmptyValue(); } Strong(HashTableEmptyValueTag) : Handle(hashTableEmptyValue()) { } ~Strong() override { clear(); } bool operator!() const { return !slot() || !*slot(); } explicit operator bool() const { return !!*this; } void swap(Strong& other) { Handle::swap(other); } ExternalType get() const { return HandleTypes::getFromSlot(this->slot()); } inline void set(VM&, ExternalType); // Defined in StrongInlines.h template inline Strong& operator=(const Strong& other); // Defined in StrongInlines.h Strong& operator=(const Strong& other); // Defined in StrongInlines.h void clear() { if (!slot()) return; auto* heap = HandleSet::heapFor(slot()); if (shouldStrongDestructorGrabLock == ShouldStrongDestructorGrabLock::Yes) { JSLockHolder holder(heap->vm()); heap->deallocate(slot()); setSlot(nullptr); } else { heap->deallocate(slot()); setSlot(nullptr); } } private: static HandleSlot hashTableDeletedValue() { return reinterpret_cast(-1); } static HandleSlot hashTableEmptyValue() { return reinterpret_cast(0); } void set(ExternalType externalType) { ASSERT(slot()); JSValue value = HandleTypes::toJSValue(externalType); HandleSet::heapFor(slot())->template writeBarrier>(slot(), value); *slot() = value; } REFTRACKER_MEMBERS(StrongRefTracker); }; template inline void swap(Strong& a, Strong& b) { a.swap(b); } } // namespace JSC namespace WTF { template struct VectorTraits> : SimpleClassVectorTraits { static constexpr bool canCompareWithMemcmp = false; #if ENABLE(REFTRACKER) static constexpr bool canInitializeWithMemset = false; static constexpr bool canMoveWithMemcpy = false; #endif }; template struct IsSmartPtr> { static constexpr bool value = true; static constexpr bool isNullable = true; }; template struct DefaultHash; template struct DefaultHash> : PtrHash> { }; template struct HashTraits> : SimpleClassHashTraits> { #if ENABLE(REFTRACKER) using S = JSC::Strong

; static constexpr bool emptyValueIsZero = false; static S emptyValue() { return S::HashTableEmptyValue; } template static void constructEmptyValue(S& slot) { new (NotNull, std::addressof(slot)) S(S::HashTableEmptyValue); } static constexpr bool hasIsEmptyValueFunction = true; static bool isEmptyValue(const S& value) { return value.isHashTableEmptyValue(); } static void constructDeletedValue(S& slot) { new (NotNull, &slot) S(S::HashTableDeletedValue); } static bool isDeletedValue(const S& value) { return value.isHashTableDeletedValue(); } #endif }; } // namespace WTF WTF_ALLOW_UNSAFE_BUFFER_USAGE_END