123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275 |
- #pragma once
- #ifdef __GNUC__
- #pragma GCC diagnostic push
- #pragma GCC diagnostic ignored "-Wunused-parameter"
- #endif
- //===- IntervalIterator.h - Interval Iterator Declaration -------*- C++ -*-===//
- //
- // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
- // See https://llvm.org/LICENSE.txt for license information.
- // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
- //
- //===----------------------------------------------------------------------===//
- //
- // This file defines an iterator that enumerates the intervals in a control flow
- // graph of some sort. This iterator is parametric, allowing iterator over the
- // following types of graphs:
- //
- // 1. A Function* object, composed of BasicBlock nodes.
- // 2. An IntervalPartition& object, composed of Interval nodes.
- //
- // This iterator is defined to walk the control flow graph, returning intervals
- // in depth first order. These intervals are completely filled in except for
- // the predecessor fields (the successor information is filled in however).
- //
- // By default, the intervals created by this iterator are deleted after they
- // are no longer any use to the iterator. This behavior can be changed by
- // passing a false value into the intervals_begin() function. This causes the
- // IOwnMem member to be set, and the intervals to not be deleted.
- //
- // It is only safe to use this if all of the intervals are deleted by the caller
- // and all of the intervals are processed. However, the user of the iterator is
- // not allowed to modify or delete the intervals until after the iterator has
- // been used completely. The IntervalPartition class uses this functionality.
- //
- //===----------------------------------------------------------------------===//
- #ifndef LLVM_ANALYSIS_INTERVALITERATOR_H
- #define LLVM_ANALYSIS_INTERVALITERATOR_H
- #include "llvm/ADT/GraphTraits.h"
- #include "llvm/Analysis/Interval.h"
- #include "llvm/Analysis/IntervalPartition.h"
- #include "llvm/IR/CFG.h"
- #include <algorithm>
- #include <cassert>
- #include <iterator>
- #include <set>
- #include <utility>
- #include <vector>
- namespace llvm {
- class BasicBlock;
- class Function;
- // getNodeHeader - Given a source graph node and the source graph, return the
- // BasicBlock that is the header node. This is the opposite of
- // getSourceGraphNode.
- inline BasicBlock *getNodeHeader(BasicBlock *BB) { return BB; }
- inline BasicBlock *getNodeHeader(Interval *I) { return I->getHeaderNode(); }
- // getSourceGraphNode - Given a BasicBlock and the source graph, return the
- // source graph node that corresponds to the BasicBlock. This is the opposite
- // of getNodeHeader.
- inline BasicBlock *getSourceGraphNode(Function *, BasicBlock *BB) {
- return BB;
- }
- inline Interval *getSourceGraphNode(IntervalPartition *IP, BasicBlock *BB) {
- return IP->getBlockInterval(BB);
- }
- // addNodeToInterval - This method exists to assist the generic ProcessNode
- // with the task of adding a node to the new interval, depending on the
- // type of the source node. In the case of a CFG source graph (BasicBlock
- // case), the BasicBlock itself is added to the interval.
- inline void addNodeToInterval(Interval *Int, BasicBlock *BB) {
- Int->Nodes.push_back(BB);
- }
- // addNodeToInterval - This method exists to assist the generic ProcessNode
- // with the task of adding a node to the new interval, depending on the
- // type of the source node. In the case of a CFG source graph (BasicBlock
- // case), the BasicBlock itself is added to the interval. In the case of
- // an IntervalPartition source graph (Interval case), all of the member
- // BasicBlocks are added to the interval.
- inline void addNodeToInterval(Interval *Int, Interval *I) {
- // Add all of the nodes in I as new nodes in Int.
- llvm::append_range(Int->Nodes, I->Nodes);
- }
- template<class NodeTy, class OrigContainer_t, class GT = GraphTraits<NodeTy *>,
- class IGT = GraphTraits<Inverse<NodeTy *>>>
- class IntervalIterator {
- std::vector<std::pair<Interval *, typename Interval::succ_iterator>> IntStack;
- std::set<BasicBlock *> Visited;
- OrigContainer_t *OrigContainer;
- bool IOwnMem; // If True, delete intervals when done with them
- // See file header for conditions of use
- public:
- using iterator_category = std::forward_iterator_tag;
- IntervalIterator() = default; // End iterator, empty stack
- IntervalIterator(Function *M, bool OwnMemory) : IOwnMem(OwnMemory) {
- OrigContainer = M;
- if (!ProcessInterval(&M->front())) {
- llvm_unreachable("ProcessInterval should never fail for first interval!");
- }
- }
- IntervalIterator(IntervalIterator &&x)
- : IntStack(std::move(x.IntStack)), Visited(std::move(x.Visited)),
- OrigContainer(x.OrigContainer), IOwnMem(x.IOwnMem) {
- x.IOwnMem = false;
- }
- IntervalIterator(IntervalPartition &IP, bool OwnMemory) : IOwnMem(OwnMemory) {
- OrigContainer = &IP;
- if (!ProcessInterval(IP.getRootInterval())) {
- llvm_unreachable("ProcessInterval should never fail for first interval!");
- }
- }
- ~IntervalIterator() {
- if (IOwnMem)
- while (!IntStack.empty()) {
- delete operator*();
- IntStack.pop_back();
- }
- }
- bool operator==(const IntervalIterator &x) const {
- return IntStack == x.IntStack;
- }
- bool operator!=(const IntervalIterator &x) const { return !(*this == x); }
- const Interval *operator*() const { return IntStack.back().first; }
- Interval *operator*() { return IntStack.back().first; }
- const Interval *operator->() const { return operator*(); }
- Interval *operator->() { return operator*(); }
- IntervalIterator &operator++() { // Preincrement
- assert(!IntStack.empty() && "Attempting to use interval iterator at end!");
- do {
- // All of the intervals on the stack have been visited. Try visiting
- // their successors now.
- Interval::succ_iterator &SuccIt = IntStack.back().second,
- EndIt = succ_end(IntStack.back().first);
- while (SuccIt != EndIt) { // Loop over all interval succs
- bool Done = ProcessInterval(getSourceGraphNode(OrigContainer, *SuccIt));
- ++SuccIt; // Increment iterator
- if (Done) return *this; // Found a new interval! Use it!
- }
- // Free interval memory... if necessary
- if (IOwnMem) delete IntStack.back().first;
- // We ran out of successors for this interval... pop off the stack
- IntStack.pop_back();
- } while (!IntStack.empty());
- return *this;
- }
- IntervalIterator operator++(int) { // Postincrement
- IntervalIterator tmp = *this;
- ++*this;
- return tmp;
- }
- private:
- // ProcessInterval - This method is used during the construction of the
- // interval graph. It walks through the source graph, recursively creating
- // an interval per invocation until the entire graph is covered. This uses
- // the ProcessNode method to add all of the nodes to the interval.
- //
- // This method is templated because it may operate on two different source
- // graphs: a basic block graph, or a preexisting interval graph.
- bool ProcessInterval(NodeTy *Node) {
- BasicBlock *Header = getNodeHeader(Node);
- if (!Visited.insert(Header).second)
- return false;
- Interval *Int = new Interval(Header);
- // Check all of our successors to see if they are in the interval...
- for (typename GT::ChildIteratorType I = GT::child_begin(Node),
- E = GT::child_end(Node); I != E; ++I)
- ProcessNode(Int, getSourceGraphNode(OrigContainer, *I));
- IntStack.push_back(std::make_pair(Int, succ_begin(Int)));
- return true;
- }
- // ProcessNode - This method is called by ProcessInterval to add nodes to the
- // interval being constructed, and it is also called recursively as it walks
- // the source graph. A node is added to the current interval only if all of
- // its predecessors are already in the graph. This also takes care of keeping
- // the successor set of an interval up to date.
- //
- // This method is templated because it may operate on two different source
- // graphs: a basic block graph, or a preexisting interval graph.
- void ProcessNode(Interval *Int, NodeTy *Node) {
- assert(Int && "Null interval == bad!");
- assert(Node && "Null Node == bad!");
- BasicBlock *NodeHeader = getNodeHeader(Node);
- if (Visited.count(NodeHeader)) { // Node already been visited?
- if (Int->contains(NodeHeader)) { // Already in this interval...
- return;
- } else { // In other interval, add as successor
- if (!Int->isSuccessor(NodeHeader)) // Add only if not already in set
- Int->Successors.push_back(NodeHeader);
- }
- } else { // Otherwise, not in interval yet
- for (typename IGT::ChildIteratorType I = IGT::child_begin(Node),
- E = IGT::child_end(Node); I != E; ++I) {
- if (!Int->contains(*I)) { // If pred not in interval, we can't be
- if (!Int->isSuccessor(NodeHeader)) // Add only if not already in set
- Int->Successors.push_back(NodeHeader);
- return; // See you later
- }
- }
- // If we get here, then all of the predecessors of BB are in the interval
- // already. In this case, we must add BB to the interval!
- addNodeToInterval(Int, Node);
- Visited.insert(NodeHeader); // The node has now been visited!
- if (Int->isSuccessor(NodeHeader)) {
- // If we were in the successor list from before... remove from succ list
- llvm::erase_value(Int->Successors, NodeHeader);
- }
- // Now that we have discovered that Node is in the interval, perhaps some
- // of its successors are as well?
- for (typename GT::ChildIteratorType It = GT::child_begin(Node),
- End = GT::child_end(Node); It != End; ++It)
- ProcessNode(Int, getSourceGraphNode(OrigContainer, *It));
- }
- }
- };
- using function_interval_iterator = IntervalIterator<BasicBlock, Function>;
- using interval_part_interval_iterator =
- IntervalIterator<Interval, IntervalPartition>;
- inline function_interval_iterator intervals_begin(Function *F,
- bool DeleteInts = true) {
- return function_interval_iterator(F, DeleteInts);
- }
- inline function_interval_iterator intervals_end(Function *) {
- return function_interval_iterator();
- }
- inline interval_part_interval_iterator
- intervals_begin(IntervalPartition &IP, bool DeleteIntervals = true) {
- return interval_part_interval_iterator(IP, DeleteIntervals);
- }
- inline interval_part_interval_iterator intervals_end(IntervalPartition &IP) {
- return interval_part_interval_iterator();
- }
- } // end namespace llvm
- #endif // LLVM_ANALYSIS_INTERVALITERATOR_H
- #ifdef __GNUC__
- #pragma GCC diagnostic pop
- #endif
|