Changeset 248642 in webkit
- Timestamp:
- Aug 13, 2019, 4:31:19 PM (6 years ago)
- Location:
- trunk/Source/JavaScriptCore
- Files:
-
- 9 edited
Legend:
- Unmodified
- Added
- Removed
-
trunk/Source/JavaScriptCore/ChangeLog
r248605 r248642 1 2019-08-13 Mark Lam <mark.lam@apple.com> 2 3 Add phase, block, and node numbers to left margin of DFG graph dumps. 4 https://bugs.webkit.org/show_bug.cgi?id=200693 5 6 Reviewed by Saam Barati. 7 8 When scrolling through the DFG graph dumps, it's easy to get lost as to which phase 9 or block one is looking at, especially if the blocks are long. This patch adds 10 node index, block number, and phase number on the left margin of the dumps. 11 Here's a sample: 12 13 53: %Bd:Function = 0x1079fd960:[Function, {}, NonArray, Proto:0x1079d8000, Leaf] 14 53: %Bf:Function = 0x1079b0700:[Function, {name:100, prototype:101, length:102, stackTraceLimit:103}, NonArray, Proto:0x1079d8000, Leaf] 15 53: %Bj:Function = 0x1079fd5e0:[Function, {name:100, length:101, toString:102, apply:103, call:104, bind:105, Symbol.hasInstance:106, caller:107, arguments:108, constructor:109}, NonArray, Proto:0x1079c0000, Leaf] 16 53: %CV:JSGlobalLexicalEnvironment = 0x1079fd6c0:[JSGlobalLexicalEnvironment, {}, NonArray, Leaf] 17 18 53: Phase liveness analysis changed the IR. 19 20 54: Beginning DFG phase OSR availability analysis. 21 54: Before OSR availability analysis: 22 23 54: DFG for foo#DXMNag:[0x1079a4850->0x1079a4130->0x1079c7600, DFGFunctionCall, 204 (NeverInline)]: 24 54: Fixpoint state: FixpointConverged; Form: SSA; Unification state: GloballyUnified; Ref count state: ExactRefCount 25 54: Argument formats for entrypoint index: 0 : FlushedJSValue, FlushedCell, FlushedJSValue 26 27 0 54: Block #0 (bc#0): (OSR target) 28 0 54: Execution count: 1.000000 29 0 54: Predecessors: 30 0 54: Successors: 31 0 54: Dominated by: #0 32 0 54: Dominates: #0 33 0 54: Dominance Frontier: 34 0 54: Iterated Dominance Frontier: 35 0 54: Backwards dominates by: #root #0 36 0 54: Backwards dominates: #0 37 0 54: Control equivalent to: #0 38 0 54: States: StructuresAreWatched 39 0 54: Live: 40 0 54: Values 41 0 0 54: 53:< 1:-> JSConstant(JS|UseAsOther, Other, Null, bc#0, ExitValid) 42 1 0 54: 64:< 2:-> JSConstant(JS|UseAsOther, NonBoolInt32, Int32: 10, bc#0, ExitValid) 43 2 0 54: 3:< 5:-> JSConstant(JS|PureInt, Other, Undefined, bc#0, ExitValid) 44 3 0 54: 32:< 1:-> JSConstant(JS|UseAsOther, Bool, False, bc#0, ExitValid) 45 4 0 54: 19:< 2:-> JSConstant(JS|UseAsOther, OtherObj, Weak:Object: 0x1079d4000 with butterfly 0x0 (Structure %CV:JSGlobalLexicalEnvironment), StructureID: 31423, bc#0, ExitValid) 46 47 The numbers in the left margin before the ':' are node index (i.e. the index of the 48 node in the block, not to be confused with node->index() which is the node ID), block 49 number, and phase number respectively. Now, we can scroll thru the dumps quickly 50 and tell at a glance when we've scrolled passed the end of a phase, or block. 51 These sets of numbers can also serve as a positional marker that we can search for 52 to return to a node in the dump after scrolling away. 53 54 Currently, these numbers are only added to the DFG part. The FTL (from lowering 55 to B3 onwards) does not have this feature yet. 56 57 * dfg/DFGDesiredWatchpoints.cpp: 58 (JSC::DFG::DesiredWatchpoints::dumpInContext const): 59 * dfg/DFGDesiredWatchpoints.h: 60 * dfg/DFGGraph.cpp: 61 (JSC::DFG::Graph::dumpCodeOrigin): 62 (JSC::DFG::Graph::dump): 63 (JSC::DFG::Graph::dumpBlockHeader): 64 (JSC::DFG::Prefix::dump const): 65 * dfg/DFGGraph.h: 66 (JSC::DFG::Prefix::Prefix): 67 (JSC::DFG::Prefix::clearBlockIndex): 68 (JSC::DFG::Prefix::clearNodeIndex): 69 (JSC::DFG::Prefix::enable): 70 (JSC::DFG::Prefix::disable): 71 (JSC::DFG::Graph::prefix): 72 (JSC::DFG::Graph::nextPhase): 73 * dfg/DFGPhase.cpp: 74 (JSC::DFG::Phase::beginPhase): 75 * dfg/DFGPhase.h: 76 (JSC::DFG::runAndLog): 77 * dfg/DFGPlan.cpp: 78 (JSC::DFG::Plan::compileInThreadImpl): 79 * dfg/DFGValueRepReductionPhase.cpp: 80 (JSC::DFG::ValueRepReductionPhase::convertValueRepsToDouble): 81 1 82 2019-08-13 Michael Saboff <msaboff@apple.com> 2 83 -
trunk/Source/JavaScriptCore/dfg/DFGDesiredWatchpoints.cpp
r246073 r248642 1 1 /* 2 * Copyright (C) 2013-201 6Apple Inc. All rights reserved.2 * Copyright (C) 2013-2019 Apple Inc. All rights reserved. 3 3 * 4 4 * Redistribution and use in source and binary forms, with or without … … 31 31 #include "ArrayBufferNeuteringWatchpointSet.h" 32 32 #include "CodeBlock.h" 33 #include "DFGGraph.h" 33 34 #include "JSCInlines.h" 34 35 … … 140 141 void DesiredWatchpoints::dumpInContext(PrintStream& out, DumpContext* context) const 141 142 { 142 out.print("Desired watchpoints:\n"); 143 out.print(" Watchpoint sets: ", inContext(m_sets, context), "\n"); 144 out.print(" Inline watchpoint sets: ", inContext(m_inlineSets, context), "\n"); 145 out.print(" SymbolTables: ", inContext(m_symbolTables, context), "\n"); 146 out.print(" FunctionExecutables: ", inContext(m_functionExecutables, context), "\n"); 147 out.print(" Buffer views: ", inContext(m_bufferViews, context), "\n"); 148 out.print(" Object property conditions: ", inContext(m_adaptiveStructureSets, context), "\n"); 143 Prefix noPrefix(Prefix::NoHeader); 144 Prefix& prefix = context && context->graph ? context->graph->prefix() : noPrefix; 145 out.print(prefix, "Desired watchpoints:\n"); 146 out.print(prefix, " Watchpoint sets: ", inContext(m_sets, context), "\n"); 147 out.print(prefix, " Inline watchpoint sets: ", inContext(m_inlineSets, context), "\n"); 148 out.print(prefix, " SymbolTables: ", inContext(m_symbolTables, context), "\n"); 149 out.print(prefix, " FunctionExecutables: ", inContext(m_functionExecutables, context), "\n"); 150 out.print(prefix, " Buffer views: ", inContext(m_bufferViews, context), "\n"); 151 out.print(prefix, " Object property conditions: ", inContext(m_adaptiveStructureSets, context), "\n"); 149 152 } 150 153 -
trunk/Source/JavaScriptCore/dfg/DFGDesiredWatchpoints.h
r246073 r248642 1 1 /* 2 * Copyright (C) 2013-201 5Apple Inc. All rights reserved.2 * Copyright (C) 2013-2019 Apple Inc. All rights reserved. 3 3 * 4 4 * Redistribution and use in source and binary forms, with or without … … 40 40 41 41 class Graph; 42 struct Prefix; 42 43 43 44 template<typename T> … … 207 208 } 208 209 void dumpInContext(PrintStream&, DumpContext*) const; 209 void dump(PrintStream&) const;210 210 211 211 private: -
trunk/Source/JavaScriptCore/dfg/DFGGraph.cpp
r244764 r248642 1 1 /* 2 * Copyright (C) 2011-201 8Apple Inc. All rights reserved.2 * Copyright (C) 2011-2019 Apple Inc. All rights reserved. 3 3 * 4 4 * Redistribution and use in source and binary forms, with or without … … 110 110 } 111 111 112 bool Graph::dumpCodeOrigin(PrintStream& out, const char* prefix, Node*& previousNodeRef, Node* currentNode, DumpContext* context) 113 { 112 bool Graph::dumpCodeOrigin(PrintStream& out, const char* prefixStr, Node*& previousNodeRef, Node* currentNode, DumpContext* context) 113 { 114 Prefix myPrefix(prefixStr); 115 Prefix& prefix = prefixStr ? myPrefix : m_prefix; 116 114 117 if (!currentNode->origin.semantic) 115 118 return false; … … 166 169 } 167 170 168 void Graph::dump(PrintStream& out, const char* prefix, Node* node, DumpContext* context) 169 { 171 void Graph::dump(PrintStream& out, const char* prefixStr, Node* node, DumpContext* context) 172 { 173 Prefix myPrefix(prefixStr); 174 Prefix& prefix = prefixStr ? myPrefix : m_prefix; 175 170 176 NodeType op = node->op(); 171 177 … … 424 430 static BasicBlock* unboxLoopNode(BasicBlock* block) { return block; } 425 431 426 void Graph::dumpBlockHeader(PrintStream& out, const char* prefix, BasicBlock* block, PhiNodeDumpMode phiNodeDumpMode, DumpContext* context) 427 { 432 void Graph::dumpBlockHeader(PrintStream& out, const char* prefixStr, BasicBlock* block, PhiNodeDumpMode phiNodeDumpMode, DumpContext* context) 433 { 434 Prefix myPrefix(prefixStr); 435 Prefix& prefix = prefixStr ? myPrefix : m_prefix; 436 428 437 out.print(prefix, "Block ", *block, " (", inContext(block->at(0)->origin.semantic, context), "):", 429 438 block->isReachable ? "" : " (skipped)", block->isOSRTarget ? " (OSR target)" : "", block->isCatchEntrypoint ? " (Catch Entrypoint)" : "", "\n"); … … 520 529 void Graph::dump(PrintStream& out, DumpContext* context) 521 530 { 531 Prefix& prefix = m_prefix; 522 532 DumpContext myContext; 523 533 myContext.graph = this; … … 526 536 527 537 out.print("\n"); 528 out.print( "DFG for ", CodeBlockWithJITType(m_codeBlock, JITType::DFGJIT), ":\n");529 out.print( " Fixpoint state: ", m_fixpointState, "; Form: ", m_form, "; Unification state: ", m_unificationState, "; Ref count state: ", m_refCountState, "\n");538 out.print(prefix, "DFG for ", CodeBlockWithJITType(m_codeBlock, JITType::DFGJIT), ":\n"); 539 out.print(prefix, " Fixpoint state: ", m_fixpointState, "; Form: ", m_form, "; Unification state: ", m_unificationState, "; Ref count state: ", m_refCountState, "\n"); 530 540 if (m_form == SSA) { 531 541 for (unsigned entrypointIndex = 0; entrypointIndex < m_argumentFormats.size(); ++entrypointIndex) 532 out.print( " Argument formats for entrypoint index: ", entrypointIndex, " : ", listDump(m_argumentFormats[entrypointIndex]), "\n");542 out.print(prefix, " Argument formats for entrypoint index: ", entrypointIndex, " : ", listDump(m_argumentFormats[entrypointIndex]), "\n"); 533 543 } 534 544 else { 535 545 for (auto pair : m_rootToArguments) 536 out.print( " Arguments for block#", pair.key->index, ": ", listDump(pair.value), "\n");546 out.print(prefix, " Arguments for block#", pair.key->index, ": ", listDump(pair.value), "\n"); 537 547 } 538 548 out.print("\n"); … … 543 553 if (!block) 544 554 continue; 545 dumpBlockHeader(out, "", block, DumpAllPhis, context); 546 out.print(" States: ", block->cfaStructureClobberStateAtHead); 555 prefix.blockIndex = block->index; 556 dumpBlockHeader(out, Prefix::noString, block, DumpAllPhis, context); 557 out.print(prefix, " States: ", block->cfaStructureClobberStateAtHead); 547 558 if (!block->cfaHasVisited) 548 559 out.print(", CurrentlyCFAUnreachable"); … … 553 564 case LoadStore: 554 565 case ThreadedCPS: { 555 out.print( " Vars Before: ");566 out.print(prefix, " Vars Before: "); 556 567 if (block->cfaHasVisited) 557 568 out.print(inContext(block->valuesAtHead, context)); … … 559 570 out.print("<empty>"); 560 571 out.print("\n"); 561 out.print( " Intersected Vars Before: ");572 out.print(prefix, " Intersected Vars Before: "); 562 573 if (block->intersectionOfCFAHasVisited) 563 574 out.print(inContext(block->intersectionOfPastValuesAtHead, context)); … … 565 576 out.print("<empty>"); 566 577 out.print("\n"); 567 out.print( " Var Links: ", block->variablesAtHead, "\n");578 out.print(prefix, " Var Links: ", block->variablesAtHead, "\n"); 568 579 break; 569 580 } … … 572 583 RELEASE_ASSERT(block->ssa); 573 584 if (dumpOSRAvailabilityData) 574 out.print( " Availability: ", block->ssa->availabilityAtHead, "\n");575 out.print( " Live: ", nodeListDump(block->ssa->liveAtHead), "\n");576 out.print( " Values: ", nodeValuePairListDump(block->ssa->valuesAtHead, context), "\n");585 out.print(prefix, " Availability: ", block->ssa->availabilityAtHead, "\n"); 586 out.print(prefix, " Live: ", nodeListDump(block->ssa->liveAtHead), "\n"); 587 out.print(prefix, " Values: ", nodeValuePairListDump(block->ssa->valuesAtHead, context), "\n"); 577 588 break; 578 589 } } 579 590 for (size_t i = 0; i < block->size(); ++i) { 580 dumpCodeOrigin(out, "", lastNode, block->at(i), context); 581 dump(out, "", block->at(i), context); 582 } 583 out.print(" States: ", block->cfaBranchDirection, ", ", block->cfaStructureClobberStateAtTail); 591 prefix.clearNodeIndex(); 592 dumpCodeOrigin(out, Prefix::noString, lastNode, block->at(i), context); 593 prefix.nodeIndex = i; 594 dump(out, Prefix::noString, block->at(i), context); 595 } 596 prefix.clearNodeIndex(); 597 out.print(prefix, " States: ", block->cfaBranchDirection, ", ", block->cfaStructureClobberStateAtTail); 584 598 if (!block->cfaDidFinish) 585 599 out.print(", CFAInvalidated"); … … 588 602 case LoadStore: 589 603 case ThreadedCPS: { 590 out.print( " Vars After: ");604 out.print(prefix, " Vars After: "); 591 605 if (block->cfaHasVisited) 592 606 out.print(inContext(block->valuesAtTail, context)); … … 594 608 out.print("<empty>"); 595 609 out.print("\n"); 596 out.print( " Var Links: ", block->variablesAtTail, "\n");610 out.print(prefix, " Var Links: ", block->variablesAtTail, "\n"); 597 611 break; 598 612 } … … 601 615 RELEASE_ASSERT(block->ssa); 602 616 if (dumpOSRAvailabilityData) 603 out.print( " Availability: ", block->ssa->availabilityAtTail, "\n");604 out.print( " Live: ", nodeListDump(block->ssa->liveAtTail), "\n");605 out.print( " Values: ", nodeValuePairListDump(block->ssa->valuesAtTail, context), "\n");617 out.print(prefix, " Availability: ", block->ssa->availabilityAtTail, "\n"); 618 out.print(prefix, " Live: ", nodeListDump(block->ssa->liveAtTail), "\n"); 619 out.print(prefix, " Values: ", nodeValuePairListDump(block->ssa->valuesAtTail, context), "\n"); 606 620 break; 607 621 } } 608 622 out.print("\n"); 609 623 } 610 611 out.print("GC Values:\n"); 624 prefix.clearBlockIndex(); 625 626 out.print(prefix, "GC Values:\n"); 612 627 for (FrozenValue* value : m_frozenValues) { 613 628 if (value->pointsToHeap()) 614 out.print( " ", inContext(*value, &myContext), "\n");629 out.print(prefix, " ", inContext(*value, &myContext), "\n"); 615 630 } 616 631 … … 618 633 619 634 if (!myContext.isEmpty()) { 620 myContext.dump(out); 635 StringPrintStream prefixStr; 636 prefixStr.print(prefix); 637 myContext.dump(out, prefixStr.toCString().data()); 621 638 out.print("\n"); 622 639 } … … 1788 1805 } 1789 1806 1807 void Prefix::dump(PrintStream& out) const 1808 { 1809 if (!m_enabled) 1810 return; 1811 1812 if (!noHeader) { 1813 if (nodeIndex >= 0) 1814 out.printf("%3d ", nodeIndex); 1815 else 1816 out.printf(" "); 1817 1818 if (blockIndex >= 0) 1819 out.printf("%2d ", blockIndex); 1820 else 1821 out.printf(" "); 1822 1823 if (phaseNumber >= 0) 1824 out.printf("%2d: ", phaseNumber); 1825 else 1826 out.printf(" : "); 1827 } 1828 if (prefixStr) 1829 out.printf("%s", prefixStr); 1830 } 1831 1790 1832 } } // namespace JSC::DFG 1791 1833 -
trunk/Source/JavaScriptCore/dfg/DFGGraph.h
r244193 r248642 1 1 /* 2 * Copyright (C) 2011-201 8Apple Inc. All rights reserved.2 * Copyright (C) 2011-2019 Apple Inc. All rights reserved. 3 3 * 4 4 * Redistribution and use in source and binary forms, with or without … … 120 120 }; 121 121 122 struct Prefix { 123 enum NoHeaderTag { NoHeader }; 124 125 Prefix() { } 126 127 Prefix(const char* prefixStr, NoHeaderTag tag = NoHeader) 128 : prefixStr(prefixStr) 129 , noHeader(tag == NoHeader) 130 { } 131 132 Prefix(NoHeaderTag) 133 : noHeader(true) 134 { } 135 136 void dump(PrintStream& out) const; 137 138 void clearBlockIndex() { blockIndex = -1; } 139 void clearNodeIndex() { nodeIndex = -1; } 140 141 void enable() { m_enabled = true; } 142 void disable() { m_enabled = false; } 143 144 int32_t phaseNumber { -1 }; 145 int32_t blockIndex { -1 }; 146 int32_t nodeIndex { -1 }; 147 const char* prefixStr { nullptr }; 148 bool noHeader { false }; 149 150 static constexpr const char* noString = nullptr; 151 152 private: 153 bool m_enabled { true }; 154 }; 155 122 156 // 123 157 // === Graph === … … 223 257 // CodeBlock is optional, but may allow additional information to be dumped (e.g. Identifier names). 224 258 void dump(PrintStream& = WTF::dataFile(), DumpContext* = 0); 225 259 226 260 bool terminalsAreValid(); 227 261 … … 969 1003 return result; 970 1004 } 1005 1006 Prefix& prefix() { return m_prefix; } 1007 void nextPhase() { m_prefix.phaseNumber++; } 971 1008 972 1009 VM& m_vm; … … 1120 1157 B3::SparseCollection<Node> m_nodes; 1121 1158 SegmentedVector<RegisteredStructureSet, 16> m_structureSets; 1159 Prefix m_prefix; 1122 1160 }; 1123 1161 -
trunk/Source/JavaScriptCore/dfg/DFGPhase.cpp
r234178 r248642 50 50 return; 51 51 52 dataLog( "Beginning DFG phase ", m_name, ".\n");53 dataLog( "Before ", m_name, ":\n");52 dataLog(m_graph.prefix(), "Beginning DFG phase ", m_name, ".\n"); 53 dataLog(m_graph.prefix(), "Before ", m_name, ":\n"); 54 54 m_graph.dump(); 55 55 } -
trunk/Source/JavaScriptCore/dfg/DFGPhase.h
r234178 r248642 54 54 // Each phase must have a run() method. 55 55 56 Prefix prefix; 57 56 58 protected: 57 59 // Things you need to have a DFG compiler phase. … … 83 85 84 86 if (result && logCompilationChanges(phase.graph().m_plan.mode())) 85 dataLog F("Phase %s changed the IR.\n", phase.name());87 dataLogLn(phase.graph().prefix(), "Phase ", phase.name(), " changed the IR.\n"); 86 88 return result; 87 89 } -
trunk/Source/JavaScriptCore/dfg/DFGPlan.cpp
r248533 r248642 265 265 return CancelPath; \ 266 266 } \ 267 dfg.nextPhase(); \ 267 268 changed |= phase(dfg); \ 268 269 } while (false); \ … … 482 483 } 483 484 485 dfg.nextPhase(); 484 486 dumpAndVerifyGraph(dfg, "Graph just before FTL lowering:", shouldDumpDisassembly(m_mode)); 485 487 … … 492 494 return CancelPath; 493 495 496 dfg.nextPhase(); 494 497 FTL::State state(dfg); 495 498 FTL::lowerDFGToB3(state); 496 499 497 500 if (UNLIKELY(computeCompileTimes())) 498 501 m_timeBeforeFTL = MonotonicTime::now(); -
trunk/Source/JavaScriptCore/dfg/DFGValueRepReductionPhase.cpp
r243744 r248642 144 144 dataLogLn(description); 145 145 dataLog(" candidate: "); 146 m_graph.dump(WTF::dataFile(), "", candidate);146 m_graph.dump(WTF::dataFile(), Prefix::noString, candidate); 147 147 dataLog(" reason: "); 148 m_graph.dump(WTF::dataFile(), "", node);148 m_graph.dump(WTF::dataFile(), Prefix::noString, node); 149 149 dataLogLn(); 150 150 };
Note:
See TracChangeset
for help on using the changeset viewer.