289 lines
11 KiB
C
289 lines
11 KiB
C
|
//===- llvm/CodeGen/MachineInstrBundleIterator.h ----------------*- 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
|
||
|
//
|
||
|
//===----------------------------------------------------------------------===//
|
||
|
//
|
||
|
// Defines an iterator class that bundles MachineInstr.
|
||
|
//
|
||
|
//===----------------------------------------------------------------------===//
|
||
|
|
||
|
#ifndef LLVM_CODEGEN_MACHINEINSTRBUNDLEITERATOR_H
|
||
|
#define LLVM_CODEGEN_MACHINEINSTRBUNDLEITERATOR_H
|
||
|
|
||
|
#include "llvm/ADT/ilist.h"
|
||
|
#include "llvm/ADT/simple_ilist.h"
|
||
|
#include <cassert>
|
||
|
#include <iterator>
|
||
|
#include <type_traits>
|
||
|
|
||
|
namespace llvm {
|
||
|
|
||
|
template <class T, bool IsReverse> struct MachineInstrBundleIteratorTraits;
|
||
|
template <class T> struct MachineInstrBundleIteratorTraits<T, false> {
|
||
|
using list_type = simple_ilist<T, ilist_sentinel_tracking<true>>;
|
||
|
using instr_iterator = typename list_type::iterator;
|
||
|
using nonconst_instr_iterator = typename list_type::iterator;
|
||
|
using const_instr_iterator = typename list_type::const_iterator;
|
||
|
};
|
||
|
template <class T> struct MachineInstrBundleIteratorTraits<T, true> {
|
||
|
using list_type = simple_ilist<T, ilist_sentinel_tracking<true>>;
|
||
|
using instr_iterator = typename list_type::reverse_iterator;
|
||
|
using nonconst_instr_iterator = typename list_type::reverse_iterator;
|
||
|
using const_instr_iterator = typename list_type::const_reverse_iterator;
|
||
|
};
|
||
|
template <class T> struct MachineInstrBundleIteratorTraits<const T, false> {
|
||
|
using list_type = simple_ilist<T, ilist_sentinel_tracking<true>>;
|
||
|
using instr_iterator = typename list_type::const_iterator;
|
||
|
using nonconst_instr_iterator = typename list_type::iterator;
|
||
|
using const_instr_iterator = typename list_type::const_iterator;
|
||
|
};
|
||
|
template <class T> struct MachineInstrBundleIteratorTraits<const T, true> {
|
||
|
using list_type = simple_ilist<T, ilist_sentinel_tracking<true>>;
|
||
|
using instr_iterator = typename list_type::const_reverse_iterator;
|
||
|
using nonconst_instr_iterator = typename list_type::reverse_iterator;
|
||
|
using const_instr_iterator = typename list_type::const_reverse_iterator;
|
||
|
};
|
||
|
|
||
|
template <bool IsReverse> struct MachineInstrBundleIteratorHelper;
|
||
|
template <> struct MachineInstrBundleIteratorHelper<false> {
|
||
|
/// Get the beginning of the current bundle.
|
||
|
template <class Iterator> static Iterator getBundleBegin(Iterator I) {
|
||
|
if (!I.isEnd())
|
||
|
while (I->isBundledWithPred())
|
||
|
--I;
|
||
|
return I;
|
||
|
}
|
||
|
|
||
|
/// Get the final node of the current bundle.
|
||
|
template <class Iterator> static Iterator getBundleFinal(Iterator I) {
|
||
|
if (!I.isEnd())
|
||
|
while (I->isBundledWithSucc())
|
||
|
++I;
|
||
|
return I;
|
||
|
}
|
||
|
|
||
|
/// Increment forward ilist iterator.
|
||
|
template <class Iterator> static void increment(Iterator &I) {
|
||
|
I = std::next(getBundleFinal(I));
|
||
|
}
|
||
|
|
||
|
/// Decrement forward ilist iterator.
|
||
|
template <class Iterator> static void decrement(Iterator &I) {
|
||
|
I = getBundleBegin(std::prev(I));
|
||
|
}
|
||
|
};
|
||
|
|
||
|
template <> struct MachineInstrBundleIteratorHelper<true> {
|
||
|
/// Get the beginning of the current bundle.
|
||
|
template <class Iterator> static Iterator getBundleBegin(Iterator I) {
|
||
|
return MachineInstrBundleIteratorHelper<false>::getBundleBegin(
|
||
|
I.getReverse())
|
||
|
.getReverse();
|
||
|
}
|
||
|
|
||
|
/// Get the final node of the current bundle.
|
||
|
template <class Iterator> static Iterator getBundleFinal(Iterator I) {
|
||
|
return MachineInstrBundleIteratorHelper<false>::getBundleFinal(
|
||
|
I.getReverse())
|
||
|
.getReverse();
|
||
|
}
|
||
|
|
||
|
/// Increment reverse ilist iterator.
|
||
|
template <class Iterator> static void increment(Iterator &I) {
|
||
|
I = getBundleBegin(std::next(I));
|
||
|
}
|
||
|
|
||
|
/// Decrement reverse ilist iterator.
|
||
|
template <class Iterator> static void decrement(Iterator &I) {
|
||
|
I = std::prev(getBundleFinal(I));
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/// MachineBasicBlock iterator that automatically skips over MIs that are
|
||
|
/// inside bundles (i.e. walk top level MIs only).
|
||
|
template <typename Ty, bool IsReverse = false>
|
||
|
class MachineInstrBundleIterator : MachineInstrBundleIteratorHelper<IsReverse> {
|
||
|
using Traits = MachineInstrBundleIteratorTraits<Ty, IsReverse>;
|
||
|
using instr_iterator = typename Traits::instr_iterator;
|
||
|
|
||
|
instr_iterator MII;
|
||
|
|
||
|
public:
|
||
|
using value_type = typename instr_iterator::value_type;
|
||
|
using difference_type = typename instr_iterator::difference_type;
|
||
|
using pointer = typename instr_iterator::pointer;
|
||
|
using reference = typename instr_iterator::reference;
|
||
|
using const_pointer = typename instr_iterator::const_pointer;
|
||
|
using const_reference = typename instr_iterator::const_reference;
|
||
|
using iterator_category = std::bidirectional_iterator_tag;
|
||
|
|
||
|
private:
|
||
|
using nonconst_instr_iterator = typename Traits::nonconst_instr_iterator;
|
||
|
using const_instr_iterator = typename Traits::const_instr_iterator;
|
||
|
using nonconst_iterator =
|
||
|
MachineInstrBundleIterator<typename nonconst_instr_iterator::value_type,
|
||
|
IsReverse>;
|
||
|
using reverse_iterator = MachineInstrBundleIterator<Ty, !IsReverse>;
|
||
|
|
||
|
public:
|
||
|
MachineInstrBundleIterator(instr_iterator MI) : MII(MI) {
|
||
|
assert((!MI.getNodePtr() || MI.isEnd() || !MI->isBundledWithPred()) &&
|
||
|
"It's not legal to initialize MachineInstrBundleIterator with a "
|
||
|
"bundled MI");
|
||
|
}
|
||
|
|
||
|
MachineInstrBundleIterator(reference MI) : MII(MI) {
|
||
|
assert(!MI.isBundledWithPred() && "It's not legal to initialize "
|
||
|
"MachineInstrBundleIterator with a "
|
||
|
"bundled MI");
|
||
|
}
|
||
|
|
||
|
MachineInstrBundleIterator(pointer MI) : MII(MI) {
|
||
|
// FIXME: This conversion should be explicit.
|
||
|
assert((!MI || !MI->isBundledWithPred()) && "It's not legal to initialize "
|
||
|
"MachineInstrBundleIterator "
|
||
|
"with a bundled MI");
|
||
|
}
|
||
|
|
||
|
// Template allows conversion from const to nonconst.
|
||
|
template <class OtherTy>
|
||
|
MachineInstrBundleIterator(
|
||
|
const MachineInstrBundleIterator<OtherTy, IsReverse> &I,
|
||
|
std::enable_if_t<std::is_convertible<OtherTy *, Ty *>::value, void *> =
|
||
|
nullptr)
|
||
|
: MII(I.getInstrIterator()) {}
|
||
|
|
||
|
MachineInstrBundleIterator() : MII(nullptr) {}
|
||
|
|
||
|
/// Explicit conversion between forward/reverse iterators.
|
||
|
///
|
||
|
/// Translate between forward and reverse iterators without changing range
|
||
|
/// boundaries. The resulting iterator will dereference (and have a handle)
|
||
|
/// to the previous node, which is somewhat unexpected; but converting the
|
||
|
/// two endpoints in a range will give the same range in reverse.
|
||
|
///
|
||
|
/// This matches std::reverse_iterator conversions.
|
||
|
explicit MachineInstrBundleIterator(
|
||
|
const MachineInstrBundleIterator<Ty, !IsReverse> &I)
|
||
|
: MachineInstrBundleIterator(++I.getReverse()) {}
|
||
|
|
||
|
/// Get the bundle iterator for the given instruction's bundle.
|
||
|
static MachineInstrBundleIterator getAtBundleBegin(instr_iterator MI) {
|
||
|
return MachineInstrBundleIteratorHelper<IsReverse>::getBundleBegin(MI);
|
||
|
}
|
||
|
|
||
|
reference operator*() const { return *MII; }
|
||
|
pointer operator->() const { return &operator*(); }
|
||
|
|
||
|
/// Check for null.
|
||
|
bool isValid() const { return MII.getNodePtr(); }
|
||
|
|
||
|
friend bool operator==(const MachineInstrBundleIterator &L,
|
||
|
const MachineInstrBundleIterator &R) {
|
||
|
return L.MII == R.MII;
|
||
|
}
|
||
|
friend bool operator==(const MachineInstrBundleIterator &L,
|
||
|
const const_instr_iterator &R) {
|
||
|
return L.MII == R; // Avoid assertion about validity of R.
|
||
|
}
|
||
|
friend bool operator==(const const_instr_iterator &L,
|
||
|
const MachineInstrBundleIterator &R) {
|
||
|
return L == R.MII; // Avoid assertion about validity of L.
|
||
|
}
|
||
|
friend bool operator==(const MachineInstrBundleIterator &L,
|
||
|
const nonconst_instr_iterator &R) {
|
||
|
return L.MII == R; // Avoid assertion about validity of R.
|
||
|
}
|
||
|
friend bool operator==(const nonconst_instr_iterator &L,
|
||
|
const MachineInstrBundleIterator &R) {
|
||
|
return L == R.MII; // Avoid assertion about validity of L.
|
||
|
}
|
||
|
friend bool operator==(const MachineInstrBundleIterator &L, const_pointer R) {
|
||
|
return L == const_instr_iterator(R); // Avoid assertion about validity of R.
|
||
|
}
|
||
|
friend bool operator==(const_pointer L, const MachineInstrBundleIterator &R) {
|
||
|
return const_instr_iterator(L) == R; // Avoid assertion about validity of L.
|
||
|
}
|
||
|
friend bool operator==(const MachineInstrBundleIterator &L,
|
||
|
const_reference R) {
|
||
|
return L == &R; // Avoid assertion about validity of R.
|
||
|
}
|
||
|
friend bool operator==(const_reference L,
|
||
|
const MachineInstrBundleIterator &R) {
|
||
|
return &L == R; // Avoid assertion about validity of L.
|
||
|
}
|
||
|
|
||
|
friend bool operator!=(const MachineInstrBundleIterator &L,
|
||
|
const MachineInstrBundleIterator &R) {
|
||
|
return !(L == R);
|
||
|
}
|
||
|
friend bool operator!=(const MachineInstrBundleIterator &L,
|
||
|
const const_instr_iterator &R) {
|
||
|
return !(L == R);
|
||
|
}
|
||
|
friend bool operator!=(const const_instr_iterator &L,
|
||
|
const MachineInstrBundleIterator &R) {
|
||
|
return !(L == R);
|
||
|
}
|
||
|
friend bool operator!=(const MachineInstrBundleIterator &L,
|
||
|
const nonconst_instr_iterator &R) {
|
||
|
return !(L == R);
|
||
|
}
|
||
|
friend bool operator!=(const nonconst_instr_iterator &L,
|
||
|
const MachineInstrBundleIterator &R) {
|
||
|
return !(L == R);
|
||
|
}
|
||
|
friend bool operator!=(const MachineInstrBundleIterator &L, const_pointer R) {
|
||
|
return !(L == R);
|
||
|
}
|
||
|
friend bool operator!=(const_pointer L, const MachineInstrBundleIterator &R) {
|
||
|
return !(L == R);
|
||
|
}
|
||
|
friend bool operator!=(const MachineInstrBundleIterator &L,
|
||
|
const_reference R) {
|
||
|
return !(L == R);
|
||
|
}
|
||
|
friend bool operator!=(const_reference L,
|
||
|
const MachineInstrBundleIterator &R) {
|
||
|
return !(L == R);
|
||
|
}
|
||
|
|
||
|
// Increment and decrement operators...
|
||
|
MachineInstrBundleIterator &operator--() {
|
||
|
this->decrement(MII);
|
||
|
return *this;
|
||
|
}
|
||
|
MachineInstrBundleIterator &operator++() {
|
||
|
this->increment(MII);
|
||
|
return *this;
|
||
|
}
|
||
|
MachineInstrBundleIterator operator--(int) {
|
||
|
MachineInstrBundleIterator Temp = *this;
|
||
|
--*this;
|
||
|
return Temp;
|
||
|
}
|
||
|
MachineInstrBundleIterator operator++(int) {
|
||
|
MachineInstrBundleIterator Temp = *this;
|
||
|
++*this;
|
||
|
return Temp;
|
||
|
}
|
||
|
|
||
|
instr_iterator getInstrIterator() const { return MII; }
|
||
|
|
||
|
nonconst_iterator getNonConstIterator() const { return MII.getNonConst(); }
|
||
|
|
||
|
/// Get a reverse iterator to the same node.
|
||
|
///
|
||
|
/// Gives a reverse iterator that will dereference (and have a handle) to the
|
||
|
/// same node. Converting the endpoint iterators in a range will give a
|
||
|
/// different range; for range operations, use the explicit conversions.
|
||
|
reverse_iterator getReverse() const { return MII.getReverse(); }
|
||
|
};
|
||
|
|
||
|
} // end namespace llvm
|
||
|
|
||
|
#endif // LLVM_CODEGEN_MACHINEINSTRBUNDLEITERATOR_H
|