| 1 | // Licensed to the .NET Foundation under one or more agreements. |
| 2 | // The .NET Foundation licenses this file to you under the MIT license. |
| 3 | // See the LICENSE file in the project root for more information. |
| 4 | |
| 5 | #pragma once |
| 6 | #pragma warning(disable : 4503) // 'identifier' : decorated name length exceeded, name was truncated |
| 7 | |
| 8 | #undef SSA_FEATURE_DOMARR |
| 9 | |
| 10 | #include "compiler.h" |
| 11 | |
| 12 | struct SsaRenameState; |
| 13 | |
| 14 | typedef int LclVarNum; |
| 15 | |
| 16 | // Pair of a local var name eg: V01 and Ssa number; eg: V01_01 |
| 17 | typedef jitstd::pair<LclVarNum, int> SsaVarName; |
| 18 | |
| 19 | class SsaBuilder |
| 20 | { |
| 21 | private: |
| 22 | inline void EndPhase(Phases phase) |
| 23 | { |
| 24 | m_pCompiler->EndPhase(phase); |
| 25 | } |
| 26 | |
| 27 | bool IncludeInSsa(unsigned lclNum); |
| 28 | |
| 29 | public: |
| 30 | // Constructor |
| 31 | SsaBuilder(Compiler* pCompiler); |
| 32 | |
| 33 | // Requires stmt nodes to be already sequenced in evaluation order. Analyzes the graph |
| 34 | // for introduction of phi-nodes as GT_PHI tree nodes at the beginning of each block. |
| 35 | // Each GT_LCL_VAR is given its ssa number through its gtSsaNum field in the node. |
| 36 | // Each GT_PHI node will have gtOp1 set to lhs of the phi node and the gtOp2 to be a |
| 37 | // GT_LIST of GT_PHI_ARG. Each use or def is denoted by the corresponding GT_LCL_VAR |
| 38 | // tree. For example, to get all uses of a particular variable fully defined by its |
| 39 | // lclNum and ssaNum, one would use m_uses and look up all the uses. Similarly, a single |
| 40 | // def of an SSA variable can be looked up similarly using m_defs member. |
| 41 | void Build(); |
| 42 | |
| 43 | // Requires "bbIDom" of each block to be computed. Requires "domTree" to be allocated |
| 44 | // and can be updated, i.e., by adding mapping from a block to it's dominated children. |
| 45 | // Using IDom of each basic block, compute the whole domTree. If a block "b" has IDom "i", |
| 46 | // then, block "b" is dominated by "i". The mapping then is i -> { ..., b, ... }, in |
| 47 | // other words, "domTree" is a tree represented by nodes mapped to their children. |
| 48 | static void ComputeDominators(Compiler* pCompiler, BlkToBlkVectorMap* domTree); |
| 49 | |
| 50 | private: |
| 51 | // Ensures that the basic block graph has a root for the dominator graph, by ensuring |
| 52 | // that there is a first block that is not in a try region (adding an empty block for that purpose |
| 53 | // if necessary). Eventually should move to Compiler. |
| 54 | void SetupBBRoot(); |
| 55 | |
| 56 | // Requires "postOrder" to be an array of size "count". Requires "count" to at least |
| 57 | // be the size of the flow graph. Sorts the current compiler's flow-graph and places |
| 58 | // the blocks in post order (i.e., a node's children first) in the array. Returns the |
| 59 | // number of nodes visited while sorting the graph. In other words, valid entries in |
| 60 | // the output array. |
| 61 | int TopologicalSort(BasicBlock** postOrder, int count); |
| 62 | |
| 63 | // Requires "postOrder" to hold the blocks of the flowgraph in topologically sorted |
| 64 | // order. Requires count to be the valid entries in the "postOrder" array. Computes |
| 65 | // each block's immediate dominator and records it in the BasicBlock in bbIDom. |
| 66 | void ComputeImmediateDom(BasicBlock** postOrder, int count); |
| 67 | |
| 68 | #ifdef SSA_FEATURE_DOMARR |
| 69 | // Requires "curBlock" to be the first basic block at the first step of the recursion. |
| 70 | // Requires "domTree" to be a adjacency list (actually, a set of blocks with a set of blocks |
| 71 | // as children.) Requires "preIndex" and "postIndex" to be initialized to 0 at entry into recursion. |
| 72 | // Computes arrays "m_pDomPreOrder" and "m_pDomPostOrder" of block indices such that the blocks of a |
| 73 | // "domTree" are in pre and postorder respectively. |
| 74 | void DomTreeWalk(BasicBlock* curBlock, BlkToBlkVectorMap* domTree, int* preIndex, int* postIndex); |
| 75 | #endif |
| 76 | |
| 77 | // Requires all blocks to have computed "bbIDom." Requires "domTree" to be a preallocated BlkToBlkVectorMap. |
| 78 | // Helper to compute "domTree" from the pre-computed bbIDom of the basic blocks. |
| 79 | static void ConstructDomTreeForBlock(Compiler* pCompiler, BasicBlock* block, BlkToBlkVectorMap* domTree); |
| 80 | |
| 81 | // Requires "postOrder" to hold the blocks of the flowgraph in topologically sorted order. Requires |
| 82 | // count to be the valid entries in the "postOrder" array. Computes "domTree" as a adjacency list |
| 83 | // like object, i.e., a set of blocks with a set of blocks as children defining the DOM relation. |
| 84 | void ComputeDominators(BasicBlock** postOrder, int count, BlkToBlkVectorMap* domTree); |
| 85 | |
| 86 | #ifdef DEBUG |
| 87 | // Display the dominator tree. |
| 88 | static void DisplayDominators(BlkToBlkVectorMap* domTree); |
| 89 | #endif // DEBUG |
| 90 | |
| 91 | // Compute flow graph dominance frontiers. |
| 92 | void ComputeDominanceFrontiers(BasicBlock** postOrder, int count, BlkToBlkVectorMap* mapDF); |
| 93 | |
| 94 | // Compute the iterated dominance frontier for the specified block. |
| 95 | void ComputeIteratedDominanceFrontier(BasicBlock* b, const BlkToBlkVectorMap* mapDF, BlkVector* bIDF); |
| 96 | |
| 97 | // Requires "postOrder" to hold the blocks of the flowgraph in topologically sorted order. Requires |
| 98 | // count to be the valid entries in the "postOrder" array. Inserts GT_PHI nodes at the beginning |
| 99 | // of basic blocks that require them like so: |
| 100 | // GT_ASG(GT_LCL_VAR, GT_PHI(GT_PHI_ARG(GT_LCL_VAR, Block*), GT_LIST(GT_PHI_ARG(GT_LCL_VAR, Block*), NULL)); |
| 101 | void InsertPhiFunctions(BasicBlock** postOrder, int count); |
| 102 | |
| 103 | // Requires "domTree" to be the dominator tree relation defined by a DOM b. |
| 104 | // Requires "pRenameState" to have counts and stacks at their initial state. |
| 105 | // Assigns gtSsaNames to all variables. |
| 106 | void RenameVariables(BlkToBlkVectorMap* domTree, SsaRenameState* pRenameState); |
| 107 | |
| 108 | // Requires "block" to be any basic block participating in variable renaming, and has at least a |
| 109 | // definition that pushed a ssa number into the rename stack for a variable. Requires "pRenameState" |
| 110 | // to have variable stacks that have counts pushed into them for the block while assigning def |
| 111 | // numbers. Pops the stack for any local variable that has an entry for block on top. |
| 112 | void BlockPopStacks(BasicBlock* block, SsaRenameState* pRenameState); |
| 113 | |
| 114 | // Requires "block" to be non-NULL; and is searched for defs and uses to assign ssa numbers. |
| 115 | // Requires "pRenameState" to be non-NULL and be currently used for variables renaming. |
| 116 | void BlockRenameVariables(BasicBlock* block, SsaRenameState* pRenameState); |
| 117 | |
| 118 | // Requires "tree" (assumed to be a statement in "block") to be searched for defs and uses to assign ssa numbers. |
| 119 | // Requires "pRenameState" to be non-NULL and be currently used for variables renaming. Assumes that "isPhiDefn" |
| 120 | // implies that any definition occurring within "tree" is a phi definition. |
| 121 | void TreeRenameVariables(GenTree* tree, BasicBlock* block, SsaRenameState* pRenameState, bool isPhiDefn); |
| 122 | |
| 123 | // Assumes that "block" contains a definition for local var "lclNum", with SSA number "count". |
| 124 | // IF "block" is within one or more try blocks, |
| 125 | // and the local variable is live at the start of the corresponding handlers, |
| 126 | // add this SSA number "count" to the argument list of the phi for the variable in the start |
| 127 | // block of those handlers. |
| 128 | void AddDefToHandlerPhis(BasicBlock* block, unsigned lclNum, unsigned count); |
| 129 | |
| 130 | // Same as above, for memory. |
| 131 | void AddMemoryDefToHandlerPhis(MemoryKind memoryKind, BasicBlock* block, unsigned count); |
| 132 | |
| 133 | // Requires "block" to be non-NULL. Requires "pRenameState" to be non-NULL and be currently used |
| 134 | // for variables renaming. Assigns the rhs arguments to the phi, i.e., block's phi node arguments. |
| 135 | void AssignPhiNodeRhsVariables(BasicBlock* block, SsaRenameState* pRenameState); |
| 136 | |
| 137 | #ifdef DEBUG |
| 138 | void Print(BasicBlock** postOrder, int count); |
| 139 | #endif |
| 140 | |
| 141 | private: |
| 142 | Compiler* m_pCompiler; |
| 143 | CompAllocator m_allocator; |
| 144 | |
| 145 | // Bit vector used by TopologicalSort and ComputeImmediateDom to track already visited blocks. |
| 146 | BitVecTraits m_visitedTraits; |
| 147 | BitVec m_visited; |
| 148 | |
| 149 | #ifdef SSA_FEATURE_DOMARR |
| 150 | // To answer queries of type a DOM b. |
| 151 | // Do not move these outside of this class, use accessors/interface methods. |
| 152 | int* m_pDomPreOrder; |
| 153 | int* m_pDomPostOrder; |
| 154 | #endif |
| 155 | }; |
| 156 | |