mirror of
https://github.com/darlinghq/darling-JavaScriptCore.git
synced 2025-04-10 02:40:54 +00:00
116 lines
4.2 KiB
C++
116 lines
4.2 KiB
C++
/*
|
|
* Copyright (C) 2020 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. ``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
|
|
* 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 "JSInternalFieldObjectImpl.h"
|
|
|
|
namespace JSC {
|
|
|
|
class JSFinalizationRegistry final : public JSInternalFieldObjectImpl<1> {
|
|
public:
|
|
using Base = JSInternalFieldObjectImpl<1>;
|
|
|
|
enum class Field : uint8_t {
|
|
Callback,
|
|
};
|
|
|
|
static size_t allocationSize(Checked<size_t> inlineCapacity)
|
|
{
|
|
ASSERT_UNUSED(inlineCapacity, inlineCapacity == 0U);
|
|
return sizeof(JSFinalizationRegistry);
|
|
}
|
|
|
|
template<typename CellType, SubspaceAccess mode>
|
|
static IsoSubspace* subspaceFor(VM& vm)
|
|
{
|
|
return vm.finalizationRegistrySpace<mode>();
|
|
}
|
|
|
|
static std::array<JSValue, numberOfInternalFields> initialValues()
|
|
{
|
|
return { {
|
|
jsNull(),
|
|
} };
|
|
}
|
|
|
|
const WriteBarrier<Unknown>& internalField(Field field) const { return Base::internalField(static_cast<uint32_t>(field)); }
|
|
WriteBarrier<Unknown>& internalField(Field field) { return Base::internalField(static_cast<uint32_t>(field)); }
|
|
|
|
JSObject* callback() const { return jsCast<JSObject*>(internalField(Field::Callback).get()); }
|
|
|
|
static JSFinalizationRegistry* create(VM&, Structure*, JSObject* callback);
|
|
static JSFinalizationRegistry* createWithInitialValues(VM&, Structure*);
|
|
static Structure* createStructure(VM&, JSGlobalObject*, JSValue prototype);
|
|
|
|
void runFinalizationCleanup(JSGlobalObject*);
|
|
|
|
DECLARE_EXPORT_INFO;
|
|
|
|
void finalizeUnconditionally(VM&);
|
|
static void visitChildren(JSCell*, SlotVisitor&);
|
|
static void destroy(JSCell*);
|
|
static constexpr bool needsDestruction = true;
|
|
|
|
JSValue takeDeadHoldingsValue();
|
|
|
|
bool unregister(VM&, JSObject* token);
|
|
// token should be a JSObject or undefined.
|
|
void registerTarget(VM&, JSObject* target, JSValue holdings, JSValue token);
|
|
|
|
JS_EXPORT_PRIVATE size_t liveCount(const Locker<JSCellLock>&);
|
|
JS_EXPORT_PRIVATE size_t deadCount(const Locker<JSCellLock>&);
|
|
|
|
private:
|
|
JSFinalizationRegistry(VM& vm, Structure* structure)
|
|
: Base(vm, structure)
|
|
{
|
|
}
|
|
|
|
JS_EXPORT_PRIVATE void finishCreation(VM&, JSGlobalObject*, JSObject* callback);
|
|
|
|
static String toStringName(const JSObject*, JSGlobalObject*);
|
|
|
|
struct Registration {
|
|
JSObject* target;
|
|
WriteBarrier<Unknown> holdings;
|
|
};
|
|
|
|
using LiveRegistrations = Vector<Registration>;
|
|
// We don't need the target anymore since we know it's dead.
|
|
using DeadRegistrations = Vector<WriteBarrier<Unknown>>;
|
|
|
|
// Note that we don't bother putting a write barrier on the key or target because they are weakly referenced.
|
|
HashMap<JSObject*, LiveRegistrations> m_liveRegistrations;
|
|
HashMap<JSObject*, DeadRegistrations> m_deadRegistrations;
|
|
// We use a separate list for no unregister values instead of a special key in the tables above because the HashMap has a tendency to reallocate under us when iterating...
|
|
LiveRegistrations m_noUnregistrationLive;
|
|
DeadRegistrations m_noUnregistrationDead;
|
|
};
|
|
|
|
} // namespace JSC
|
|
|
|
|