]>
git.saurik.com Git - apple/javascriptcore.git/blob - dfg/DFGGraph.cpp
b1e69914d41030f49ca72d5f65b617b7222f7aa4
2 * Copyright (C) 2011 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.
29 #include "CodeBlock.h"
33 namespace JSC
{ namespace DFG
{
37 // Creates an array of stringized names.
38 static const char* dfgOpNames
[] = {
39 #define STRINGIZE_DFG_OP_ENUM(opcode, flags) #opcode ,
40 FOR_EACH_DFG_OP(STRINGIZE_DFG_OP_ENUM
)
41 #undef STRINGIZE_DFG_OP_ENUM
44 void Graph::dump(NodeIndex nodeIndex
, CodeBlock
* codeBlock
)
46 Node
& node
= at(nodeIndex
);
47 NodeType op
= node
.op
;
49 unsigned refCount
= node
.refCount();
52 bool mustGenerate
= node
.mustGenerate();
56 // Example/explanation of dataflow dump output
58 // 14: <!2:7> GetByVal(@3, @13)
61 // (1) The nodeIndex of this operation.
62 // (2) The reference count. The number printed is the 'real' count,
63 // not including the 'mustGenerate' ref. If the node is
64 // 'mustGenerate' then the count it prefixed with '!'.
65 // (3) The virtual register slot assigned to this node.
66 // (4) The name of the operation.
67 // (5) The arguments to the operation. The may be of the form:
68 // @# - a NodeIndex referencing a prior node in the graph.
69 // arg# - an argument number.
70 // $# - the index in the CodeBlock of a constant { for numeric constants the value is displayed | for integers, in both decimal and hex }.
71 // id# - the index in the CodeBlock of an identifier { if codeBlock is passed to dump(), the string representation is displayed }.
72 // var# - the index of a var on the global object, used by GetGlobalVar/PutGlobalVar operations.
73 printf("% 4d:\t<%c%u:", (int)nodeIndex
, mustGenerate
? '!' : ' ', refCount
);
75 printf("%u", node
.virtualRegister());
78 printf(">\t%s(", dfgOpNames
[op
& NodeIdMask
]);
79 if (node
.child1
!= NoNode
)
80 printf("@%u", node
.child1
);
81 if (node
.child2
!= NoNode
)
82 printf(", @%u", node
.child2
);
83 if (node
.child3
!= NoNode
)
84 printf(", @%u", node
.child3
);
85 bool hasPrinted
= node
.child1
!= NoNode
;
87 if (node
.hasVarNumber()) {
88 printf("%svar%u", hasPrinted
? ", " : "", node
.varNumber());
91 if (node
.hasIdentifier()) {
93 printf("%sid%u{%s}", hasPrinted
? ", " : "", node
.identifierNumber(), codeBlock
->identifier(node
.identifierNumber()).ustring().utf8().data());
95 printf("%sid%u", hasPrinted
? ", " : "", node
.identifierNumber());
98 if (node
.hasLocal()) {
99 int local
= node
.local();
100 if (operandIsArgument(local
))
101 printf("%sarg%u", hasPrinted
? ", " : "", local
- codeBlock
->thisRegister());
103 printf("%sr%u", hasPrinted
? ", " : "", local
);
106 if (op
== Int32Constant
) {
107 printf("%s$%u{%d|0x%08x}", hasPrinted
? ", " : "", node
.constantNumber(), node
.int32Constant(), node
.int32Constant());
110 if (op
== DoubleConstant
) {
111 printf("%s$%u{%f})", hasPrinted
? ", " : "", node
.constantNumber(), node
.numericConstant());
114 if (op
== JSConstant
) {
115 printf("%s$%u", hasPrinted
? ", " : "", node
.constantNumber());
118 if (node
.isBranch() || node
.isJump()) {
119 printf("%sT:#%u", hasPrinted
? ", " : "", blockIndexForBytecodeOffset(node
.takenBytecodeOffset()));
122 if (node
.isBranch()) {
123 printf("%sF:#%u", hasPrinted
? ", " : "", blockIndexForBytecodeOffset(node
.notTakenBytecodeOffset()));
130 void Graph::dump(CodeBlock
* codeBlock
)
132 for (size_t b
= 0; b
< m_blocks
.size(); ++b
) {
133 printf("Block #%u:\n", (int)b
);
134 for (size_t i
= m_blocks
[b
]->begin
; i
< m_blocks
[b
]->end
; ++i
)
137 printf("Phi Nodes:\n");
138 for (size_t i
= m_blocks
.last()->end
; i
< size(); ++i
)
144 // FIXME: Convert this method to be iterative, not recursive.
145 void Graph::refChildren(NodeIndex op
)
149 if (node
.child1
== NoNode
) {
150 ASSERT(node
.child2
== NoNode
&& node
.child3
== NoNode
);
155 if (node
.child2
== NoNode
) {
156 ASSERT(node
.child3
== NoNode
);
161 if (node
.child3
== NoNode
)
166 } } // namespace JSC::DFG