20#include "llvm/ADT/BitVector.h"
21#include "llvm/ADT/SmallVector.h"
22#include "llvm/Support/TimeProfiler.h"
23#include "llvm/Support/raw_ostream.h"
31 llvm::TimeTraceScope(
"ComputePersistentOrigins");
33 llvm::BitVector PersistentOrigins(NumOrigins);
39 auto CheckOrigin = [&](
OriginID OID) {
40 if (PersistentOrigins.test(OID.Value))
42 auto &FirstSeenBlock = OriginToFirstSeenBlock[OID.Value];
43 if (FirstSeenBlock ==
nullptr)
45 if (FirstSeenBlock != B) {
47 PersistentOrigins.set(OID.Value);
51 switch (F->getKind()) {
57 CheckOrigin(OF->getDestOriginID());
58 CheckOrigin(OF->getSrcOriginID());
64 CheckOrigin(Cur->getOuterOriginID());
78 return PersistentOrigins;
97 : PersistentOrigins(Persistent), BlockLocalOrigins(BlockLocal) {}
101 return PersistentOrigins ==
Other.PersistentOrigins &&
102 BlockLocalOrigins ==
Other.BlockLocalOrigins;
106 void dump(llvm::raw_ostream &OS)
const {
107 OS <<
"LoanPropagationLattice State:\n";
108 OS <<
" Persistent Origins:\n";
109 if (PersistentOrigins.isEmpty())
111 for (
const auto &Entry : PersistentOrigins) {
112 if (Entry.second.isEmpty())
113 OS <<
" Origin " << Entry.first <<
" contains no loans\n";
114 for (
const LoanID &LID : Entry.second)
115 OS <<
" Origin " << Entry.first <<
" contains Loan " << LID <<
"\n";
117 OS <<
" Block-Local Origins:\n";
118 if (BlockLocalOrigins.isEmpty())
120 for (
const auto &Entry : BlockLocalOrigins) {
121 if (Entry.second.isEmpty())
122 OS <<
" Origin " << Entry.first <<
" contains no loans\n";
123 for (
const LoanID &LID : Entry.second)
124 OS <<
" Origin " << Entry.first <<
" contains Loan " << LID <<
"\n";
132 AnalysisImpl(
const CFG &
C, AnalysisDeclContext &AC, FactManager &F,
133 OriginLoanMap::Factory &OriginLoanMapFactory,
134 LoanSet::Factory &LoanSetFactory)
135 : DataflowAnalysis(
C, AC, F), OriginLoanMapFactory(OriginLoanMapFactory),
136 LoanSetFactory(LoanSetFactory),
139 using Base::transfer;
141 StringRef getAnalysisName()
const {
return "LoanPropagation"; }
143 Lattice getInitialState() {
return Lattice{}; }
147 Lattice join(Lattice A, Lattice B) {
149 A.PersistentOrigins, B.PersistentOrigins, OriginLoanMapFactory,
151 assert((S1 || S2) &&
"unexpectedly merging 2 empty sets");
156 return utils::join(*S1, *S2, LoanSetFactory);
161 return Lattice(JoinedOrigins, OriginLoanMapFactory.getEmptyMap());
165 Lattice
transfer(Lattice In,
const IssueFact &F) {
167 LoanID LID = F.getLoanID();
168 LoanSet NewLoans = LoanSetFactory.add(LoanSetFactory.getEmptySet(), LID);
169 return setLoans(In, OID, NewLoans);
175 Lattice
transfer(Lattice In,
const OriginFlowFact &F) {
176 OriginID DestOID = F.getDestOriginID();
177 OriginID SrcOID = F.getSrcOriginID();
180 F.getKillDest() ? LoanSetFactory.getEmptySet() : getLoans(In, DestOID);
181 LoanSet SrcLoans = getLoans(In, SrcOID);
184 return setLoans(In, DestOID, MergedLoans);
187 Lattice
transfer(Lattice In,
const KillOriginFact &F) {
188 return setLoans(In, F.getKilledOrigin(), LoanSetFactory.getEmptySet());
191 Lattice
transfer(Lattice In,
const ExpireFact &F) {
192 if (
auto OID = F.getOriginID())
193 return setLoans(In, *OID, LoanSetFactory.getEmptySet());
198 return getLoans(getState(P), OID);
203 bool isPersistent(
OriginID OID)
const {
204 return PersistentOrigins.test(OID.Value);
208 if (isPersistent(OID))
209 return Lattice(OriginLoanMapFactory.add(L.PersistentOrigins, OID, Loans),
210 L.BlockLocalOrigins);
211 return Lattice(L.PersistentOrigins,
212 OriginLoanMapFactory.add(L.BlockLocalOrigins, OID, Loans));
217 isPersistent(OID) ? &L.PersistentOrigins : &L.BlockLocalOrigins;
218 if (
auto *Loans = Map->lookup(OID))
220 return LoanSetFactory.getEmptySet();
223 OriginLoanMap::Factory &OriginLoanMapFactory;
224 LoanSet::Factory &LoanSetFactory;
228 llvm::BitVector PersistentOrigins;
233 using AnalysisImpl::AnalysisImpl;
238 OriginLoanMap::Factory &OriginLoanMapFactory,
239 LoanSet::Factory &LoanSetFactory)
240 : PImpl(
std::make_unique<
Impl>(
C, AC, F, OriginLoanMapFactory,
248 return PImpl->getLoans(OID, P);
This file defines AnalysisDeclContext, a class that manages the analysis context data for context sen...
static void dump(llvm::raw_ostream &OS, StringRef FunctionName, ArrayRef< CounterExpression > Expressions, ArrayRef< CounterMappingRegion > Regions)
Forward-declares and imports various common LLVM datatypes that clang wants to use unqualified.
AnalysisDeclContext contains the context data for the function, method or block under analysis.
Represents a single basic block in a source-level CFG.
Represents a source-level, intra-procedural CFG that represents the control-flow of a Stmt.
A generic, policy-based driver for dataflow analyses.
llvm::ArrayRef< const Fact * > getFacts(const CFGBlock *B) const
OriginManager & getOriginMgr()
An abstract base class for a single, atomic lifetime-relevant event.
@ InvalidateOrigin
An origin is invalidated (e.g. vector resized).
@ TestPoint
A marker for a specific point in the code, for testing.
@ Expire
A loan expires as its underlying storage is freed (e.g., variable goes out of scope).
@ Issue
A new loan is issued from a borrow expression (e.g., &x).
@ OriginFlow
An origin is propagated from a source to a destination (e.g., p = q).
@ MovedOrigin
An origin that is moved (e.g., passed to an rvalue reference parameter).
@ Use
An origin is used (eg. appears as l-value expression like DeclRefExpr).
@ OriginEscapes
An origin that escapes the function scope (e.g., via return).
@ KillOrigin
All loans of an origin are cleared.
OriginID getOriginID() const
All loans are cleared from an origin (e.g., assigning a callable without tracked origins to std::func...
OriginID getKilledOrigin() const
LoanSet getLoans(OriginID OID, ProgramPoint P) const
~LoanPropagationAnalysis()
LoanPropagationAnalysis(const CFG &C, AnalysisDeclContext &AC, FactManager &F, OriginLoanMap::Factory &OriginLoanMapFactory, LoanSet::Factory &LoanSetFactory)
A list of origins representing levels of indirection for pointer-like types.
OriginList * peelOuterOrigin() const
unsigned getNumOrigins() const
const OriginList * getUsedOrigins() const
void transfer(const StmtToEnvMap &StmtToEnv, const Stmt &S, Environment &Env, Environment::ValueModel &Model)
Evaluates S and updates Env accordingly.
@ OS
Indicates that the tracking object is a descendant of a referenced-counted OSObject,...
@ Asymmetric
An asymmetric join preserves keys unique to the first map as-is, while applying the JoinValues operat...
static llvm::ImmutableSet< T > join(llvm::ImmutableSet< T > A, llvm::ImmutableSet< T > B, typename llvm::ImmutableSet< T >::Factory &F)
Computes the union of two ImmutableSets.
const Fact * ProgramPoint
A ProgramPoint identifies a location in the CFG by pointing to a specific Fact.
llvm::ImmutableSet< LoanID > LoanSet
utils::ID< struct LoanTag > LoanID
utils::ID< struct OriginTag > OriginID
llvm::ImmutableMap< OriginID, LoanSet > OriginLoanMap
static llvm::BitVector computePersistentOrigins(const FactManager &FactMgr, const CFG &C)
bool operator==(const CallGraphNode::CallRecord &LHS, const CallGraphNode::CallRecord &RHS)
bool operator!=(CanQual< T > x, CanQual< U > y)
@ Other
Other implicit parameter.