| //===- subzero/src/IceCfgNode.h - Control flow graph node -------*- C++ -*-===// |
| // |
| // The Subzero Code Generator |
| // |
| // This file is distributed under the University of Illinois Open Source |
| // License. See LICENSE.TXT for details. |
| // |
| //===----------------------------------------------------------------------===// |
| /// |
| /// \file |
| /// \brief Declares the CfgNode class, which represents a single basic block as |
| /// its instruction list, in-edge list, and out-edge list. |
| /// |
| //===----------------------------------------------------------------------===// |
| |
| #ifndef SUBZERO_SRC_ICECFGNODE_H |
| #define SUBZERO_SRC_ICECFGNODE_H |
| |
| #include "IceDefs.h" |
| #include "IceInst.h" // InstList traits |
| #include "IceStringPool.h" |
| |
| namespace Ice { |
| |
| class CfgNode { |
| CfgNode() = delete; |
| CfgNode(const CfgNode &) = delete; |
| CfgNode &operator=(const CfgNode &) = delete; |
| |
| public: |
| static CfgNode *create(Cfg *Func, SizeT Number) { |
| return new (Func->allocate<CfgNode>()) CfgNode(Func, Number); |
| } |
| |
| Cfg *getCfg() const { return Func; } |
| |
| /// Access the label number and name for this node. |
| SizeT getIndex() const { return Number; } |
| void resetIndex(SizeT NewNumber) { Number = NewNumber; } |
| std::string getName() const { |
| if (Name.hasStdString()) |
| return Name.toString(); |
| return "__" + std::to_string(NumberOrig); |
| } |
| void setName(const std::string &NewName) { |
| if (NewName.empty()) |
| return; |
| Name = NodeString::createWithString(Func, NewName); |
| } |
| std::string getAsmName() const { |
| return ".L" + Func->getFunctionName() + "$" + getName(); |
| } |
| |
| void incrementLoopNestDepth() { ++LoopNestDepth; } |
| void setLoopNestDepth(SizeT NewDepth) { LoopNestDepth = NewDepth; } |
| SizeT getLoopNestDepth() const { return LoopNestDepth; } |
| |
| /// The HasReturn flag indicates that this node contains a return instruction |
| /// and therefore needs an epilog. |
| void setHasReturn() { HasReturn = true; } |
| bool getHasReturn() const { return HasReturn; } |
| |
| void setNeedsPlacement(bool Value) { NeedsPlacement = Value; } |
| bool needsPlacement() const { return NeedsPlacement; } |
| |
| void setNeedsAlignment() { NeedsAlignment = true; } |
| bool needsAlignment() const { return NeedsAlignment; } |
| |
| /// \name Access predecessor and successor edge lists. |
| /// @{ |
| const NodeList &getInEdges() const { return InEdges; } |
| const NodeList &getOutEdges() const { return OutEdges; } |
| /// @} |
| |
| /// \name Manage the instruction list. |
| /// @{ |
| InstList &getInsts() { return Insts; } |
| PhiList &getPhis() { return Phis; } |
| const InstList &getInsts() const { return Insts; } |
| const PhiList &getPhis() const { return Phis; } |
| void appendInst(Inst *Instr); |
| void renumberInstructions(); |
| /// Rough and generally conservative estimate of the number of instructions in |
| /// the block. It is updated when an instruction is added, but not when |
| /// deleted. It is recomputed during renumberInstructions(). |
| InstNumberT getInstCountEstimate() const { return InstCountEstimate; } |
| /// @} |
| |
| /// \name Manage predecessors and successors. |
| /// @{ |
| |
| /// Add a predecessor edge to the InEdges list for each of this node's |
| /// successors. |
| void computePredecessors(); |
| void computeSuccessors(); |
| CfgNode *splitIncomingEdge(CfgNode *Pred, SizeT InEdgeIndex); |
| /// @} |
| |
| void enforcePhiConsistency(); |
| void placePhiLoads(); |
| void placePhiStores(); |
| void deletePhis(); |
| void advancedPhiLowering(); |
| void doAddressOpt(); |
| void doNopInsertion(RandomNumberGenerator &RNG); |
| void genCode(); |
| void livenessLightweight(); |
| bool liveness(Liveness *Liveness); |
| void livenessAddIntervals(Liveness *Liveness, InstNumberT FirstInstNum, |
| InstNumberT LastInstNum); |
| void contractIfEmpty(); |
| void doBranchOpt(const CfgNode *NextNode); |
| void emit(Cfg *Func) const; |
| void emitIAS(Cfg *Func) const; |
| void dump(Cfg *Func) const; |
| |
| void profileExecutionCount(VariableDeclaration *Var); |
| |
| void addOutEdge(CfgNode *Out) { OutEdges.push_back(Out); } |
| void addInEdge(CfgNode *In) { InEdges.push_back(In); } |
| void replaceInEdge(CfgNode *Old, CfgNode *New); |
| void removeAllOutEdges() { OutEdges.clear(); } |
| void removeInEdge(CfgNode *In); |
| |
| bool hasSingleOutEdge() const { |
| return (getOutEdges().size() == 1 || getOutEdges()[0] == getOutEdges()[1]); |
| } |
| CfgNode *shortCircuit(); |
| |
| inline void* getExternalData() const { return externalData; } |
| inline void setExternalData(void* data) { externalData = data; } |
| |
| private: |
| CfgNode(Cfg *Func, SizeT Number) |
| : Func(Func), Number(Number), NumberOrig(Number), |
| Name(NodeString::createWithoutString(Func)) {} |
| bool livenessValidateIntervals(Liveness *Liveness) const; |
| Cfg *const Func; |
| SizeT Number; /// invariant: Func->Nodes[Number]==this |
| const SizeT NumberOrig; /// used for name auto-generation |
| NodeString Name; |
| SizeT LoopNestDepth = 0; /// the loop nest depth of this node |
| bool HasReturn = false; /// does this block need an epilog? |
| bool NeedsPlacement = false; |
| bool NeedsAlignment = false; /// is sandboxing required? |
| InstNumberT InstCountEstimate = 0; /// rough instruction count estimate |
| NodeList InEdges; /// in no particular order |
| NodeList OutEdges; /// in no particular order |
| PhiList Phis; /// unordered set of phi instructions |
| InstList Insts; /// ordered list of non-phi instructions |
| |
| /// External data can be set by an optimizer to compute and retain any |
| /// information related to the current node. All the memory used to |
| /// store this information must be managed by the optimizer. |
| void* externalData = nullptr; |
| }; |
| |
| } // end of namespace Ice |
| |
| #endif // SUBZERO_SRC_ICECFGNODE_H |