2017-08-12 16:48:01 +00:00
|
|
|
/*
|
|
|
|
* Copyright (C) 1999-2001 Harri Porten (porten@kde.org)
|
|
|
|
* Copyright (C) 2001 Peter Kelly (pmk@post.com)
|
2022-10-23 02:55:20 +00:00
|
|
|
* Copyright (C) 2003-2019 Apple Inc. All rights reserved.
|
2017-08-12 16:48:01 +00:00
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Library General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Library General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Library General Public License
|
|
|
|
* along with this library; see the file COPYING.LIB. If not, write to
|
|
|
|
* the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
|
|
|
|
* Boston, MA 02110-1301, USA.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
2020-08-29 13:27:11 +00:00
|
|
|
#include "JSCast.h"
|
2017-08-12 16:48:01 +00:00
|
|
|
|
|
|
|
#include "CallFrame.h"
|
|
|
|
#include "JSGlobalObject.h"
|
|
|
|
#include "NullGetterFunction.h"
|
|
|
|
#include "NullSetterFunction.h"
|
|
|
|
#include "Structure.h"
|
|
|
|
|
|
|
|
namespace JSC {
|
|
|
|
|
|
|
|
class JSObject;
|
|
|
|
|
|
|
|
// This is an internal value object which stores getter and setter functions
|
|
|
|
// for a property. Instances of this class have the property that once a getter
|
|
|
|
// or setter is set to a non-null value, then they cannot be changed. This means
|
|
|
|
// that if a property holding a GetterSetter reference is constant-inferred and
|
|
|
|
// that constant is observed to have a non-null setter (or getter) then we can
|
|
|
|
// constant fold that setter (or getter).
|
2022-10-23 02:55:20 +00:00
|
|
|
class GetterSetter final : public JSCell {
|
2017-08-12 16:48:01 +00:00
|
|
|
friend class JIT;
|
2022-10-23 02:55:20 +00:00
|
|
|
using Base = JSCell;
|
2017-08-12 16:48:01 +00:00
|
|
|
private:
|
2020-08-29 13:27:11 +00:00
|
|
|
GetterSetter(VM& vm, JSGlobalObject* globalObject, JSObject* getter, JSObject* setter)
|
2022-10-23 02:55:20 +00:00
|
|
|
: Base(vm, vm.getterSetterStructure.get())
|
2017-08-12 16:48:01 +00:00
|
|
|
{
|
2020-08-29 13:27:11 +00:00
|
|
|
WTF::storeStoreFence();
|
|
|
|
m_getter.set(vm, this, getter ? getter : globalObject->nullGetterFunction());
|
|
|
|
m_setter.set(vm, this, setter ? setter : globalObject->nullSetterFunction());
|
2017-08-12 16:48:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
2022-10-23 02:55:20 +00:00
|
|
|
static constexpr unsigned StructureFlags = Base::StructureFlags | OverridesGetOwnPropertySlot | StructureIsImmortal;
|
|
|
|
|
|
|
|
template<typename CellType, SubspaceAccess>
|
|
|
|
static IsoSubspace* subspaceFor(VM& vm)
|
|
|
|
{
|
|
|
|
return &vm.getterSetterSpace;
|
|
|
|
}
|
2017-08-12 16:48:01 +00:00
|
|
|
|
2020-08-29 13:27:11 +00:00
|
|
|
static GetterSetter* create(VM& vm, JSGlobalObject* globalObject, JSObject* getter, JSObject* setter)
|
2017-08-12 16:48:01 +00:00
|
|
|
{
|
2020-08-29 13:27:11 +00:00
|
|
|
GetterSetter* getterSetter = new (NotNull, allocateCell<GetterSetter>(vm.heap)) GetterSetter(vm, globalObject, getter, setter);
|
2017-08-12 16:48:01 +00:00
|
|
|
getterSetter->finishCreation(vm);
|
|
|
|
return getterSetter;
|
|
|
|
}
|
|
|
|
|
2020-08-29 13:27:11 +00:00
|
|
|
static GetterSetter* create(VM& vm, JSGlobalObject* globalObject, JSValue getter, JSValue setter)
|
|
|
|
{
|
|
|
|
ASSERT(getter.isUndefined() || getter.isObject());
|
|
|
|
ASSERT(setter.isUndefined() || setter.isObject());
|
|
|
|
JSObject* getterObject { nullptr };
|
|
|
|
JSObject* setterObject { nullptr };
|
|
|
|
if (getter.isObject())
|
|
|
|
getterObject = asObject(getter);
|
|
|
|
if (setter.isObject())
|
|
|
|
setterObject = asObject(setter);
|
|
|
|
return create(vm, globalObject, getterObject, setterObject);
|
|
|
|
}
|
|
|
|
|
2017-08-12 16:48:01 +00:00
|
|
|
static void visitChildren(JSCell*, SlotVisitor&);
|
|
|
|
|
|
|
|
JSObject* getter() const { return m_getter.get(); }
|
|
|
|
|
|
|
|
JSObject* getterConcurrently() const
|
|
|
|
{
|
|
|
|
JSObject* result = getter();
|
|
|
|
WTF::loadLoadFence();
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2022-10-23 02:55:20 +00:00
|
|
|
bool isGetterNull() const { return !!jsDynamicCast<NullGetterFunction*>(m_getter.get()->vm(), m_getter.get()); }
|
|
|
|
bool isSetterNull() const { return !!jsDynamicCast<NullSetterFunction*>(m_setter.get()->vm(), m_setter.get()); }
|
2017-08-12 16:48:01 +00:00
|
|
|
|
|
|
|
JSObject* setter() const { return m_setter.get(); }
|
|
|
|
|
|
|
|
JSObject* setterConcurrently() const
|
|
|
|
{
|
|
|
|
JSObject* result = setter();
|
|
|
|
WTF::loadLoadFence();
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
static Structure* createStructure(VM& vm, JSGlobalObject* globalObject, JSValue prototype)
|
|
|
|
{
|
2022-10-23 02:55:20 +00:00
|
|
|
return Structure::create(vm, globalObject, prototype, TypeInfo(GetterSetterType, StructureFlags), info());
|
2017-08-12 16:48:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static ptrdiff_t offsetOfGetter()
|
|
|
|
{
|
|
|
|
return OBJECT_OFFSETOF(GetterSetter, m_getter);
|
|
|
|
}
|
|
|
|
|
|
|
|
static ptrdiff_t offsetOfSetter()
|
|
|
|
{
|
|
|
|
return OBJECT_OFFSETOF(GetterSetter, m_setter);
|
|
|
|
}
|
|
|
|
|
|
|
|
DECLARE_EXPORT_INFO;
|
|
|
|
|
2022-10-23 02:55:20 +00:00
|
|
|
static bool getOwnPropertySlot(JSObject*, JSGlobalObject*, PropertyName, PropertySlot&) { RELEASE_ASSERT_NOT_REACHED(); return false; }
|
|
|
|
static bool put(JSCell*, JSGlobalObject*, PropertyName, JSValue, PutPropertySlot&) { RELEASE_ASSERT_NOT_REACHED(); return false; }
|
|
|
|
static bool putByIndex(JSCell*, JSGlobalObject*, unsigned, JSValue, bool) { RELEASE_ASSERT_NOT_REACHED(); return false; }
|
|
|
|
static bool setPrototype(JSObject*, JSGlobalObject*, JSValue, bool) { RELEASE_ASSERT_NOT_REACHED(); return false; }
|
|
|
|
static bool defineOwnProperty(JSObject*, JSGlobalObject*, PropertyName, const PropertyDescriptor&, bool) { RELEASE_ASSERT_NOT_REACHED(); return false; }
|
|
|
|
static bool deleteProperty(JSCell*, JSGlobalObject*, PropertyName, DeletePropertySlot&) { RELEASE_ASSERT_NOT_REACHED(); return false; }
|
2017-08-12 16:48:01 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
WriteBarrier<JSObject> m_getter;
|
|
|
|
WriteBarrier<JSObject> m_setter;
|
|
|
|
};
|
|
|
|
|
2022-10-23 02:55:20 +00:00
|
|
|
JSValue callGetter(JSGlobalObject*, JSValue base, JSValue getterSetter);
|
|
|
|
JS_EXPORT_PRIVATE bool callSetter(JSGlobalObject*, JSValue base, JSValue getterSetter, JSValue, ECMAMode);
|
2017-08-12 16:48:01 +00:00
|
|
|
|
|
|
|
} // namespace JSC
|