]>
git.saurik.com Git - apple/javascriptcore.git/blob - bytecode/CodeOrigin.cpp
2 * Copyright (C) 2012, 2013 Apple Inc. All rights reserved.
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions
7 * 1. Redistributions of source code must retain the above copyright
8 * notice, this list of conditions and the following disclaimer.
9 * 2. Redistributions in binary form must reproduce the above copyright
10 * notice, this list of conditions and the following disclaimer in the
11 * documentation and/or other materials provided with the distribution.
13 * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
14 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
17 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
18 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
19 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
20 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
21 * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
23 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27 #include "CodeOrigin.h"
29 #include "CallFrame.h"
30 #include "CodeBlock.h"
31 #include "Executable.h"
32 #include "JSCInlines.h"
36 unsigned CodeOrigin::inlineDepthForCallFrame(InlineCallFrame
* inlineCallFrame
)
39 for (InlineCallFrame
* current
= inlineCallFrame
; current
; current
= current
->caller
.inlineCallFrame
)
44 unsigned CodeOrigin::inlineDepth() const
46 return inlineDepthForCallFrame(inlineCallFrame
);
49 bool CodeOrigin::isApproximatelyEqualTo(const CodeOrigin
& other
) const
59 if (a
.isHashTableDeletedValue())
60 return b
.isHashTableDeletedValue();
61 if (b
.isHashTableDeletedValue())
68 if (a
.bytecodeIndex
!= b
.bytecodeIndex
)
71 if ((!!a
.inlineCallFrame
) != (!!b
.inlineCallFrame
))
74 if (!a
.inlineCallFrame
)
77 if (a
.inlineCallFrame
->executable
!= b
.inlineCallFrame
->executable
)
80 a
= a
.inlineCallFrame
->caller
;
81 b
= b
.inlineCallFrame
->caller
;
85 unsigned CodeOrigin::approximateHash() const
89 if (isHashTableDeletedValue())
93 CodeOrigin codeOrigin
= *this;
95 result
+= codeOrigin
.bytecodeIndex
;
97 if (!codeOrigin
.inlineCallFrame
)
100 result
+= WTF::PtrHash
<JSCell
*>::hash(codeOrigin
.inlineCallFrame
->executable
.get());
102 codeOrigin
= codeOrigin
.inlineCallFrame
->caller
;
106 Vector
<CodeOrigin
> CodeOrigin::inlineStack() const
108 Vector
<CodeOrigin
> result(inlineDepth());
109 result
.last() = *this;
110 unsigned index
= result
.size() - 2;
111 for (InlineCallFrame
* current
= inlineCallFrame
; current
; current
= current
->caller
.inlineCallFrame
)
112 result
[index
--] = current
->caller
;
113 RELEASE_ASSERT(!result
[0].inlineCallFrame
);
117 void CodeOrigin::dump(PrintStream
& out
) const
124 Vector
<CodeOrigin
> stack
= inlineStack();
125 for (unsigned i
= 0; i
< stack
.size(); ++i
) {
129 if (InlineCallFrame
* frame
= stack
[i
].inlineCallFrame
) {
130 out
.print(frame
->briefFunctionInformation(), ":<", RawPointer(frame
->executable
.get()), "> ");
131 if (frame
->isClosureCall
)
132 out
.print("(closure) ");
135 out
.print("bc#", stack
[i
].bytecodeIndex
);
139 void CodeOrigin::dumpInContext(PrintStream
& out
, DumpContext
*) const
144 JSFunction
* InlineCallFrame::calleeForCallFrame(ExecState
* exec
) const
146 return jsCast
<JSFunction
*>(calleeRecovery
.recover(exec
));
149 CodeBlockHash
InlineCallFrame::hash() const
151 return jsCast
<FunctionExecutable
*>(executable
.get())->codeBlockFor(
152 specializationKind())->hash();
155 CString
InlineCallFrame::hashAsStringIfPossible() const
157 return jsCast
<FunctionExecutable
*>(executable
.get())->codeBlockFor(
158 specializationKind())->hashAsStringIfPossible();
161 CString
InlineCallFrame::inferredName() const
163 return jsCast
<FunctionExecutable
*>(executable
.get())->inferredName().utf8();
166 CodeBlock
* InlineCallFrame::baselineCodeBlock() const
168 return jsCast
<FunctionExecutable
*>(executable
.get())->baselineCodeBlockFor(specializationKind());
171 void InlineCallFrame::dumpBriefFunctionInformation(PrintStream
& out
) const
173 out
.print(inferredName(), "#", hashAsStringIfPossible());
176 void InlineCallFrame::dumpInContext(PrintStream
& out
, DumpContext
* context
) const
178 out
.print(briefFunctionInformation(), ":<", RawPointer(executable
.get()));
179 if (executable
->isStrictMode())
180 out
.print(" (StrictMode)");
181 out
.print(", bc#", caller
.bytecodeIndex
, ", ", specializationKind());
183 out
.print(", closure call");
185 out
.print(", known callee: ", inContext(calleeRecovery
.constant(), context
));
186 out
.print(", numArgs+this = ", arguments
.size());
187 out
.print(", stack < loc", VirtualRegister(stackOffset
).toLocal());
191 void InlineCallFrame::dump(PrintStream
& out
) const
193 dumpInContext(out
, 0);