2 * Copyright (C) 1999-2001 Harri Porten (porten@kde.org)
3 * Copyright (C) 2003, 2004, 2005, 2006, 2007, 2008 Apple Inc. All rights reserved.
5 * This library is free software; you can redistribute it and/or
6 * modify it under the terms of the GNU Lesser General Public
7 * License as published by the Free Software Foundation; either
8 * version 2 of the License, or (at your option) any later version.
10 * This library is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * Lesser General Public License for more details.
15 * You should have received a copy of the GNU Lesser General Public
16 * License along with this library; if not, write to the Free Software
17 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
22 #include "FunctionConstructor.h"
25 #include "ExceptionHelpers.h"
26 #include "FunctionPrototype.h"
27 #include "JSFunction.h"
28 #include "JSGlobalObject.h"
33 #include "UStringBuilder.h"
34 #include "UStringConcatenate.h"
38 ASSERT_CLASS_FITS_IN_CELL(FunctionConstructor
);
39 ASSERT_HAS_TRIVIAL_DESTRUCTOR(FunctionConstructor
);
41 const ClassInfo
FunctionConstructor::s_info
= { "Function", &Base::s_info
, 0, 0, CREATE_METHOD_TABLE(FunctionConstructor
) };
43 FunctionConstructor::FunctionConstructor(JSGlobalObject
* globalObject
, Structure
* structure
)
44 : InternalFunction(globalObject
, structure
)
48 void FunctionConstructor::finishCreation(ExecState
* exec
, FunctionPrototype
* functionPrototype
)
50 Base::finishCreation(exec
->globalData(), Identifier(exec
, functionPrototype
->classInfo()->className
));
51 putDirectWithoutTransition(exec
->globalData(), exec
->propertyNames().prototype
, functionPrototype
, DontEnum
| DontDelete
| ReadOnly
);
53 // Number of arguments for constructor
54 putDirectWithoutTransition(exec
->globalData(), exec
->propertyNames().length
, jsNumber(1), ReadOnly
| DontDelete
| DontEnum
);
57 static EncodedJSValue JSC_HOST_CALL
constructWithFunctionConstructor(ExecState
* exec
)
60 return JSValue::encode(constructFunction(exec
, asInternalFunction(exec
->callee())->globalObject(), args
));
63 ConstructType
FunctionConstructor::getConstructData(JSCell
*, ConstructData
& constructData
)
65 constructData
.native
.function
= constructWithFunctionConstructor
;
66 return ConstructTypeHost
;
69 static EncodedJSValue JSC_HOST_CALL
callFunctionConstructor(ExecState
* exec
)
72 return JSValue::encode(constructFunction(exec
, asInternalFunction(exec
->callee())->globalObject(), args
));
75 // ECMA 15.3.1 The Function Constructor Called as a Function
76 CallType
FunctionConstructor::getCallData(JSCell
*, CallData
& callData
)
78 callData
.native
.function
= callFunctionConstructor
;
82 // ECMA 15.3.2 The Function Constructor
83 JSObject
* constructFunction(ExecState
* exec
, JSGlobalObject
* globalObject
, const ArgList
& args
, const Identifier
& functionName
, const UString
& sourceURL
, const TextPosition
& position
)
85 if (!globalObject
->evalEnabled())
86 return throwError(exec
, createEvalError(exec
, "Function constructor is disabled"));
87 return constructFunctionSkippingEvalEnabledCheck(exec
, globalObject
, args
, functionName
, sourceURL
, position
);
90 JSObject
* constructFunctionSkippingEvalEnabledCheck(ExecState
* exec
, JSGlobalObject
* globalObject
, const ArgList
& args
, const Identifier
& functionName
, const UString
& sourceURL
, const TextPosition
& position
)
92 // Functions need to have a space following the opening { due to for web compatibility
93 // see https://bugs.webkit.org/show_bug.cgi?id=24350
94 // We also need \n before the closing } to handle // comments at the end of the last line
97 program
= "(function() { \n})";
98 else if (args
.size() == 1)
99 program
= makeUString("(function() { ", args
.at(0).toString(exec
)->value(exec
), "\n})");
101 UStringBuilder builder
;
102 builder
.append("(function(");
103 builder
.append(args
.at(0).toString(exec
)->value(exec
));
104 for (size_t i
= 1; i
< args
.size() - 1; i
++) {
106 builder
.append(args
.at(i
).toString(exec
)->value(exec
));
108 builder
.append(") { ");
109 builder
.append(args
.at(args
.size() - 1).toString(exec
)->value(exec
));
110 builder
.append("\n})");
111 program
= builder
.toUString();
114 JSGlobalData
& globalData
= globalObject
->globalData();
115 SourceCode source
= makeSource(program
, sourceURL
, position
);
116 JSObject
* exception
= 0;
117 FunctionExecutable
* function
= FunctionExecutable::fromGlobalCode(functionName
, exec
, exec
->dynamicGlobalObject()->debugger(), source
, &exception
);
120 return throwError(exec
, exception
);
123 ScopeChainNode
* scopeChain
= ScopeChainNode::create(exec
, 0, globalObject
, &globalData
, globalObject
, exec
->globalThisValue());
124 return JSFunction::create(exec
, function
, scopeChain
);
127 // ECMA 15.3.2 The Function Constructor
128 JSObject
* constructFunction(ExecState
* exec
, JSGlobalObject
* globalObject
, const ArgList
& args
)
130 return constructFunction(exec
, globalObject
, args
, Identifier(exec
, "anonymous"), UString(), TextPosition::minimumPosition());