Get rid of HeapRootVisitor and make SlotVisitor less painful to use
[WebKit-https.git] / Source / JavaScriptCore / runtime / GetterSetter.cpp
1 /*
2  *  Copyright (C) 1999-2002 Harri Porten (porten@kde.org)
3  *  Copyright (C) 2001 Peter Kelly (pmk@post.com)
4  *  Copyright (C) 2004, 2007-2009, 2014, 2016 Apple Inc. All rights reserved.
5  *
6  *  This library is free software; you can redistribute it and/or
7  *  modify it under the terms of the GNU Library General Public
8  *  License as published by the Free Software Foundation; either
9  *  version 2 of the License, or (at your option) any later version.
10  *
11  *  This library is distributed in the hope that it will be useful,
12  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
13  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14  *  Library General Public License for more details.
15  *
16  *  You should have received a copy of the GNU Library General Public License
17  *  along with this library; see the file COPYING.LIB.  If not, write to
18  *  the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
19  *  Boston, MA 02110-1301, USA.
20  *
21  */
22
23 #include "config.h"
24 #include "GetterSetter.h"
25
26 #include "Error.h"
27 #include "Exception.h"
28 #include "JSObject.h"
29 #include "JSCInlines.h"
30 #include <wtf/Assertions.h>
31
32 namespace JSC {
33
34 STATIC_ASSERT_IS_TRIVIALLY_DESTRUCTIBLE(GetterSetter);
35
36 const ClassInfo GetterSetter::s_info = { "GetterSetter", &Base::s_info, 0, CREATE_METHOD_TABLE(GetterSetter) };
37
38 void GetterSetter::visitChildren(JSCell* cell, SlotVisitor& visitor)
39 {
40     GetterSetter* thisObject = jsCast<GetterSetter*>(cell);
41     ASSERT_GC_OBJECT_INHERITS(thisObject, info());
42     JSCell::visitChildren(thisObject, visitor);
43
44     visitor.append(thisObject->m_getter);
45     visitor.append(thisObject->m_setter);
46 }
47
48 GetterSetter* GetterSetter::withGetter(VM& vm, JSGlobalObject* globalObject, JSObject* newGetter)
49 {
50     if (isGetterNull()) {
51         setGetter(vm, globalObject, newGetter);
52         return this;
53     }
54     
55     GetterSetter* result = GetterSetter::create(vm, globalObject);
56     result->setGetter(vm, globalObject, newGetter);
57     result->setSetter(vm, globalObject, setter());
58     return result;
59 }
60
61 GetterSetter* GetterSetter::withSetter(VM& vm, JSGlobalObject* globalObject, JSObject* newSetter)
62 {
63     if (isSetterNull()) {
64         setSetter(vm, globalObject, newSetter);
65         return this;
66     }
67     
68     GetterSetter* result = GetterSetter::create(vm, globalObject);
69     result->setGetter(vm, globalObject, getter());
70     result->setSetter(vm, globalObject, newSetter);
71     return result;
72 }
73
74 JSValue callGetter(ExecState* exec, JSValue base, JSValue getterSetter)
75 {
76     VM& vm = exec->vm();
77     auto scope = DECLARE_THROW_SCOPE(vm);
78     // FIXME: Some callers may invoke get() without checking for an exception first.
79     // We work around that by checking here.
80     RETURN_IF_EXCEPTION(scope, scope.exception()->value());
81
82     JSObject* getter = jsCast<GetterSetter*>(getterSetter)->getter();
83
84     CallData callData;
85     CallType callType = getter->methodTable(vm)->getCallData(getter, callData);
86     scope.release();
87     return call(exec, getter, callType, callData, base, ArgList());
88 }
89
90 bool callSetter(ExecState* exec, JSValue base, JSValue getterSetter, JSValue value, ECMAMode ecmaMode)
91 {
92     VM& vm = exec->vm();
93     auto scope = DECLARE_THROW_SCOPE(vm);
94
95     GetterSetter* getterSetterObj = jsCast<GetterSetter*>(getterSetter);
96
97     if (getterSetterObj->isSetterNull())
98         return typeError(exec, scope, ecmaMode == StrictMode, ASCIILiteral(ReadonlyPropertyWriteError));
99
100     JSObject* setter = getterSetterObj->setter();
101
102     MarkedArgumentBuffer args;
103     args.append(value);
104
105     CallData callData;
106     CallType callType = setter->methodTable(vm)->getCallData(setter, callData);
107     scope.release();
108     call(exec, setter, callType, callData, base, args);
109     return true;
110 }
111
112 } // namespace JSC