13 #include "llvm/ADT/iterator.h"
14 #include "llvm/Config/abi-breaking.h"
15 #include "llvm/Support/Casting.h"
16 #include "llvm/Support/Debug.h"
17 #include "llvm/Support/raw_ostream.h"
19 #define DEBUG_TYPE "dataflow"
20 #if LLVM_ENABLE_ABI_BREAKING_CHECKS
21 #define DATAFLOW_DEBUG(X) LLVM_DEBUG(X)
23 #define DATAFLOW_DEBUG(X)
42 auto inserted = dependents.insert({dependent, analysis});
46 llvm::dbgs() <<
"Creating dependency between " << debugName <<
" of "
47 <<
anchor <<
"\nand " << debugName <<
" on " << dependent
64 if (
Operation *op = llvm::dyn_cast<Operation *>(*
this)) {
67 return get<Block *>()->print(os);
75 if (
auto *
LatticeAnchor = llvm::dyn_cast<GenericLatticeAnchor *>(*
this))
77 if (
auto value = llvm::dyn_cast<Value>(*
this)) {
81 return get<ProgramPoint>().print(os);
85 if (
auto *
LatticeAnchor = llvm::dyn_cast<GenericLatticeAnchor *>(*
this))
87 if (
auto value = llvm::dyn_cast<Value>(*
this))
88 return value.getLoc();
91 if (
auto *op = llvm::dyn_cast<Operation *>(pp))
93 return pp.get<
Block *>()->getParent()->getLoc();
102 for (
DataFlowAnalysis &analysis : llvm::make_pointee_range(childAnalyses)) {
104 <<
"Priming analysis: " << analysis.debugName <<
"\n");
105 if (failed(analysis.initialize(top)))
112 while (!worklist.empty()) {
113 auto [point, analysis] = worklist.front();
116 DATAFLOW_DEBUG(llvm::dbgs() <<
"Invoking '" << analysis->debugName
117 <<
"' on: " << point <<
"\n");
118 if (failed(analysis->visit(point)))
124 }
while (!worklist.empty());
132 DATAFLOW_DEBUG(llvm::dbgs() <<
"Propagating update to " << state->debugName
133 <<
" of " << state->anchor <<
"\n"
134 <<
"Value: " << *state <<
"\n");
135 state->onUpdate(
this);
148 state->addDependency(point,
this);
#define DATAFLOW_DEBUG(X)
Base class for generic analysis states.
LLVM_DUMP_METHOD void dump() const
void addDependency(ProgramPoint point, DataFlowAnalysis *analysis)
Add a dependency to this analysis state on a lattice anchor and an analysis.
virtual void print(raw_ostream &os) const =0
Print the contents of the analysis state.
LatticeAnchor anchor
The lattice anchor to which the state belongs.
Block represents an ordered list of Operations.
Base class for all data-flow analyses.
void propagateIfChanged(AnalysisState *state, ChangeResult changed)
Propagate an update to a state if it changed.
virtual ~DataFlowAnalysis()
DataFlowAnalysis(DataFlowSolver &solver)
Create an analysis with a reference to the parent solver.
void addDependency(AnalysisState *state, ProgramPoint point)
Create a dependency between the given analysis state and lattice anchor on this analysis.
The general data-flow analysis solver.
void propagateIfChanged(AnalysisState *state, ChangeResult changed)
Propagate an update to an analysis state if it changed by pushing dependent work items to the back of...
LogicalResult initializeAndRun(Operation *top)
Initialize the children analyses starting from the provided top-level operation and run the analysis ...
virtual ~GenericLatticeAnchor()
This class defines the main interface for locations in MLIR and acts as a non-nullable wrapper around...
Set of flags used to control the behavior of the various IR print methods (e.g.
Operation is the basic unit of execution within MLIR.
void print(raw_ostream &os, const OpPrintingFlags &flags=std::nullopt)
Location getLoc()
The source location the operation was defined or derived from.
Include the generated interface declarations.
ChangeResult
A result type used to indicate if a change happened.
Fundamental IR components are supported as first-class lattice anchor.
Location getLoc() const
Get the source location of the lattice anchor.
void print(raw_ostream &os) const
Print the lattice anchor.
Program point represents a specific location in the execution of a program.
void print(raw_ostream &os) const
Print the program point.