You've already forked linux-packaging-mono
Imported Upstream version 5.18.0.246
Former-commit-id: 0c7ce5b1a7851e13f22acfd379b7f9fb304e4833
This commit is contained in:
parent
a7724cd563
commit
279aa8f685
218
external/llvm/lib/Target/AMDGPU/GCNRegPressure.h
vendored
Normal file
218
external/llvm/lib/Target/AMDGPU/GCNRegPressure.h
vendored
Normal file
@ -0,0 +1,218 @@
|
||||
//===- GCNRegPressure.h -----------------------------------------*- C++ -*-===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#ifndef LLVM_LIB_TARGET_AMDGPU_GCNREGPRESSURE_H
|
||||
#define LLVM_LIB_TARGET_AMDGPU_GCNREGPRESSURE_H
|
||||
|
||||
#include "AMDGPUSubtarget.h"
|
||||
#include "llvm/ADT/DenseMap.h"
|
||||
#include "llvm/CodeGen/LiveIntervals.h"
|
||||
#include "llvm/CodeGen/MachineBasicBlock.h"
|
||||
#include "llvm/CodeGen/MachineInstr.h"
|
||||
#include "llvm/CodeGen/SlotIndexes.h"
|
||||
#include "llvm/MC/LaneBitmask.h"
|
||||
#include "llvm/Support/Debug.h"
|
||||
#include <algorithm>
|
||||
#include <limits>
|
||||
|
||||
namespace llvm {
|
||||
|
||||
class MachineRegisterInfo;
|
||||
class raw_ostream;
|
||||
|
||||
struct GCNRegPressure {
|
||||
enum RegKind {
|
||||
SGPR32,
|
||||
SGPR_TUPLE,
|
||||
VGPR32,
|
||||
VGPR_TUPLE,
|
||||
TOTAL_KINDS
|
||||
};
|
||||
|
||||
GCNRegPressure() {
|
||||
clear();
|
||||
}
|
||||
|
||||
bool empty() const { return getSGPRNum() == 0 && getVGPRNum() == 0; }
|
||||
|
||||
void clear() { std::fill(&Value[0], &Value[TOTAL_KINDS], 0); }
|
||||
|
||||
unsigned getSGPRNum() const { return Value[SGPR32]; }
|
||||
unsigned getVGPRNum() const { return Value[VGPR32]; }
|
||||
|
||||
unsigned getVGPRTuplesWeight() const { return Value[VGPR_TUPLE]; }
|
||||
unsigned getSGPRTuplesWeight() const { return Value[SGPR_TUPLE]; }
|
||||
|
||||
unsigned getOccupancy(const SISubtarget &ST) const {
|
||||
return std::min(ST.getOccupancyWithNumSGPRs(getSGPRNum()),
|
||||
ST.getOccupancyWithNumVGPRs(getVGPRNum()));
|
||||
}
|
||||
|
||||
void inc(unsigned Reg,
|
||||
LaneBitmask PrevMask,
|
||||
LaneBitmask NewMask,
|
||||
const MachineRegisterInfo &MRI);
|
||||
|
||||
bool higherOccupancy(const SISubtarget &ST, const GCNRegPressure& O) const {
|
||||
return getOccupancy(ST) > O.getOccupancy(ST);
|
||||
}
|
||||
|
||||
bool less(const SISubtarget &ST, const GCNRegPressure& O,
|
||||
unsigned MaxOccupancy = std::numeric_limits<unsigned>::max()) const;
|
||||
|
||||
bool operator==(const GCNRegPressure &O) const {
|
||||
return std::equal(&Value[0], &Value[TOTAL_KINDS], O.Value);
|
||||
}
|
||||
|
||||
bool operator!=(const GCNRegPressure &O) const {
|
||||
return !(*this == O);
|
||||
}
|
||||
|
||||
void print(raw_ostream &OS, const SISubtarget *ST = nullptr) const;
|
||||
void dump() const { print(dbgs()); }
|
||||
|
||||
private:
|
||||
unsigned Value[TOTAL_KINDS];
|
||||
|
||||
static unsigned getRegKind(unsigned Reg, const MachineRegisterInfo &MRI);
|
||||
|
||||
friend GCNRegPressure max(const GCNRegPressure &P1,
|
||||
const GCNRegPressure &P2);
|
||||
};
|
||||
|
||||
inline GCNRegPressure max(const GCNRegPressure &P1, const GCNRegPressure &P2) {
|
||||
GCNRegPressure Res;
|
||||
for (unsigned I = 0; I < GCNRegPressure::TOTAL_KINDS; ++I)
|
||||
Res.Value[I] = std::max(P1.Value[I], P2.Value[I]);
|
||||
return Res;
|
||||
}
|
||||
|
||||
class GCNRPTracker {
|
||||
public:
|
||||
using LiveRegSet = DenseMap<unsigned, LaneBitmask>;
|
||||
|
||||
protected:
|
||||
const LiveIntervals &LIS;
|
||||
LiveRegSet LiveRegs;
|
||||
GCNRegPressure CurPressure, MaxPressure;
|
||||
const MachineInstr *LastTrackedMI = nullptr;
|
||||
mutable const MachineRegisterInfo *MRI = nullptr;
|
||||
|
||||
GCNRPTracker(const LiveIntervals &LIS_) : LIS(LIS_) {}
|
||||
|
||||
public:
|
||||
// live regs for the current state
|
||||
const decltype(LiveRegs) &getLiveRegs() const { return LiveRegs; }
|
||||
const MachineInstr *getLastTrackedMI() const { return LastTrackedMI; }
|
||||
|
||||
void clearMaxPressure() { MaxPressure.clear(); }
|
||||
|
||||
// returns MaxPressure, resetting it
|
||||
decltype(MaxPressure) moveMaxPressure() {
|
||||
auto Res = MaxPressure;
|
||||
MaxPressure.clear();
|
||||
return Res;
|
||||
}
|
||||
|
||||
decltype(LiveRegs) moveLiveRegs() {
|
||||
return std::move(LiveRegs);
|
||||
}
|
||||
|
||||
static void printLiveRegs(raw_ostream &OS, const LiveRegSet& LiveRegs,
|
||||
const MachineRegisterInfo &MRI);
|
||||
};
|
||||
|
||||
class GCNUpwardRPTracker : public GCNRPTracker {
|
||||
public:
|
||||
GCNUpwardRPTracker(const LiveIntervals &LIS_) : GCNRPTracker(LIS_) {}
|
||||
|
||||
// reset tracker to the point just below MI
|
||||
// filling live regs upon this point using LIS
|
||||
void reset(const MachineInstr &MI, const LiveRegSet *LiveRegs = nullptr);
|
||||
|
||||
// move to the state just above the MI
|
||||
void recede(const MachineInstr &MI);
|
||||
|
||||
// checks whether the tracker's state after receding MI corresponds
|
||||
// to reported by LIS
|
||||
bool isValid() const;
|
||||
};
|
||||
|
||||
class GCNDownwardRPTracker : public GCNRPTracker {
|
||||
// Last position of reset or advanceBeforeNext
|
||||
MachineBasicBlock::const_iterator NextMI;
|
||||
|
||||
MachineBasicBlock::const_iterator MBBEnd;
|
||||
|
||||
public:
|
||||
GCNDownwardRPTracker(const LiveIntervals &LIS_) : GCNRPTracker(LIS_) {}
|
||||
|
||||
const MachineBasicBlock::const_iterator getNext() const { return NextMI; }
|
||||
|
||||
// Reset tracker to the point before the MI
|
||||
// filling live regs upon this point using LIS.
|
||||
// Returns false if block is empty except debug values.
|
||||
bool reset(const MachineInstr &MI, const LiveRegSet *LiveRegs = nullptr);
|
||||
|
||||
// Move to the state right before the next MI. Returns false if reached
|
||||
// end of the block.
|
||||
bool advanceBeforeNext();
|
||||
|
||||
// Move to the state at the MI, advanceBeforeNext has to be called first.
|
||||
void advanceToNext();
|
||||
|
||||
// Move to the state at the next MI. Returns false if reached end of block.
|
||||
bool advance();
|
||||
|
||||
// Advance instructions until before End.
|
||||
bool advance(MachineBasicBlock::const_iterator End);
|
||||
|
||||
// Reset to Begin and advance to End.
|
||||
bool advance(MachineBasicBlock::const_iterator Begin,
|
||||
MachineBasicBlock::const_iterator End,
|
||||
const LiveRegSet *LiveRegsCopy = nullptr);
|
||||
};
|
||||
|
||||
LaneBitmask getLiveLaneMask(unsigned Reg,
|
||||
SlotIndex SI,
|
||||
const LiveIntervals &LIS,
|
||||
const MachineRegisterInfo &MRI);
|
||||
|
||||
GCNRPTracker::LiveRegSet getLiveRegs(SlotIndex SI,
|
||||
const LiveIntervals &LIS,
|
||||
const MachineRegisterInfo &MRI);
|
||||
|
||||
inline GCNRPTracker::LiveRegSet getLiveRegsAfter(const MachineInstr &MI,
|
||||
const LiveIntervals &LIS) {
|
||||
return getLiveRegs(LIS.getInstructionIndex(MI).getDeadSlot(), LIS,
|
||||
MI.getParent()->getParent()->getRegInfo());
|
||||
}
|
||||
|
||||
inline GCNRPTracker::LiveRegSet getLiveRegsBefore(const MachineInstr &MI,
|
||||
const LiveIntervals &LIS) {
|
||||
return getLiveRegs(LIS.getInstructionIndex(MI).getBaseIndex(), LIS,
|
||||
MI.getParent()->getParent()->getRegInfo());
|
||||
}
|
||||
|
||||
template <typename Range>
|
||||
GCNRegPressure getRegPressure(const MachineRegisterInfo &MRI,
|
||||
Range &&LiveRegs) {
|
||||
GCNRegPressure Res;
|
||||
for (const auto &RM : LiveRegs)
|
||||
Res.inc(RM.first, LaneBitmask::getNone(), RM.second, MRI);
|
||||
return Res;
|
||||
}
|
||||
|
||||
void printLivesAt(SlotIndex SI,
|
||||
const LiveIntervals &LIS,
|
||||
const MachineRegisterInfo &MRI);
|
||||
|
||||
} // end namespace llvm
|
||||
|
||||
#endif // LLVM_LIB_TARGET_AMDGPU_GCNREGPRESSURE_H
|
Reference in New Issue
Block a user