20 auto Skip = [&DAG](
auto OpIt) {
22 return I ==
nullptr || DAG.getNode(
I) ==
nullptr;
24 while (OpIt != OpItE &&
Skip(OpIt))
32 assert(OpIt != OpItE &&
"Can't dereference end iterator!");
41 "Cant' dereference end iterator!");
48 assert(OpIt != OpItE &&
"Already at end!");
51 OpIt = PredIterator::skipBadIt(OpIt, OpItE, *DAG);
59 OpIt = PredIterator::skipBadIt(OpIt, OpItE, *DAG);
69 assert(DAG ==
Other.DAG &&
"Iterators of different DAGs!");
70 assert(N ==
Other.N &&
"Iterators of different nodes!");
71 return OpIt ==
Other.OpIt && MemIt ==
Other.MemIt;
79 return I ==
nullptr || DAG.
getNode(
I) ==
nullptr;
81 while (UserIt != UserItE && Skip(UserIt))
89 assert(UserIt != UserItE &&
"Can't dereference end iterator!");
94 if (UserIt != UserItE)
98 "Cant' dereference end iterator!");
105 assert(UserIt != UserItE &&
"Already at end!");
108 UserIt = SuccIterator::skipOutOfScope(UserIt, UserItE, *DAG);
113 if (UserIt != UserItE) {
116 UserIt = SuccIterator::skipOutOfScope(UserIt, UserItE, *DAG);
126 assert(DAG ==
Other.DAG &&
"Iterators of different DAGs!");
127 assert(N ==
Other.N &&
"Iterators of different nodes!");
128 return UserIt ==
Other.UserIt && MemIt ==
Other.MemIt;
132 if (this->SB !=
nullptr)
133 this->SB->eraseFromBundle(
this);
140 SB->eraseFromBundle(
this);
152 static constexpr unsigned Indent = 4;
153 for (
auto *Pred : MemPreds)
154 OS.
indent(Indent) <<
"<-" << *Pred->getInstruction() <<
"\n";
166 I =
I->getNextNode();
179 I =
I->getPrevNode();
192 if (TopMemN ==
nullptr)
195 assert(BotMemN !=
nullptr &&
"TopMemN should be null too!");
200DependencyGraph::DependencyType
205 return DependencyType::ReadAfterWrite;
207 return DependencyType::WriteAfterWrite;
210 return DependencyType::WriteAfterRead;
213 return DependencyType::Control;
215 return DependencyType::Control;
218 return DependencyType::Other;
219 return DependencyType::None;
225 return !LI->isUnordered();
227 return !
SI->isUnordered();
232 bool Is = IsOrdered(
I);
234 "An ordered instruction must be a MemDepCandidate!");
239 DependencyType DepType) {
240 std::optional<MemoryLocation> DstLocOpt =
246 "Expected a mem instr");
253 case DependencyType::ReadAfterWrite:
254 case DependencyType::WriteAfterWrite:
256 case DependencyType::WriteAfterRead:
264 DependencyType RoughDepType = getRoughDepType(SrcI, DstI);
265 switch (RoughDepType) {
266 case DependencyType::ReadAfterWrite:
267 case DependencyType::WriteAfterWrite:
268 case DependencyType::WriteAfterRead:
269 return alias(SrcI, DstI, RoughDepType);
270 case DependencyType::Control:
276 case DependencyType::Other:
278 case DependencyType::None:
284void DependencyGraph::scanAndAddDeps(
MemDGNode &DstN,
287 "DstN is the mem dep destination, so it must be mem");
293 if (hasDep(SrcI, DstI))
294 DstN.addMemPred(&SrcN);
298void DependencyGraph::setDefUseUnscheduledSuccs(
313 if (OpI->getParent() !=
I.getParent())
315 if (!NewInterval.contains(OpI))
320 OpN->incrUnscheduledSuccs();
325 bool NewIsAbove = DAGInterval.empty() || NewInterval.comesBefore(DAGInterval);
326 const auto &TopInterval = NewIsAbove ? NewInterval : DAGInterval;
327 const auto &BotInterval = NewIsAbove ? DAGInterval : NewInterval;
341 if (BotN->scheduled())
343 for (
Value *
Op : BotI.operands()) {
350 if (!TopInterval.contains(OpI))
352 OpN->incrUnscheduledSuccs();
365 MemN->setPrevNode(LastMemN);
370 if (!DAGInterval.empty()) {
371 bool NewIsAbove = NewInterval.comesBefore(DAGInterval);
372 const auto &TopInterval = NewIsAbove ? NewInterval : DAGInterval;
373 const auto &BotInterval = NewIsAbove ? DAGInterval : NewInterval;
378 assert((LinkTopN ==
nullptr || LinkBotN ==
nullptr ||
379 LinkTopN->comesBefore(LinkBotN)) &&
381 if (LinkTopN !=
nullptr && LinkBotN !=
nullptr) {
382 LinkTopN->setNextNode(LinkBotN);
387 auto UnionIntvl = DAGInterval.getUnionInterval(NewInterval);
392 if (ChainTopN !=
nullptr && ChainBotN !=
nullptr) {
393 for (
auto *
N = ChainTopN->getNextNode(), *LastN = ChainTopN;
N !=
nullptr;
394 LastN =
N,
N =
N->getNextNode()) {
395 assert(
N == LastN->getNextNode() &&
"Bad chain!");
396 assert(
N->getPrevNode() == LastN &&
"Bad chain!");
402 setDefUseUnscheduledSuccs(NewInterval);
408 for (
auto *PrevI = IncludingN ?
I :
I->getPrevNode(); PrevI !=
nullptr;
409 PrevI = PrevI->getPrevNode()) {
411 if (PrevN ==
nullptr)
414 if (PrevMemN !=
nullptr && PrevMemN != SkipN)
423 for (
auto *NextI = IncludingN ?
I :
I->getNextNode(); NextI !=
nullptr;
424 NextI = NextI->getNextNode()) {
426 if (NextN ==
nullptr)
429 if (NextMemN !=
nullptr && NextMemN != SkipN)
435void DependencyGraph::notifyCreateInstr(
Instruction *
I) {
440 if (!(DAGInterval.contains(
I) || DAGInterval.touches(
I)))
443 DAGInterval = DAGInterval.getUnionInterval({
I,
I});
448 if (MemN !=
nullptr) {
449 if (
auto *PrevMemN = getMemDGNodeBefore(MemN,
false)) {
450 PrevMemN->NextMemN = MemN;
451 MemN->PrevMemN = PrevMemN;
453 if (
auto *NextMemN = getMemDGNodeAfter(MemN,
false)) {
454 NextMemN->PrevMemN = MemN;
455 MemN->NextMemN = NextMemN;
460 if (DAGInterval.top()->comesBefore(
I)) {
463 scanAndAddDeps(*MemN, SrcInterval);
466 if (
I->comesBefore(DAGInterval.bottom())) {
475void DependencyGraph::notifyMoveInstr(
Instruction *
I,
const BBIterator &To) {
481 assert(!(To != BB->end() && &*To ==
I->getNextNode()) &&
482 !(To == BB->end() && std::next(
I->getIterator()) == BB->end()) &&
483 "Should not have been called if destination is same as origin.");
487 assert(To.getNodeParent() ==
I->getParent() &&
488 "TODO: We don't support movement across BBs!");
490 (To == std::next(DAGInterval.bottom()->getIterator()) ||
491 (To != BB->end() && std::next(To) == DAGInterval.top()->getIterator()) ||
492 (To != BB->end() && DAGInterval.contains(&*To))) &&
493 "TODO: To should be either within the DAGInterval or right "
497 auto OrigDAGInterval = DAGInterval;
500 DAGInterval.notifyMoveInstr(
I, To);
513 MemN->detachFromChain();
528 if (To == BB->end() ||
529 To == std::next(OrigDAGInterval.bottom()->getIterator())) {
534 getMemDGNodeBefore(InsertAfterN,
true, MemN));
539 getMemDGNodeBefore(BeforeToN,
false, MemN));
541 getMemDGNodeAfter(BeforeToN,
true, MemN));
555 auto *PrevMemN = getMemDGNodeBefore(MemN,
false);
556 auto *NextMemN = getMemDGNodeAfter(MemN,
false);
557 if (PrevMemN !=
nullptr)
558 PrevMemN->NextMemN = NextMemN;
559 if (NextMemN !=
nullptr)
560 NextMemN->PrevMemN = PrevMemN;
563 while (!MemN->memPreds().empty()) {
564 auto *PredN = *MemN->memPreds().begin();
565 MemN->removeMemPred(PredN);
567 while (!MemN->memSuccs().empty()) {
568 auto *SuccN = *MemN->memSuccs().begin();
569 SuccN->removeMemPred(MemN);
575 for (
auto *PredN :
N->preds(*
this))
576 PredN->decrUnscheduledSuccs();
579 InstrToNodeMap.erase(
I);
582void DependencyGraph::notifySetUse(
const Use &U,
Value *NewSrc) {
591 if (UserI ==
nullptr)
594 if (UserN ==
nullptr)
598 if (UserN->scheduled())
603 if (
auto *CurrSrcN =
getNode(CurrSrcI)) {
605 if (!CurrSrcN->scheduled())
606 CurrSrcN->decrUnscheduledSuccs();
610 if (
auto *NewSrcN =
getNode(NewSrcI)) {
612 if (!NewSrcN->scheduled())
613 NewSrcN->incrUnscheduledSuccs();
624 auto NewInterval = Union.getSingleDiff(DAGInterval);
625 if (NewInterval.empty())
628 createNewNodes(NewInterval);
644 if (!DstRange.empty()) {
647 scanAndAddDeps(DstN, SrcRange);
652 if (MemDAGInterval.empty()) {
653 FullScan(NewInterval);
670 else if (DAGInterval.bottom()->comesBefore(NewInterval.top())) {
672 auto SrcRangeFull = MemDAGInterval.getUnionInterval(DstRange);
676 scanAndAddDeps(DstN, SrcRange);
680 else if (NewInterval.bottom()->comesBefore(DAGInterval.top())) {
694 FullScan(NewInterval);
710 auto DstRangeOld = MemDAGInterval;
713 scanAndAddDeps(DstN, SrcRange);
726 Nodes.
reserve(InstrToNodeMap.size());
727 for (
const auto &Pair : InstrToNodeMap)
733 for (
auto *
N : Nodes)
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
Represent a constant reference to an array (0 or more elements consecutively in memory),...
bool empty() const
Check if the array is empty.
LLVM_ABI bool mayWriteToMemory() const LLVM_READONLY
Return true if this instruction may modify memory.
LLVM_ABI bool mayReadFromMemory() const LLVM_READONLY
Return true if this instruction may read memory.
void reserve(size_type N)
void push_back(const T &Elt)
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
This class implements an extremely fast bulk output stream that can only output to a stream.
raw_ostream & indent(unsigned NumSpaces)
indent - Insert 'NumSpaces' spaces.
A DependencyGraph Node that points to an Instruction and contains memory dependency edges.
virtual void print(raw_ostream &OS, bool PrintDeps=true) const
static bool isMemDepCandidate(Instruction *I)
We consider I as a Memory Dependency Candidate instruction if it reads/write memory or if it has side...
void setSchedBundle(SchedBundle &SB)
SchedBundle * SB
The scheduler bundle that this node belongs to.
bool Scheduled
This is true if this node has been scheduled.
std::optional< unsigned > UnscheduledSuccs
The number of unscheduled successors.
static bool isMemDepNodeCandidate(Instruction *I)
\Returns true if I is a memory dependency candidate instruction.
static bool isFenceLike(Instruction *I)
\Returns true if I is fence like. It excludes non-mem intrinsics.
LLVM_DUMP_METHOD void dump() const
Instruction * getInstruction() const
static bool isStackSaveOrRestoreIntrinsic(Instruction *I)
LLVM_DUMP_METHOD void dump() const
DGNode * getNode(Instruction *I) const
DGNode * getNodeOrNull(Instruction *I) const
Like getNode() but returns nullptr if I is nullptr.
void print(raw_ostream &OS) const
DGNode * getOrCreateNode(Instruction *I)
LLVM_ABI Interval< Instruction > extend(ArrayRef< Instruction * > Instrs)
Build/extend the dependency graph such that it includes Instrs.
A sandboxir::User with operands, opcode and linked with previous/next instructions in an instruction ...
bool mayWriteToMemory() const
bool mayReadFromMemory() const
bool comesBefore(const Instruction *Other) const
Given an instruction Other in the same basic block as this instruction, return true if this instructi...
bool isTerminator() const
static LLVM_ABI MemDGNode * getBotMemDGNode(const Interval< Instruction > &Intvl, const DependencyGraph &DAG)
Scans the instruction chain in Intvl bottom-up, returning the bottom-most MemDGNode,...
static LLVM_ABI MemDGNode * getTopMemDGNode(const Interval< Instruction > &Intvl, const DependencyGraph &DAG)
Scans the instruction chain in Intvl top-down, returning the top-most MemDGNode, or nullptr.
static LLVM_ABI Interval< MemDGNode > make(const Interval< Instruction > &Instrs, DependencyGraph &DAG)
Given Instrs it finds their closest mem nodes in the interval and returns the corresponding mem range...
A DependencyGraph Node for instructions that may read/write memory, or have some ordering constraints...
void print(raw_ostream &OS, bool PrintDeps=true) const override
LLVM_ABI value_type operator*()
LLVM_ABI PredIterator & operator++()
LLVM_ABI bool operator==(const PredIterator &Other) const
LLVM_ABI value_type operator*()
LLVM_ABI bool operator==(const SuccIterator &Other) const
LLVM_ABI SuccIterator & operator++()
Represents a Def-use/Use-def edge in SandboxIR.
OperandUseIterator op_iterator
static ModRefInfo aliasAnalysisGetModRefInfo(BatchAAResults &BatchAA, const Instruction *I, const std::optional< MemoryLocation > &OptLoc)
Equivalent to BatchAA::getModRefInfo().
static std::optional< llvm::MemoryLocation > memoryLocationGetOrNone(const Instruction *I)
Equivalent to MemoryLocation::getOrNone(I).
A SandboxIR Value has users. This is the base class.
mapped_iterator< sandboxir::UserUseIterator, UseToUser > user_iterator
#define llvm_unreachable(msg)
Marks that the current location is not supposed to be reachable.
static bool isOrdered(Instruction *I)
template class LLVM_TEMPLATE_ABI Interval< MemDGNode >
BasicBlock(llvm::BasicBlock *BB, Context &SBCtx)
template class LLVM_TEMPLATE_ABI Interval< Instruction >
friend class Instruction
Iterator for Instructions in a `BasicBlock.
auto drop_begin(T &&RangeOrContainer, size_t N=1)
Return a range covering RangeOrContainer with the first N elements excluded.
FunctionAddr VTableAddr Value
decltype(auto) dyn_cast(const From &Val)
dyn_cast<X> - Return the argument parameter cast to the specified type.
auto dyn_cast_or_null(const Y &Val)
auto reverse(ContainerTy &&C)
bool isModSet(const ModRefInfo MRI)
void sort(IteratorTy Start, IteratorTy End)
LLVM_ABI raw_ostream & dbgs()
dbgs() - This returns a reference to a raw_ostream for debugging messages.
bool isa(const From &Val)
isa<X> - Return true if the parameter to the template is an instance of one of the template type argu...
ModRefInfo
Flags indicating whether a memory access modifies or references memory.
@ ModRef
The access may reference and may modify the value stored in memory.
DWARFExpression::Operation Op
decltype(auto) cast(const From &Val)
cast<X> - Return the argument parameter cast to the specified type.
bool isRefSet(const ModRefInfo MRI)