fea00b083ae8202796e9869a6a58672f6d2a4fd8
[WebKit-https.git] / Source / JavaScriptCore / runtime / InternalFunction.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-2008, 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 "InternalFunction.h"
25
26 #include "FunctionPrototype.h"
27 #include "JSGlobalObject.h"
28 #include "JSString.h"
29 #include "JSCInlines.h"
30
31 namespace JSC {
32
33 STATIC_ASSERT_IS_TRIVIALLY_DESTRUCTIBLE(InternalFunction);
34
35 const ClassInfo InternalFunction::s_info = { "Function", &Base::s_info, 0, CREATE_METHOD_TABLE(InternalFunction) };
36
37 InternalFunction::InternalFunction(VM& vm, Structure* structure)
38     : JSDestructibleObject(vm, structure)
39 {
40     // exec->vm() wants callees to not be large allocations.
41     RELEASE_ASSERT(!isLargeAllocation());
42 }
43
44 void InternalFunction::finishCreation(VM& vm, const String& name)
45 {
46     Base::finishCreation(vm);
47     ASSERT(inherits(info()));
48     ASSERT(methodTable()->getCallData != InternalFunction::info()->methodTable.getCallData);
49     JSString* nameString = jsString(&vm, name);
50     m_originalName.set(vm, this, nameString);
51     putDirect(vm, vm.propertyNames->name, nameString, ReadOnly | DontEnum);
52 }
53
54 void InternalFunction::visitChildren(JSCell* cell, SlotVisitor& visitor)
55 {
56     InternalFunction* thisObject = jsCast<InternalFunction*>(cell);
57     ASSERT_GC_OBJECT_INHERITS(thisObject, info());
58     Base::visitChildren(thisObject, visitor);
59     
60     visitor.append(&thisObject->m_originalName);
61 }
62
63 const String& InternalFunction::name()
64 {
65     const String& name = m_originalName->tryGetValue();
66     ASSERT(name); // m_originalName was built from a String, and hence, there is no rope to resolve.
67     return name;
68 }
69
70 const String InternalFunction::displayName(VM& vm)
71 {
72     JSValue displayName = getDirect(vm, vm.propertyNames->displayName);
73     
74     if (displayName && isJSString(displayName))
75         return asString(displayName)->tryGetValue();
76     
77     return String();
78 }
79
80 CallType InternalFunction::getCallData(JSCell*, CallData&)
81 {
82     RELEASE_ASSERT_NOT_REACHED();
83     return CallType::None;
84 }
85
86 const String InternalFunction::calculatedDisplayName(VM& vm)
87 {
88     const String explicitName = displayName(vm);
89     
90     if (!explicitName.isEmpty())
91         return explicitName;
92     
93     return name();
94 }
95
96 Structure* InternalFunction::createSubclassStructure(ExecState* exec, JSValue newTarget, Structure* baseClass)
97 {
98
99     VM& vm = exec->vm();
100     auto scope = DECLARE_THROW_SCOPE(vm);
101     // We allow newTarget == JSValue() because the API needs to be able to create classes without having a real JS frame.
102     // Since we don't allow subclassing in the API we just treat newTarget == JSValue() as newTarget == exec->jsCallee()
103     ASSERT(!newTarget || newTarget.isConstructor());
104
105     if (newTarget && newTarget != exec->jsCallee()) {
106         // newTarget may be an InternalFunction if we were called from Reflect.construct.
107         JSFunction* targetFunction = jsDynamicCast<JSFunction*>(newTarget);
108
109         if (LIKELY(targetFunction)) {
110             Structure* structure = targetFunction->rareData(vm)->internalFunctionAllocationStructure();
111             if (LIKELY(structure && structure->classInfo() == baseClass->classInfo()))
112                 return structure;
113
114             // Note, Reflect.construct might cause the profile to churn but we don't care.
115             JSValue prototypeValue = newTarget.get(exec, exec->propertyNames().prototype);
116             RETURN_IF_EXCEPTION(scope, nullptr);
117             if (JSObject* prototype = jsDynamicCast<JSObject*>(prototypeValue))
118                 return targetFunction->rareData(vm)->createInternalFunctionAllocationStructureFromBase(vm, prototype, baseClass);
119         } else {
120             JSValue prototypeValue = newTarget.get(exec, exec->propertyNames().prototype);
121             RETURN_IF_EXCEPTION(scope, nullptr);
122             if (JSObject* prototype = jsDynamicCast<JSObject*>(prototypeValue)) {
123                 // This only happens if someone Reflect.constructs our builtin constructor with another builtin constructor as the new.target.
124                 // Thus, we don't care about the cost of looking up the structure from our hash table every time.
125                 return vm.prototypeMap.emptyStructureForPrototypeFromBaseStructure(prototype, baseClass);
126             }
127         }
128     }
129     
130     return baseClass;
131 }
132
133
134 } // namespace JSC