2 * Copyright (C) 1999-2002 Harri Porten (porten@kde.org)
3 * Copyright (C) 2001 Peter Kelly (pmk@post.com)
4 * Copyright (C) 2003, 2004, 2005, 2006, 2007, 2008, 2009 Apple Inc. All rights reserved.
5 * Copyright (C) 2007 Cameron Zwarich (cwzwarich@uwaterloo.ca)
6 * Copyright (C) 2007 Maks Orlovich
8 * This library is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU Library General Public
10 * License as published by the Free Software Foundation; either
11 * version 2 of the License, or (at your option) any later version.
13 * This library is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 * Library General Public License for more details.
18 * You should have received a copy of the GNU Library General Public License
19 * along with this library; see the file COPYING.LIB. If not, write to
20 * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
21 * Boston, MA 02110-1301, USA.
26 #include "JSFunction.h"
28 #include "CodeBlock.h"
29 #include "CommonIdentifiers.h"
30 #include "CallFrame.h"
31 #include "ExceptionHelpers.h"
32 #include "FunctionPrototype.h"
33 #include "JSGlobalObject.h"
34 #include "JSNotAnObject.h"
35 #include "Interpreter.h"
36 #include "ObjectPrototype.h"
38 #include "PropertyNameArray.h"
39 #include "ScopeChainMark.h"
42 using namespace Unicode
;
45 EncodedJSValue JSC_HOST_CALL
callHostFunctionAsConstructor(ExecState
* exec
)
47 return throwVMError(exec
, createNotAConstructorError(exec
, exec
->callee()));
50 ASSERT_CLASS_FITS_IN_CELL(JSFunction
);
52 const ClassInfo
JSFunction::s_info
= { "Function", &Base::s_info
, 0, 0 };
54 bool JSFunction::isHostFunctionNonInline() const
56 return isHostFunction();
59 JSFunction::JSFunction(VPtrStealingHackType
)
60 : Base(VPtrStealingHack
)
64 JSFunction::JSFunction(ExecState
* exec
, JSGlobalObject
* globalObject
, Structure
* structure
, int length
, const Identifier
& name
, NativeExecutable
* thunk
)
65 : Base(globalObject
, structure
)
66 , m_executable(exec
->globalData(), this, thunk
)
67 , m_scopeChain(exec
->globalData(), this, globalObject
->globalScopeChain())
69 ASSERT(inherits(&s_info
));
70 putDirect(exec
->globalData(), exec
->globalData().propertyNames
->name
, jsString(exec
, name
.isNull() ? "" : name
.ustring()), DontDelete
| ReadOnly
| DontEnum
);
71 putDirect(exec
->globalData(), exec
->propertyNames().length
, jsNumber(length
), DontDelete
| ReadOnly
| DontEnum
);
74 JSFunction::JSFunction(ExecState
* exec
, JSGlobalObject
* globalObject
, Structure
* structure
, int length
, const Identifier
& name
, NativeFunction func
)
75 : Base(globalObject
, structure
)
76 , m_scopeChain(exec
->globalData(), this, globalObject
->globalScopeChain())
78 ASSERT(inherits(&s_info
));
80 // Can't do this during initialization because getHostFunction might do a GC allocation.
81 m_executable
.set(exec
->globalData(), this, exec
->globalData().getHostFunction(func
));
83 putDirect(exec
->globalData(), exec
->globalData().propertyNames
->name
, jsString(exec
, name
.isNull() ? "" : name
.ustring()), DontDelete
| ReadOnly
| DontEnum
);
84 putDirect(exec
->globalData(), exec
->propertyNames().length
, jsNumber(length
), DontDelete
| ReadOnly
| DontEnum
);
87 JSFunction::JSFunction(ExecState
* exec
, FunctionExecutable
* executable
, ScopeChainNode
* scopeChainNode
)
88 : Base(scopeChainNode
->globalObject
.get(), scopeChainNode
->globalObject
->functionStructure())
89 , m_executable(exec
->globalData(), this, executable
)
90 , m_scopeChain(exec
->globalData(), this, scopeChainNode
)
92 ASSERT(inherits(&s_info
));
93 const Identifier
& name
= static_cast<FunctionExecutable
*>(m_executable
.get())->name();
94 putDirect(exec
->globalData(), exec
->globalData().propertyNames
->name
, jsString(exec
, name
.isNull() ? "" : name
.ustring()), DontDelete
| ReadOnly
| DontEnum
);
97 JSFunction::~JSFunction()
99 ASSERT(vptr() == JSGlobalData::jsFunctionVPtr
);
102 static const char* StrictModeCallerAccessError
= "Cannot access caller property of a strict mode function";
103 static const char* StrictModeArgumentsAccessError
= "Cannot access arguments property of a strict mode function";
105 static void createDescriptorForThrowingProperty(ExecState
* exec
, PropertyDescriptor
& descriptor
, const char* message
)
107 JSValue thrower
= createTypeErrorFunction(exec
, message
);
108 descriptor
.setAccessorDescriptor(thrower
, thrower
, DontEnum
| DontDelete
| Getter
| Setter
);
111 const UString
& JSFunction::name(ExecState
* exec
)
113 return asString(getDirect(exec
->globalData(), exec
->globalData().propertyNames
->name
))->tryGetValue();
116 const UString
JSFunction::displayName(ExecState
* exec
)
118 JSValue displayName
= getDirect(exec
->globalData(), exec
->globalData().propertyNames
->displayName
);
120 if (displayName
&& isJSString(&exec
->globalData(), displayName
))
121 return asString(displayName
)->tryGetValue();
126 const UString
JSFunction::calculatedDisplayName(ExecState
* exec
)
128 const UString explicitName
= displayName(exec
);
130 if (!explicitName
.isEmpty())
136 void JSFunction::visitChildren(SlotVisitor
& visitor
)
138 ASSERT_GC_OBJECT_INHERITS(this, &s_info
);
139 COMPILE_ASSERT(StructureFlags
& OverridesVisitChildren
, OverridesVisitChildrenWithoutSettingFlag
);
140 ASSERT(structure()->typeInfo().overridesVisitChildren());
141 Base::visitChildren(visitor
);
143 visitor
.append(&m_scopeChain
);
145 visitor
.append(&m_executable
);
148 CallType
JSFunction::getCallData(CallData
& callData
)
150 if (isHostFunction()) {
151 callData
.native
.function
= nativeFunction();
154 callData
.js
.functionExecutable
= jsExecutable();
155 callData
.js
.scopeChain
= scope();
159 JSValue
JSFunction::argumentsGetter(ExecState
* exec
, JSValue slotBase
, const Identifier
&)
161 JSFunction
* thisObj
= asFunction(slotBase
);
162 ASSERT(!thisObj
->isHostFunction());
163 return exec
->interpreter()->retrieveArguments(exec
, thisObj
);
166 JSValue
JSFunction::callerGetter(ExecState
* exec
, JSValue slotBase
, const Identifier
&)
168 JSFunction
* thisObj
= asFunction(slotBase
);
169 ASSERT(!thisObj
->isHostFunction());
170 return exec
->interpreter()->retrieveCaller(exec
, thisObj
);
173 JSValue
JSFunction::lengthGetter(ExecState
*, JSValue slotBase
, const Identifier
&)
175 JSFunction
* thisObj
= asFunction(slotBase
);
176 ASSERT(!thisObj
->isHostFunction());
177 return jsNumber(thisObj
->jsExecutable()->parameterCount());
180 static inline WriteBarrierBase
<Unknown
>* createPrototypeProperty(JSGlobalData
& globalData
, JSGlobalObject
* globalObject
, JSFunction
* function
)
182 ASSERT(!function
->isHostFunction());
184 ExecState
* exec
= globalObject
->globalExec();
185 if (WriteBarrierBase
<Unknown
>* location
= function
->getDirectLocation(globalData
, exec
->propertyNames().prototype
))
187 JSObject
* prototype
= constructEmptyObject(exec
, globalObject
->emptyObjectStructure());
188 prototype
->putDirect(globalData
, exec
->propertyNames().constructor
, function
, DontEnum
);
189 function
->putDirect(globalData
, exec
->propertyNames().prototype
, prototype
, DontDelete
| DontEnum
);
190 return function
->getDirectLocation(exec
->globalData(), exec
->propertyNames().prototype
);
193 void JSFunction::preventExtensions(JSGlobalData
& globalData
)
195 if (!isHostFunction())
196 createPrototypeProperty(globalData
, scope()->globalObject
.get(), this);
197 JSObject::preventExtensions(globalData
);
200 bool JSFunction::getOwnPropertySlot(ExecState
* exec
, const Identifier
& propertyName
, PropertySlot
& slot
)
202 if (isHostFunction())
203 return Base::getOwnPropertySlot(exec
, propertyName
, slot
);
205 if (propertyName
== exec
->propertyNames().prototype
) {
206 WriteBarrierBase
<Unknown
>* location
= getDirectLocation(exec
->globalData(), propertyName
);
209 location
= createPrototypeProperty(exec
->globalData(), scope()->globalObject
.get(), this);
211 slot
.setValue(this, location
->get(), offsetForLocation(location
));
214 if (propertyName
== exec
->propertyNames().arguments
) {
215 if (jsExecutable()->isStrictMode()) {
216 throwTypeError(exec
, "Can't access arguments object of a strict mode function");
217 slot
.setValue(jsNull());
221 slot
.setCacheableCustom(this, argumentsGetter
);
225 if (propertyName
== exec
->propertyNames().length
) {
226 slot
.setCacheableCustom(this, lengthGetter
);
230 if (propertyName
== exec
->propertyNames().caller
) {
231 if (jsExecutable()->isStrictMode()) {
232 throwTypeError(exec
, StrictModeCallerAccessError
);
233 slot
.setValue(jsNull());
236 slot
.setCacheableCustom(this, callerGetter
);
240 return Base::getOwnPropertySlot(exec
, propertyName
, slot
);
243 bool JSFunction::getOwnPropertyDescriptor(ExecState
* exec
, const Identifier
& propertyName
, PropertyDescriptor
& descriptor
)
245 if (isHostFunction())
246 return Base::getOwnPropertyDescriptor(exec
, propertyName
, descriptor
);
248 if (propertyName
== exec
->propertyNames().prototype
) {
250 getOwnPropertySlot(exec
, propertyName
, slot
);
251 return Base::getOwnPropertyDescriptor(exec
, propertyName
, descriptor
);
254 if (propertyName
== exec
->propertyNames().arguments
) {
255 if (jsExecutable()->isStrictMode())
256 createDescriptorForThrowingProperty(exec
, descriptor
, StrictModeArgumentsAccessError
);
258 descriptor
.setDescriptor(exec
->interpreter()->retrieveArguments(exec
, this), ReadOnly
| DontEnum
| DontDelete
);
262 if (propertyName
== exec
->propertyNames().length
) {
263 descriptor
.setDescriptor(jsNumber(jsExecutable()->parameterCount()), ReadOnly
| DontEnum
| DontDelete
);
267 if (propertyName
== exec
->propertyNames().caller
) {
268 if (jsExecutable()->isStrictMode())
269 createDescriptorForThrowingProperty(exec
, descriptor
, StrictModeCallerAccessError
);
271 descriptor
.setDescriptor(exec
->interpreter()->retrieveCaller(exec
, this), ReadOnly
| DontEnum
| DontDelete
);
275 return Base::getOwnPropertyDescriptor(exec
, propertyName
, descriptor
);
278 void JSFunction::getOwnPropertyNames(ExecState
* exec
, PropertyNameArray
& propertyNames
, EnumerationMode mode
)
280 if (!isHostFunction() && (mode
== IncludeDontEnumProperties
)) {
281 // Make sure prototype has been reified.
283 getOwnPropertySlot(exec
, exec
->propertyNames().prototype
, slot
);
285 propertyNames
.add(exec
->propertyNames().arguments
);
286 propertyNames
.add(exec
->propertyNames().callee
);
287 propertyNames
.add(exec
->propertyNames().caller
);
288 propertyNames
.add(exec
->propertyNames().length
);
290 Base::getOwnPropertyNames(exec
, propertyNames
, mode
);
293 void JSFunction::put(ExecState
* exec
, const Identifier
& propertyName
, JSValue value
, PutPropertySlot
& slot
)
295 if (isHostFunction()) {
296 Base::put(exec
, propertyName
, value
, slot
);
299 if (propertyName
== exec
->propertyNames().prototype
) {
300 // Make sure prototype has been reified, such that it can only be overwritten
301 // following the rules set out in ECMA-262 8.12.9.
303 getOwnPropertySlot(exec
, propertyName
, slot
);
305 if (jsExecutable()->isStrictMode()) {
306 if (propertyName
== exec
->propertyNames().arguments
) {
307 throwTypeError(exec
, StrictModeArgumentsAccessError
);
310 if (propertyName
== exec
->propertyNames().caller
) {
311 throwTypeError(exec
, StrictModeCallerAccessError
);
315 if (propertyName
== exec
->propertyNames().arguments
|| propertyName
== exec
->propertyNames().length
)
317 Base::put(exec
, propertyName
, value
, slot
);
320 bool JSFunction::deleteProperty(ExecState
* exec
, const Identifier
& propertyName
)
322 if (isHostFunction())
323 return Base::deleteProperty(exec
, propertyName
);
324 if (propertyName
== exec
->propertyNames().arguments
|| propertyName
== exec
->propertyNames().length
)
326 return Base::deleteProperty(exec
, propertyName
);
329 // ECMA 13.2.2 [[Construct]]
330 ConstructType
JSFunction::getConstructData(ConstructData
& constructData
)
332 if (isHostFunction())
333 return ConstructTypeNone
;
334 constructData
.js
.functionExecutable
= jsExecutable();
335 constructData
.js
.scopeChain
= scope();
336 return ConstructTypeJS
;