xref: /freebsd-src/contrib/llvm-project/llvm/lib/Analysis/StackSafetyAnalysis.cpp (revision f3457ed94241be9d4c2c3ab337c9086d5c45c43f)
10b57cec5SDimitry Andric //===- StackSafetyAnalysis.cpp - Stack memory safety analysis -------------===//
20b57cec5SDimitry Andric //
30b57cec5SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
40b57cec5SDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
50b57cec5SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
60b57cec5SDimitry Andric //
70b57cec5SDimitry Andric //===----------------------------------------------------------------------===//
80b57cec5SDimitry Andric //
90b57cec5SDimitry Andric //===----------------------------------------------------------------------===//
100b57cec5SDimitry Andric 
110b57cec5SDimitry Andric #include "llvm/Analysis/StackSafetyAnalysis.h"
125ffd83dbSDimitry Andric #include "llvm/ADT/APInt.h"
135ffd83dbSDimitry Andric #include "llvm/ADT/SmallPtrSet.h"
145ffd83dbSDimitry Andric #include "llvm/ADT/SmallVector.h"
155ffd83dbSDimitry Andric #include "llvm/ADT/Statistic.h"
165ffd83dbSDimitry Andric #include "llvm/Analysis/ModuleSummaryAnalysis.h"
174824e7fdSDimitry Andric #include "llvm/Analysis/ScalarEvolution.h"
185ffd83dbSDimitry Andric #include "llvm/Analysis/StackLifetime.h"
195ffd83dbSDimitry Andric #include "llvm/IR/ConstantRange.h"
205ffd83dbSDimitry Andric #include "llvm/IR/DerivedTypes.h"
215ffd83dbSDimitry Andric #include "llvm/IR/GlobalValue.h"
220b57cec5SDimitry Andric #include "llvm/IR/InstIterator.h"
234824e7fdSDimitry Andric #include "llvm/IR/Instruction.h"
245ffd83dbSDimitry Andric #include "llvm/IR/Instructions.h"
250b57cec5SDimitry Andric #include "llvm/IR/IntrinsicInst.h"
26e8d8bef9SDimitry Andric #include "llvm/IR/ModuleSummaryIndex.h"
27480093f4SDimitry Andric #include "llvm/InitializePasses.h"
285ffd83dbSDimitry Andric #include "llvm/Support/Casting.h"
29480093f4SDimitry Andric #include "llvm/Support/CommandLine.h"
305ffd83dbSDimitry Andric #include "llvm/Support/FormatVariadic.h"
310b57cec5SDimitry Andric #include "llvm/Support/raw_ostream.h"
325ffd83dbSDimitry Andric #include <algorithm>
335ffd83dbSDimitry Andric #include <memory>
34349cc55cSDimitry Andric #include <tuple>
350b57cec5SDimitry Andric 
360b57cec5SDimitry Andric using namespace llvm;
370b57cec5SDimitry Andric 
380b57cec5SDimitry Andric #define DEBUG_TYPE "stack-safety"
390b57cec5SDimitry Andric 
405ffd83dbSDimitry Andric STATISTIC(NumAllocaStackSafe, "Number of safe allocas");
415ffd83dbSDimitry Andric STATISTIC(NumAllocaTotal, "Number of total allocas");
425ffd83dbSDimitry Andric 
43e8d8bef9SDimitry Andric STATISTIC(NumCombinedCalleeLookupTotal,
44e8d8bef9SDimitry Andric           "Number of total callee lookups on combined index.");
45e8d8bef9SDimitry Andric STATISTIC(NumCombinedCalleeLookupFailed,
46e8d8bef9SDimitry Andric           "Number of failed callee lookups on combined index.");
47e8d8bef9SDimitry Andric STATISTIC(NumModuleCalleeLookupTotal,
48e8d8bef9SDimitry Andric           "Number of total callee lookups on module index.");
49e8d8bef9SDimitry Andric STATISTIC(NumModuleCalleeLookupFailed,
50e8d8bef9SDimitry Andric           "Number of failed callee lookups on module index.");
51e8d8bef9SDimitry Andric STATISTIC(NumCombinedParamAccessesBefore,
52e8d8bef9SDimitry Andric           "Number of total param accesses before generateParamAccessSummary.");
53e8d8bef9SDimitry Andric STATISTIC(NumCombinedParamAccessesAfter,
54e8d8bef9SDimitry Andric           "Number of total param accesses after generateParamAccessSummary.");
55e8d8bef9SDimitry Andric STATISTIC(NumCombinedDataFlowNodes,
56e8d8bef9SDimitry Andric           "Number of total nodes in combined index for dataflow processing.");
57e8d8bef9SDimitry Andric STATISTIC(NumIndexCalleeUnhandled, "Number of index callee which are unhandled.");
58e8d8bef9SDimitry Andric STATISTIC(NumIndexCalleeMultipleWeak, "Number of index callee non-unique weak.");
59e8d8bef9SDimitry Andric STATISTIC(NumIndexCalleeMultipleExternal, "Number of index callee non-unique external.");
60e8d8bef9SDimitry Andric 
61e8d8bef9SDimitry Andric 
620b57cec5SDimitry Andric static cl::opt<int> StackSafetyMaxIterations("stack-safety-max-iterations",
630b57cec5SDimitry Andric                                              cl::init(20), cl::Hidden);
640b57cec5SDimitry Andric 
655ffd83dbSDimitry Andric static cl::opt<bool> StackSafetyPrint("stack-safety-print", cl::init(false),
665ffd83dbSDimitry Andric                                       cl::Hidden);
675ffd83dbSDimitry Andric 
685ffd83dbSDimitry Andric static cl::opt<bool> StackSafetyRun("stack-safety-run", cl::init(false),
695ffd83dbSDimitry Andric                                     cl::Hidden);
705ffd83dbSDimitry Andric 
710b57cec5SDimitry Andric namespace {
720b57cec5SDimitry Andric 
73e8d8bef9SDimitry Andric // Check if we should bailout for such ranges.
74e8d8bef9SDimitry Andric bool isUnsafe(const ConstantRange &R) {
75e8d8bef9SDimitry Andric   return R.isEmptySet() || R.isFullSet() || R.isUpperSignWrapped();
76e8d8bef9SDimitry Andric }
77e8d8bef9SDimitry Andric 
78e8d8bef9SDimitry Andric ConstantRange addOverflowNever(const ConstantRange &L, const ConstantRange &R) {
79e8d8bef9SDimitry Andric   assert(!L.isSignWrappedSet());
80e8d8bef9SDimitry Andric   assert(!R.isSignWrappedSet());
81e8d8bef9SDimitry Andric   if (L.signedAddMayOverflow(R) !=
82e8d8bef9SDimitry Andric       ConstantRange::OverflowResult::NeverOverflows)
83e8d8bef9SDimitry Andric     return ConstantRange::getFull(L.getBitWidth());
84e8d8bef9SDimitry Andric   ConstantRange Result = L.add(R);
85e8d8bef9SDimitry Andric   assert(!Result.isSignWrappedSet());
86e8d8bef9SDimitry Andric   return Result;
87e8d8bef9SDimitry Andric }
88e8d8bef9SDimitry Andric 
89e8d8bef9SDimitry Andric ConstantRange unionNoWrap(const ConstantRange &L, const ConstantRange &R) {
90e8d8bef9SDimitry Andric   assert(!L.isSignWrappedSet());
91e8d8bef9SDimitry Andric   assert(!R.isSignWrappedSet());
92e8d8bef9SDimitry Andric   auto Result = L.unionWith(R);
93e8d8bef9SDimitry Andric   // Two non-wrapped sets can produce wrapped.
94e8d8bef9SDimitry Andric   if (Result.isSignWrappedSet())
95e8d8bef9SDimitry Andric     Result = ConstantRange::getFull(Result.getBitWidth());
96e8d8bef9SDimitry Andric   return Result;
97e8d8bef9SDimitry Andric }
98e8d8bef9SDimitry Andric 
990b57cec5SDimitry Andric /// Describes use of address in as a function call argument.
1005ffd83dbSDimitry Andric template <typename CalleeTy> struct CallInfo {
1010b57cec5SDimitry Andric   /// Function being called.
1025ffd83dbSDimitry Andric   const CalleeTy *Callee = nullptr;
1030b57cec5SDimitry Andric   /// Index of argument which pass address.
1040b57cec5SDimitry Andric   size_t ParamNo = 0;
1050b57cec5SDimitry Andric 
106e8d8bef9SDimitry Andric   CallInfo(const CalleeTy *Callee, size_t ParamNo)
107e8d8bef9SDimitry Andric       : Callee(Callee), ParamNo(ParamNo) {}
108e8d8bef9SDimitry Andric 
109e8d8bef9SDimitry Andric   struct Less {
110e8d8bef9SDimitry Andric     bool operator()(const CallInfo &L, const CallInfo &R) const {
111e8d8bef9SDimitry Andric       return std::tie(L.ParamNo, L.Callee) < std::tie(R.ParamNo, R.Callee);
1120b57cec5SDimitry Andric     }
113e8d8bef9SDimitry Andric   };
114e8d8bef9SDimitry Andric };
1150b57cec5SDimitry Andric 
1160b57cec5SDimitry Andric /// Describe uses of address (alloca or parameter) inside of the function.
1175ffd83dbSDimitry Andric template <typename CalleeTy> struct UseInfo {
1180b57cec5SDimitry Andric   // Access range if the address (alloca or parameters).
1190b57cec5SDimitry Andric   // It is allowed to be empty-set when there are no known accesses.
1200b57cec5SDimitry Andric   ConstantRange Range;
1214824e7fdSDimitry Andric   std::set<const Instruction *> UnsafeAccesses;
1220b57cec5SDimitry Andric 
1230b57cec5SDimitry Andric   // List of calls which pass address as an argument.
124e8d8bef9SDimitry Andric   // Value is offset range of address from base address (alloca or calling
125e8d8bef9SDimitry Andric   // function argument). Range should never set to empty-set, that is an invalid
126e8d8bef9SDimitry Andric   // access range that can cause empty-set to be propagated with
127e8d8bef9SDimitry Andric   // ConstantRange::add
128e8d8bef9SDimitry Andric   using CallsTy = std::map<CallInfo<CalleeTy>, ConstantRange,
129e8d8bef9SDimitry Andric                            typename CallInfo<CalleeTy>::Less>;
130e8d8bef9SDimitry Andric   CallsTy Calls;
1310b57cec5SDimitry Andric 
1325ffd83dbSDimitry Andric   UseInfo(unsigned PointerSize) : Range{PointerSize, false} {}
1330b57cec5SDimitry Andric 
134e8d8bef9SDimitry Andric   void updateRange(const ConstantRange &R) { Range = unionNoWrap(Range, R); }
1354824e7fdSDimitry Andric   void addRange(const Instruction *I, const ConstantRange &R, bool IsSafe) {
1364824e7fdSDimitry Andric     if (!IsSafe)
1374824e7fdSDimitry Andric       UnsafeAccesses.insert(I);
138349cc55cSDimitry Andric     updateRange(R);
139349cc55cSDimitry Andric   }
1400b57cec5SDimitry Andric };
1410b57cec5SDimitry Andric 
1425ffd83dbSDimitry Andric template <typename CalleeTy>
1435ffd83dbSDimitry Andric raw_ostream &operator<<(raw_ostream &OS, const UseInfo<CalleeTy> &U) {
1440b57cec5SDimitry Andric   OS << U.Range;
1450b57cec5SDimitry Andric   for (auto &Call : U.Calls)
146e8d8bef9SDimitry Andric     OS << ", "
147e8d8bef9SDimitry Andric        << "@" << Call.first.Callee->getName() << "(arg" << Call.first.ParamNo
148e8d8bef9SDimitry Andric        << ", " << Call.second << ")";
1490b57cec5SDimitry Andric   return OS;
1500b57cec5SDimitry Andric }
1510b57cec5SDimitry Andric 
1525ffd83dbSDimitry Andric /// Calculate the allocation size of a given alloca. Returns empty range
1535ffd83dbSDimitry Andric // in case of confution.
1545ffd83dbSDimitry Andric ConstantRange getStaticAllocaSizeRange(const AllocaInst &AI) {
1550fca6ea1SDimitry Andric   const DataLayout &DL = AI.getDataLayout();
1565ffd83dbSDimitry Andric   TypeSize TS = DL.getTypeAllocSize(AI.getAllocatedType());
157349cc55cSDimitry Andric   unsigned PointerSize = DL.getPointerTypeSizeInBits(AI.getType());
1585ffd83dbSDimitry Andric   // Fallback to empty range for alloca size.
1595ffd83dbSDimitry Andric   ConstantRange R = ConstantRange::getEmpty(PointerSize);
1605ffd83dbSDimitry Andric   if (TS.isScalable())
1615ffd83dbSDimitry Andric     return R;
162bdd1243dSDimitry Andric   APInt APSize(PointerSize, TS.getFixedValue(), true);
1635ffd83dbSDimitry Andric   if (APSize.isNonPositive())
1645ffd83dbSDimitry Andric     return R;
1655ffd83dbSDimitry Andric   if (AI.isArrayAllocation()) {
1665ffd83dbSDimitry Andric     const auto *C = dyn_cast<ConstantInt>(AI.getArraySize());
1670b57cec5SDimitry Andric     if (!C)
1685ffd83dbSDimitry Andric       return R;
1695ffd83dbSDimitry Andric     bool Overflow = false;
1705ffd83dbSDimitry Andric     APInt Mul = C->getValue();
1715ffd83dbSDimitry Andric     if (Mul.isNonPositive())
1725ffd83dbSDimitry Andric       return R;
1735ffd83dbSDimitry Andric     Mul = Mul.sextOrTrunc(PointerSize);
1745ffd83dbSDimitry Andric     APSize = APSize.smul_ov(Mul, Overflow);
1755ffd83dbSDimitry Andric     if (Overflow)
1765ffd83dbSDimitry Andric       return R;
1770b57cec5SDimitry Andric   }
178349cc55cSDimitry Andric   R = ConstantRange(APInt::getZero(PointerSize), APSize);
1795ffd83dbSDimitry Andric   assert(!isUnsafe(R));
1805ffd83dbSDimitry Andric   return R;
1810b57cec5SDimitry Andric }
1820b57cec5SDimitry Andric 
1835ffd83dbSDimitry Andric template <typename CalleeTy> struct FunctionInfo {
1845ffd83dbSDimitry Andric   std::map<const AllocaInst *, UseInfo<CalleeTy>> Allocas;
1855ffd83dbSDimitry Andric   std::map<uint32_t, UseInfo<CalleeTy>> Params;
1860b57cec5SDimitry Andric   // TODO: describe return value as depending on one or more of its arguments.
1870b57cec5SDimitry Andric 
1880b57cec5SDimitry Andric   // StackSafetyDataFlowAnalysis counter stored here for faster access.
1890b57cec5SDimitry Andric   int UpdateCount = 0;
1900b57cec5SDimitry Andric 
1915ffd83dbSDimitry Andric   void print(raw_ostream &O, StringRef Name, const Function *F) const {
1920b57cec5SDimitry Andric     // TODO: Consider different printout format after
1930b57cec5SDimitry Andric     // StackSafetyDataFlowAnalysis. Calls and parameters are irrelevant then.
1945ffd83dbSDimitry Andric     O << "  @" << Name << ((F && F->isDSOLocal()) ? "" : " dso_preemptable")
1955ffd83dbSDimitry Andric       << ((F && F->isInterposable()) ? " interposable" : "") << "\n";
1965ffd83dbSDimitry Andric 
1970b57cec5SDimitry Andric     O << "    args uses:\n";
1985ffd83dbSDimitry Andric     for (auto &KV : Params) {
1995ffd83dbSDimitry Andric       O << "      ";
2005ffd83dbSDimitry Andric       if (F)
2015ffd83dbSDimitry Andric         O << F->getArg(KV.first)->getName();
2025ffd83dbSDimitry Andric       else
2035ffd83dbSDimitry Andric         O << formatv("arg{0}", KV.first);
2045ffd83dbSDimitry Andric       O << "[]: " << KV.second << "\n";
2050b57cec5SDimitry Andric     }
2060b57cec5SDimitry Andric 
2075ffd83dbSDimitry Andric     O << "    allocas uses:\n";
2085ffd83dbSDimitry Andric     if (F) {
209fcaf7f86SDimitry Andric       for (const auto &I : instructions(F)) {
2105ffd83dbSDimitry Andric         if (const AllocaInst *AI = dyn_cast<AllocaInst>(&I)) {
2115ffd83dbSDimitry Andric           auto &AS = Allocas.find(AI)->second;
2125ffd83dbSDimitry Andric           O << "      " << AI->getName() << "["
2135ffd83dbSDimitry Andric             << getStaticAllocaSizeRange(*AI).getUpper() << "]: " << AS << "\n";
2145ffd83dbSDimitry Andric         }
2155ffd83dbSDimitry Andric       }
2165ffd83dbSDimitry Andric     } else {
2175ffd83dbSDimitry Andric       assert(Allocas.empty());
2185ffd83dbSDimitry Andric     }
2195ffd83dbSDimitry Andric   }
2200b57cec5SDimitry Andric };
2210b57cec5SDimitry Andric 
2225ffd83dbSDimitry Andric using GVToSSI = std::map<const GlobalValue *, FunctionInfo<GlobalValue>>;
2235ffd83dbSDimitry Andric 
2245ffd83dbSDimitry Andric } // namespace
2255ffd83dbSDimitry Andric 
2265ffd83dbSDimitry Andric struct StackSafetyInfo::InfoTy {
2275ffd83dbSDimitry Andric   FunctionInfo<GlobalValue> Info;
2285ffd83dbSDimitry Andric };
2295ffd83dbSDimitry Andric 
2305ffd83dbSDimitry Andric struct StackSafetyGlobalInfo::InfoTy {
2315ffd83dbSDimitry Andric   GVToSSI Info;
2325ffd83dbSDimitry Andric   SmallPtrSet<const AllocaInst *, 8> SafeAllocas;
2334824e7fdSDimitry Andric   std::set<const Instruction *> UnsafeAccesses;
2345ffd83dbSDimitry Andric };
2350b57cec5SDimitry Andric 
2360b57cec5SDimitry Andric namespace {
2370b57cec5SDimitry Andric 
2380b57cec5SDimitry Andric class StackSafetyLocalAnalysis {
2395ffd83dbSDimitry Andric   Function &F;
2400b57cec5SDimitry Andric   const DataLayout &DL;
2410b57cec5SDimitry Andric   ScalarEvolution &SE;
2420b57cec5SDimitry Andric   unsigned PointerSize = 0;
2430b57cec5SDimitry Andric 
2440b57cec5SDimitry Andric   const ConstantRange UnknownRange;
2450b57cec5SDimitry Andric 
2460fca6ea1SDimitry Andric   /// FIXME: This function is a bandaid, it's only needed
2470fca6ea1SDimitry Andric   /// because this pass doesn't handle address spaces of different pointer
2480fca6ea1SDimitry Andric   /// sizes.
2490fca6ea1SDimitry Andric   ///
2500fca6ea1SDimitry Andric   /// \returns \p Val's SCEV as a pointer of AS zero, or nullptr if it can't be
2510fca6ea1SDimitry Andric   /// converted to AS 0.
2520fca6ea1SDimitry Andric   const SCEV *getSCEVAsPointer(Value *Val);
2530fca6ea1SDimitry Andric 
2545ffd83dbSDimitry Andric   ConstantRange offsetFrom(Value *Addr, Value *Base);
2555ffd83dbSDimitry Andric   ConstantRange getAccessRange(Value *Addr, Value *Base,
2565ffd83dbSDimitry Andric                                const ConstantRange &SizeRange);
2575ffd83dbSDimitry Andric   ConstantRange getAccessRange(Value *Addr, Value *Base, TypeSize Size);
2580b57cec5SDimitry Andric   ConstantRange getMemIntrinsicAccessRange(const MemIntrinsic *MI, const Use &U,
2595ffd83dbSDimitry Andric                                            Value *Base);
2600b57cec5SDimitry Andric 
261349cc55cSDimitry Andric   void analyzeAllUses(Value *Ptr, UseInfo<GlobalValue> &AS,
2625ffd83dbSDimitry Andric                       const StackLifetime &SL);
2630b57cec5SDimitry Andric 
2644824e7fdSDimitry Andric 
2654824e7fdSDimitry Andric   bool isSafeAccess(const Use &U, AllocaInst *AI, const SCEV *AccessSize);
2664824e7fdSDimitry Andric   bool isSafeAccess(const Use &U, AllocaInst *AI, Value *V);
2674824e7fdSDimitry Andric   bool isSafeAccess(const Use &U, AllocaInst *AI, TypeSize AccessSize);
2684824e7fdSDimitry Andric 
2690b57cec5SDimitry Andric public:
2705ffd83dbSDimitry Andric   StackSafetyLocalAnalysis(Function &F, ScalarEvolution &SE)
2710fca6ea1SDimitry Andric       : F(F), DL(F.getDataLayout()), SE(SE),
2720b57cec5SDimitry Andric         PointerSize(DL.getPointerSizeInBits()),
2730b57cec5SDimitry Andric         UnknownRange(PointerSize, true) {}
2740b57cec5SDimitry Andric 
2750b57cec5SDimitry Andric   // Run the transformation on the associated function.
2765ffd83dbSDimitry Andric   FunctionInfo<GlobalValue> run();
2770b57cec5SDimitry Andric };
2780b57cec5SDimitry Andric 
2790fca6ea1SDimitry Andric const SCEV *StackSafetyLocalAnalysis::getSCEVAsPointer(Value *Val) {
2800fca6ea1SDimitry Andric   Type *ValTy = Val->getType();
2810fca6ea1SDimitry Andric 
2820fca6ea1SDimitry Andric   // We don't handle targets with multiple address spaces.
2830fca6ea1SDimitry Andric   if (!ValTy->isPointerTy()) {
2840fca6ea1SDimitry Andric     auto *PtrTy = PointerType::getUnqual(SE.getContext());
2850fca6ea1SDimitry Andric     return SE.getTruncateOrZeroExtend(SE.getSCEV(Val), PtrTy);
2860fca6ea1SDimitry Andric   }
2870fca6ea1SDimitry Andric 
2880fca6ea1SDimitry Andric   if (ValTy->getPointerAddressSpace() != 0)
2890fca6ea1SDimitry Andric     return nullptr;
2900fca6ea1SDimitry Andric   return SE.getSCEV(Val);
2910fca6ea1SDimitry Andric }
2920fca6ea1SDimitry Andric 
2935ffd83dbSDimitry Andric ConstantRange StackSafetyLocalAnalysis::offsetFrom(Value *Addr, Value *Base) {
2945ffd83dbSDimitry Andric   if (!SE.isSCEVable(Addr->getType()) || !SE.isSCEVable(Base->getType()))
2950b57cec5SDimitry Andric     return UnknownRange;
2960b57cec5SDimitry Andric 
2970fca6ea1SDimitry Andric   const SCEV *AddrExp = getSCEVAsPointer(Addr);
2980fca6ea1SDimitry Andric   const SCEV *BaseExp = getSCEVAsPointer(Base);
2990fca6ea1SDimitry Andric   if (!AddrExp || !BaseExp)
3000fca6ea1SDimitry Andric     return UnknownRange;
3010fca6ea1SDimitry Andric 
3025ffd83dbSDimitry Andric   const SCEV *Diff = SE.getMinusSCEV(AddrExp, BaseExp);
303fe6060f1SDimitry Andric   if (isa<SCEVCouldNotCompute>(Diff))
304fe6060f1SDimitry Andric     return UnknownRange;
3055ffd83dbSDimitry Andric 
3065ffd83dbSDimitry Andric   ConstantRange Offset = SE.getSignedRange(Diff);
3075ffd83dbSDimitry Andric   if (isUnsafe(Offset))
3085ffd83dbSDimitry Andric     return UnknownRange;
3095ffd83dbSDimitry Andric   return Offset.sextOrTrunc(PointerSize);
3100b57cec5SDimitry Andric }
3110b57cec5SDimitry Andric 
3125ffd83dbSDimitry Andric ConstantRange
3135ffd83dbSDimitry Andric StackSafetyLocalAnalysis::getAccessRange(Value *Addr, Value *Base,
3145ffd83dbSDimitry Andric                                          const ConstantRange &SizeRange) {
3155ffd83dbSDimitry Andric   // Zero-size loads and stores do not access memory.
3165ffd83dbSDimitry Andric   if (SizeRange.isEmptySet())
3175ffd83dbSDimitry Andric     return ConstantRange::getEmpty(PointerSize);
3185ffd83dbSDimitry Andric   assert(!isUnsafe(SizeRange));
3195ffd83dbSDimitry Andric 
3205ffd83dbSDimitry Andric   ConstantRange Offsets = offsetFrom(Addr, Base);
3215ffd83dbSDimitry Andric   if (isUnsafe(Offsets))
3220b57cec5SDimitry Andric     return UnknownRange;
3230b57cec5SDimitry Andric 
3245ffd83dbSDimitry Andric   Offsets = addOverflowNever(Offsets, SizeRange);
3255ffd83dbSDimitry Andric   if (isUnsafe(Offsets))
3265ffd83dbSDimitry Andric     return UnknownRange;
3275ffd83dbSDimitry Andric   return Offsets;
3285ffd83dbSDimitry Andric }
3290b57cec5SDimitry Andric 
3305ffd83dbSDimitry Andric ConstantRange StackSafetyLocalAnalysis::getAccessRange(Value *Addr, Value *Base,
3315ffd83dbSDimitry Andric                                                        TypeSize Size) {
3325ffd83dbSDimitry Andric   if (Size.isScalable())
3335ffd83dbSDimitry Andric     return UnknownRange;
334bdd1243dSDimitry Andric   APInt APSize(PointerSize, Size.getFixedValue(), true);
3355ffd83dbSDimitry Andric   if (APSize.isNegative())
3365ffd83dbSDimitry Andric     return UnknownRange;
337349cc55cSDimitry Andric   return getAccessRange(Addr, Base,
338349cc55cSDimitry Andric                         ConstantRange(APInt::getZero(PointerSize), APSize));
3390b57cec5SDimitry Andric }
3400b57cec5SDimitry Andric 
3410b57cec5SDimitry Andric ConstantRange StackSafetyLocalAnalysis::getMemIntrinsicAccessRange(
3425ffd83dbSDimitry Andric     const MemIntrinsic *MI, const Use &U, Value *Base) {
3435ffd83dbSDimitry Andric   if (const auto *MTI = dyn_cast<MemTransferInst>(MI)) {
3440b57cec5SDimitry Andric     if (MTI->getRawSource() != U && MTI->getRawDest() != U)
3455ffd83dbSDimitry Andric       return ConstantRange::getEmpty(PointerSize);
3460b57cec5SDimitry Andric   } else {
3470b57cec5SDimitry Andric     if (MI->getRawDest() != U)
3485ffd83dbSDimitry Andric       return ConstantRange::getEmpty(PointerSize);
3490b57cec5SDimitry Andric   }
3505ffd83dbSDimitry Andric 
3515ffd83dbSDimitry Andric   auto *CalculationTy = IntegerType::getIntNTy(SE.getContext(), PointerSize);
3525ffd83dbSDimitry Andric   if (!SE.isSCEVable(MI->getLength()->getType()))
3530b57cec5SDimitry Andric     return UnknownRange;
3545ffd83dbSDimitry Andric 
3555ffd83dbSDimitry Andric   const SCEV *Expr =
3565ffd83dbSDimitry Andric       SE.getTruncateOrZeroExtend(SE.getSCEV(MI->getLength()), CalculationTy);
3575ffd83dbSDimitry Andric   ConstantRange Sizes = SE.getSignedRange(Expr);
3587a6dacacSDimitry Andric   if (!Sizes.getUpper().isStrictlyPositive() || isUnsafe(Sizes))
3595ffd83dbSDimitry Andric     return UnknownRange;
3605ffd83dbSDimitry Andric   Sizes = Sizes.sextOrTrunc(PointerSize);
361349cc55cSDimitry Andric   ConstantRange SizeRange(APInt::getZero(PointerSize), Sizes.getUpper() - 1);
3625ffd83dbSDimitry Andric   return getAccessRange(U, Base, SizeRange);
3630b57cec5SDimitry Andric }
3640b57cec5SDimitry Andric 
3654824e7fdSDimitry Andric bool StackSafetyLocalAnalysis::isSafeAccess(const Use &U, AllocaInst *AI,
3664824e7fdSDimitry Andric                                             Value *V) {
3674824e7fdSDimitry Andric   return isSafeAccess(U, AI, SE.getSCEV(V));
3684824e7fdSDimitry Andric }
3694824e7fdSDimitry Andric 
3704824e7fdSDimitry Andric bool StackSafetyLocalAnalysis::isSafeAccess(const Use &U, AllocaInst *AI,
3714824e7fdSDimitry Andric                                             TypeSize TS) {
3724824e7fdSDimitry Andric   if (TS.isScalable())
3734824e7fdSDimitry Andric     return false;
3744824e7fdSDimitry Andric   auto *CalculationTy = IntegerType::getIntNTy(SE.getContext(), PointerSize);
375bdd1243dSDimitry Andric   const SCEV *SV = SE.getConstant(CalculationTy, TS.getFixedValue());
3764824e7fdSDimitry Andric   return isSafeAccess(U, AI, SV);
3774824e7fdSDimitry Andric }
3784824e7fdSDimitry Andric 
3794824e7fdSDimitry Andric bool StackSafetyLocalAnalysis::isSafeAccess(const Use &U, AllocaInst *AI,
3804824e7fdSDimitry Andric                                             const SCEV *AccessSize) {
3814824e7fdSDimitry Andric 
3824824e7fdSDimitry Andric   if (!AI)
3835f757f3fSDimitry Andric     return true; // This only judges whether it is a safe *stack* access.
3844824e7fdSDimitry Andric   if (isa<SCEVCouldNotCompute>(AccessSize))
3854824e7fdSDimitry Andric     return false;
3864824e7fdSDimitry Andric 
3874824e7fdSDimitry Andric   const auto *I = cast<Instruction>(U.getUser());
3884824e7fdSDimitry Andric 
3890fca6ea1SDimitry Andric   const SCEV *AddrExp = getSCEVAsPointer(U.get());
3900fca6ea1SDimitry Andric   const SCEV *BaseExp = getSCEVAsPointer(AI);
3910fca6ea1SDimitry Andric   if (!AddrExp || !BaseExp)
3920fca6ea1SDimitry Andric     return false;
3934824e7fdSDimitry Andric 
3944824e7fdSDimitry Andric   const SCEV *Diff = SE.getMinusSCEV(AddrExp, BaseExp);
3954824e7fdSDimitry Andric   if (isa<SCEVCouldNotCompute>(Diff))
3964824e7fdSDimitry Andric     return false;
3974824e7fdSDimitry Andric 
3984824e7fdSDimitry Andric   auto Size = getStaticAllocaSizeRange(*AI);
3994824e7fdSDimitry Andric 
4004824e7fdSDimitry Andric   auto *CalculationTy = IntegerType::getIntNTy(SE.getContext(), PointerSize);
4014824e7fdSDimitry Andric   auto ToDiffTy = [&](const SCEV *V) {
4024824e7fdSDimitry Andric     return SE.getTruncateOrZeroExtend(V, CalculationTy);
4034824e7fdSDimitry Andric   };
4044824e7fdSDimitry Andric   const SCEV *Min = ToDiffTy(SE.getConstant(Size.getLower()));
4054824e7fdSDimitry Andric   const SCEV *Max = SE.getMinusSCEV(ToDiffTy(SE.getConstant(Size.getUpper())),
4064824e7fdSDimitry Andric                                     ToDiffTy(AccessSize));
4074824e7fdSDimitry Andric   return SE.evaluatePredicateAt(ICmpInst::Predicate::ICMP_SGE, Diff, Min, I)
40881ad6265SDimitry Andric              .value_or(false) &&
4094824e7fdSDimitry Andric          SE.evaluatePredicateAt(ICmpInst::Predicate::ICMP_SLE, Diff, Max, I)
41081ad6265SDimitry Andric              .value_or(false);
4114824e7fdSDimitry Andric }
4124824e7fdSDimitry Andric 
4130b57cec5SDimitry Andric /// The function analyzes all local uses of Ptr (alloca or argument) and
4140b57cec5SDimitry Andric /// calculates local access range and all function calls where it was used.
415349cc55cSDimitry Andric void StackSafetyLocalAnalysis::analyzeAllUses(Value *Ptr,
4165ffd83dbSDimitry Andric                                               UseInfo<GlobalValue> &US,
4175ffd83dbSDimitry Andric                                               const StackLifetime &SL) {
4180b57cec5SDimitry Andric   SmallPtrSet<const Value *, 16> Visited;
4190b57cec5SDimitry Andric   SmallVector<const Value *, 8> WorkList;
4200b57cec5SDimitry Andric   WorkList.push_back(Ptr);
4214824e7fdSDimitry Andric   AllocaInst *AI = dyn_cast<AllocaInst>(Ptr);
4220b57cec5SDimitry Andric 
4230b57cec5SDimitry Andric   // A DFS search through all uses of the alloca in bitcasts/PHI/GEPs/etc.
4240b57cec5SDimitry Andric   while (!WorkList.empty()) {
4250b57cec5SDimitry Andric     const Value *V = WorkList.pop_back_val();
4260b57cec5SDimitry Andric     for (const Use &UI : V->uses()) {
4275ffd83dbSDimitry Andric       const auto *I = cast<Instruction>(UI.getUser());
4285ffd83dbSDimitry Andric       if (!SL.isReachable(I))
4295ffd83dbSDimitry Andric         continue;
4305ffd83dbSDimitry Andric 
4310b57cec5SDimitry Andric       assert(V == UI.get());
4320b57cec5SDimitry Andric 
4335f757f3fSDimitry Andric       auto RecordStore = [&](const Value* StoredVal) {
4345f757f3fSDimitry Andric         if (V == StoredVal) {
4355f757f3fSDimitry Andric           // Stored the pointer - conservatively assume it may be unsafe.
4365f757f3fSDimitry Andric           US.addRange(I, UnknownRange, /*IsSafe=*/false);
4375f757f3fSDimitry Andric           return;
4385f757f3fSDimitry Andric         }
4395f757f3fSDimitry Andric         if (AI && !SL.isAliveAfter(AI, I)) {
4405f757f3fSDimitry Andric           US.addRange(I, UnknownRange, /*IsSafe=*/false);
4415f757f3fSDimitry Andric           return;
4425f757f3fSDimitry Andric         }
4435f757f3fSDimitry Andric         auto TypeSize = DL.getTypeStoreSize(StoredVal->getType());
4445f757f3fSDimitry Andric         auto AccessRange = getAccessRange(UI, Ptr, TypeSize);
4455f757f3fSDimitry Andric         bool Safe = isSafeAccess(UI, AI, TypeSize);
4465f757f3fSDimitry Andric         US.addRange(I, AccessRange, Safe);
4475f757f3fSDimitry Andric         return;
4485f757f3fSDimitry Andric       };
4495f757f3fSDimitry Andric 
4500b57cec5SDimitry Andric       switch (I->getOpcode()) {
4510b57cec5SDimitry Andric       case Instruction::Load: {
4525ffd83dbSDimitry Andric         if (AI && !SL.isAliveAfter(AI, I)) {
4534824e7fdSDimitry Andric           US.addRange(I, UnknownRange, /*IsSafe=*/false);
454349cc55cSDimitry Andric           break;
4555ffd83dbSDimitry Andric         }
4564824e7fdSDimitry Andric         auto TypeSize = DL.getTypeStoreSize(I->getType());
4574824e7fdSDimitry Andric         auto AccessRange = getAccessRange(UI, Ptr, TypeSize);
4584824e7fdSDimitry Andric         bool Safe = isSafeAccess(UI, AI, TypeSize);
4594824e7fdSDimitry Andric         US.addRange(I, AccessRange, Safe);
4600b57cec5SDimitry Andric         break;
4610b57cec5SDimitry Andric       }
4620b57cec5SDimitry Andric 
4630b57cec5SDimitry Andric       case Instruction::VAArg:
4640b57cec5SDimitry Andric         // "va-arg" from a pointer is safe.
4650b57cec5SDimitry Andric         break;
4665f757f3fSDimitry Andric       case Instruction::Store:
4675f757f3fSDimitry Andric         RecordStore(cast<StoreInst>(I)->getValueOperand());
468349cc55cSDimitry Andric         break;
4695f757f3fSDimitry Andric       case Instruction::AtomicCmpXchg:
4705f757f3fSDimitry Andric         RecordStore(cast<AtomicCmpXchgInst>(I)->getNewValOperand());
471349cc55cSDimitry Andric         break;
4725f757f3fSDimitry Andric       case Instruction::AtomicRMW:
4735f757f3fSDimitry Andric         RecordStore(cast<AtomicRMWInst>(I)->getValOperand());
4740b57cec5SDimitry Andric         break;
4750b57cec5SDimitry Andric 
4760b57cec5SDimitry Andric       case Instruction::Ret:
4770b57cec5SDimitry Andric         // Information leak.
4780b57cec5SDimitry Andric         // FIXME: Process parameters correctly. This is a leak only if we return
4790b57cec5SDimitry Andric         // alloca.
4804824e7fdSDimitry Andric         US.addRange(I, UnknownRange, /*IsSafe=*/false);
481349cc55cSDimitry Andric         break;
4820b57cec5SDimitry Andric 
4830b57cec5SDimitry Andric       case Instruction::Call:
4840b57cec5SDimitry Andric       case Instruction::Invoke: {
4850b57cec5SDimitry Andric         if (I->isLifetimeStartOrEnd())
4860b57cec5SDimitry Andric           break;
4870b57cec5SDimitry Andric 
4885ffd83dbSDimitry Andric         if (AI && !SL.isAliveAfter(AI, I)) {
4894824e7fdSDimitry Andric           US.addRange(I, UnknownRange, /*IsSafe=*/false);
490349cc55cSDimitry Andric           break;
4915ffd83dbSDimitry Andric         }
4920b57cec5SDimitry Andric         if (const MemIntrinsic *MI = dyn_cast<MemIntrinsic>(I)) {
4934824e7fdSDimitry Andric           auto AccessRange = getMemIntrinsicAccessRange(MI, UI, Ptr);
4944824e7fdSDimitry Andric           bool Safe = false;
4954824e7fdSDimitry Andric           if (const auto *MTI = dyn_cast<MemTransferInst>(MI)) {
4964824e7fdSDimitry Andric             if (MTI->getRawSource() != UI && MTI->getRawDest() != UI)
4974824e7fdSDimitry Andric               Safe = true;
4984824e7fdSDimitry Andric           } else if (MI->getRawDest() != UI) {
4994824e7fdSDimitry Andric             Safe = true;
5004824e7fdSDimitry Andric           }
5014824e7fdSDimitry Andric           Safe = Safe || isSafeAccess(UI, AI, MI->getLength());
5024824e7fdSDimitry Andric           US.addRange(I, AccessRange, Safe);
5030b57cec5SDimitry Andric           break;
5040b57cec5SDimitry Andric         }
5050b57cec5SDimitry Andric 
5065ffd83dbSDimitry Andric         const auto &CB = cast<CallBase>(*I);
507349cc55cSDimitry Andric         if (CB.getReturnedArgOperand() == V) {
508349cc55cSDimitry Andric           if (Visited.insert(I).second)
509349cc55cSDimitry Andric             WorkList.push_back(cast<const Instruction>(I));
510349cc55cSDimitry Andric         }
511349cc55cSDimitry Andric 
5125ffd83dbSDimitry Andric         if (!CB.isArgOperand(&UI)) {
5134824e7fdSDimitry Andric           US.addRange(I, UnknownRange, /*IsSafe=*/false);
514349cc55cSDimitry Andric           break;
5155ffd83dbSDimitry Andric         }
5165ffd83dbSDimitry Andric 
5175ffd83dbSDimitry Andric         unsigned ArgNo = CB.getArgOperandNo(&UI);
5185ffd83dbSDimitry Andric         if (CB.isByValArgument(ArgNo)) {
5194824e7fdSDimitry Andric           auto TypeSize = DL.getTypeStoreSize(CB.getParamByValType(ArgNo));
5204824e7fdSDimitry Andric           auto AccessRange = getAccessRange(UI, Ptr, TypeSize);
5214824e7fdSDimitry Andric           bool Safe = isSafeAccess(UI, AI, TypeSize);
5224824e7fdSDimitry Andric           US.addRange(I, AccessRange, Safe);
5235ffd83dbSDimitry Andric           break;
5245ffd83dbSDimitry Andric         }
5255ffd83dbSDimitry Andric 
5260b57cec5SDimitry Andric         // FIXME: consult devirt?
5270b57cec5SDimitry Andric         // Do not follow aliases, otherwise we could inadvertently follow
5280b57cec5SDimitry Andric         // dso_preemptable aliases or aliases with interposable linkage.
5298bcb0991SDimitry Andric         const GlobalValue *Callee =
5305ffd83dbSDimitry Andric             dyn_cast<GlobalValue>(CB.getCalledOperand()->stripPointerCasts());
531*f3457ed9SDimitry Andric         if (!Callee || isa<GlobalIFunc>(Callee)) {
5324824e7fdSDimitry Andric           US.addRange(I, UnknownRange, /*IsSafe=*/false);
533349cc55cSDimitry Andric           break;
5340b57cec5SDimitry Andric         }
5350b57cec5SDimitry Andric 
5360b57cec5SDimitry Andric         assert(isa<Function>(Callee) || isa<GlobalAlias>(Callee));
537e8d8bef9SDimitry Andric         ConstantRange Offsets = offsetFrom(UI, Ptr);
538e8d8bef9SDimitry Andric         auto Insert =
539e8d8bef9SDimitry Andric             US.Calls.emplace(CallInfo<GlobalValue>(Callee, ArgNo), Offsets);
540e8d8bef9SDimitry Andric         if (!Insert.second)
541e8d8bef9SDimitry Andric           Insert.first->second = Insert.first->second.unionWith(Offsets);
5420b57cec5SDimitry Andric         break;
5430b57cec5SDimitry Andric       }
5440b57cec5SDimitry Andric 
5450b57cec5SDimitry Andric       default:
5460b57cec5SDimitry Andric         if (Visited.insert(I).second)
5470b57cec5SDimitry Andric           WorkList.push_back(cast<const Instruction>(I));
5480b57cec5SDimitry Andric       }
5490b57cec5SDimitry Andric     }
5500b57cec5SDimitry Andric   }
5510b57cec5SDimitry Andric }
5520b57cec5SDimitry Andric 
5535ffd83dbSDimitry Andric FunctionInfo<GlobalValue> StackSafetyLocalAnalysis::run() {
5545ffd83dbSDimitry Andric   FunctionInfo<GlobalValue> Info;
5550b57cec5SDimitry Andric   assert(!F.isDeclaration() &&
5560b57cec5SDimitry Andric          "Can't run StackSafety on a function declaration");
5570b57cec5SDimitry Andric 
5580b57cec5SDimitry Andric   LLVM_DEBUG(dbgs() << "[StackSafety] " << F.getName() << "\n");
5590b57cec5SDimitry Andric 
5605ffd83dbSDimitry Andric   SmallVector<AllocaInst *, 64> Allocas;
5615ffd83dbSDimitry Andric   for (auto &I : instructions(F))
5625ffd83dbSDimitry Andric     if (auto *AI = dyn_cast<AllocaInst>(&I))
5635ffd83dbSDimitry Andric       Allocas.push_back(AI);
5645ffd83dbSDimitry Andric   StackLifetime SL(F, Allocas, StackLifetime::LivenessType::Must);
5655ffd83dbSDimitry Andric   SL.run();
5665ffd83dbSDimitry Andric 
5675ffd83dbSDimitry Andric   for (auto *AI : Allocas) {
5685ffd83dbSDimitry Andric     auto &UI = Info.Allocas.emplace(AI, PointerSize).first->second;
5695ffd83dbSDimitry Andric     analyzeAllUses(AI, UI, SL);
5705ffd83dbSDimitry Andric   }
5715ffd83dbSDimitry Andric 
572e8d8bef9SDimitry Andric   for (Argument &A : F.args()) {
5735ffd83dbSDimitry Andric     // Non pointers and bypass arguments are not going to be used in any global
5745ffd83dbSDimitry Andric     // processing.
5755ffd83dbSDimitry Andric     if (A.getType()->isPointerTy() && !A.hasByValAttr()) {
5765ffd83dbSDimitry Andric       auto &UI = Info.Params.emplace(A.getArgNo(), PointerSize).first->second;
5775ffd83dbSDimitry Andric       analyzeAllUses(&A, UI, SL);
5780b57cec5SDimitry Andric     }
5790b57cec5SDimitry Andric   }
5800b57cec5SDimitry Andric 
5815ffd83dbSDimitry Andric   LLVM_DEBUG(Info.print(dbgs(), F.getName(), &F));
582349cc55cSDimitry Andric   LLVM_DEBUG(dbgs() << "\n[StackSafety] done\n");
5835ffd83dbSDimitry Andric   return Info;
5840b57cec5SDimitry Andric }
5850b57cec5SDimitry Andric 
5865ffd83dbSDimitry Andric template <typename CalleeTy> class StackSafetyDataFlowAnalysis {
5875ffd83dbSDimitry Andric   using FunctionMap = std::map<const CalleeTy *, FunctionInfo<CalleeTy>>;
5880b57cec5SDimitry Andric 
5890b57cec5SDimitry Andric   FunctionMap Functions;
5900b57cec5SDimitry Andric   const ConstantRange UnknownRange;
5910b57cec5SDimitry Andric 
5925ffd83dbSDimitry Andric   // Callee-to-Caller multimap.
5935ffd83dbSDimitry Andric   DenseMap<const CalleeTy *, SmallVector<const CalleeTy *, 4>> Callers;
5945ffd83dbSDimitry Andric   SetVector<const CalleeTy *> WorkList;
5955ffd83dbSDimitry Andric 
5965ffd83dbSDimitry Andric   bool updateOneUse(UseInfo<CalleeTy> &US, bool UpdateToFullSet);
5975ffd83dbSDimitry Andric   void updateOneNode(const CalleeTy *Callee, FunctionInfo<CalleeTy> &FS);
5985ffd83dbSDimitry Andric   void updateOneNode(const CalleeTy *Callee) {
5990b57cec5SDimitry Andric     updateOneNode(Callee, Functions.find(Callee)->second);
6000b57cec5SDimitry Andric   }
6010b57cec5SDimitry Andric   void updateAllNodes() {
6020b57cec5SDimitry Andric     for (auto &F : Functions)
6030b57cec5SDimitry Andric       updateOneNode(F.first, F.second);
6040b57cec5SDimitry Andric   }
6050b57cec5SDimitry Andric   void runDataFlow();
6060b57cec5SDimitry Andric #ifndef NDEBUG
6070b57cec5SDimitry Andric   void verifyFixedPoint();
6080b57cec5SDimitry Andric #endif
6090b57cec5SDimitry Andric 
6100b57cec5SDimitry Andric public:
6115ffd83dbSDimitry Andric   StackSafetyDataFlowAnalysis(uint32_t PointerBitWidth, FunctionMap Functions)
6125ffd83dbSDimitry Andric       : Functions(std::move(Functions)),
6135ffd83dbSDimitry Andric         UnknownRange(ConstantRange::getFull(PointerBitWidth)) {}
6145ffd83dbSDimitry Andric 
6155ffd83dbSDimitry Andric   const FunctionMap &run();
6165ffd83dbSDimitry Andric 
6175ffd83dbSDimitry Andric   ConstantRange getArgumentAccessRange(const CalleeTy *Callee, unsigned ParamNo,
6185ffd83dbSDimitry Andric                                        const ConstantRange &Offsets) const;
6190b57cec5SDimitry Andric };
6200b57cec5SDimitry Andric 
6215ffd83dbSDimitry Andric template <typename CalleeTy>
6225ffd83dbSDimitry Andric ConstantRange StackSafetyDataFlowAnalysis<CalleeTy>::getArgumentAccessRange(
6235ffd83dbSDimitry Andric     const CalleeTy *Callee, unsigned ParamNo,
6245ffd83dbSDimitry Andric     const ConstantRange &Offsets) const {
6255ffd83dbSDimitry Andric   auto FnIt = Functions.find(Callee);
6260b57cec5SDimitry Andric   // Unknown callee (outside of LTO domain or an indirect call).
6275ffd83dbSDimitry Andric   if (FnIt == Functions.end())
6280b57cec5SDimitry Andric     return UnknownRange;
6295ffd83dbSDimitry Andric   auto &FS = FnIt->second;
6305ffd83dbSDimitry Andric   auto ParamIt = FS.Params.find(ParamNo);
6315ffd83dbSDimitry Andric   if (ParamIt == FS.Params.end())
6320b57cec5SDimitry Andric     return UnknownRange;
6335ffd83dbSDimitry Andric   auto &Access = ParamIt->second.Range;
6345ffd83dbSDimitry Andric   if (Access.isEmptySet())
6355ffd83dbSDimitry Andric     return Access;
6365ffd83dbSDimitry Andric   if (Access.isFullSet())
6370b57cec5SDimitry Andric     return UnknownRange;
6385ffd83dbSDimitry Andric   return addOverflowNever(Access, Offsets);
6390b57cec5SDimitry Andric }
6400b57cec5SDimitry Andric 
6415ffd83dbSDimitry Andric template <typename CalleeTy>
6425ffd83dbSDimitry Andric bool StackSafetyDataFlowAnalysis<CalleeTy>::updateOneUse(UseInfo<CalleeTy> &US,
6430b57cec5SDimitry Andric                                                          bool UpdateToFullSet) {
6440b57cec5SDimitry Andric   bool Changed = false;
645e8d8bef9SDimitry Andric   for (auto &KV : US.Calls) {
646e8d8bef9SDimitry Andric     assert(!KV.second.isEmptySet() &&
6470b57cec5SDimitry Andric            "Param range can't be empty-set, invalid offset range");
6480b57cec5SDimitry Andric 
6495ffd83dbSDimitry Andric     ConstantRange CalleeRange =
650e8d8bef9SDimitry Andric         getArgumentAccessRange(KV.first.Callee, KV.first.ParamNo, KV.second);
6510b57cec5SDimitry Andric     if (!US.Range.contains(CalleeRange)) {
6520b57cec5SDimitry Andric       Changed = true;
6530b57cec5SDimitry Andric       if (UpdateToFullSet)
6540b57cec5SDimitry Andric         US.Range = UnknownRange;
6550b57cec5SDimitry Andric       else
656e8d8bef9SDimitry Andric         US.updateRange(CalleeRange);
6570b57cec5SDimitry Andric     }
6580b57cec5SDimitry Andric   }
6590b57cec5SDimitry Andric   return Changed;
6600b57cec5SDimitry Andric }
6610b57cec5SDimitry Andric 
6625ffd83dbSDimitry Andric template <typename CalleeTy>
6635ffd83dbSDimitry Andric void StackSafetyDataFlowAnalysis<CalleeTy>::updateOneNode(
6645ffd83dbSDimitry Andric     const CalleeTy *Callee, FunctionInfo<CalleeTy> &FS) {
6650b57cec5SDimitry Andric   bool UpdateToFullSet = FS.UpdateCount > StackSafetyMaxIterations;
6660b57cec5SDimitry Andric   bool Changed = false;
6675ffd83dbSDimitry Andric   for (auto &KV : FS.Params)
6685ffd83dbSDimitry Andric     Changed |= updateOneUse(KV.second, UpdateToFullSet);
6690b57cec5SDimitry Andric 
6700b57cec5SDimitry Andric   if (Changed) {
6710b57cec5SDimitry Andric     LLVM_DEBUG(dbgs() << "=== update [" << FS.UpdateCount
6725ffd83dbSDimitry Andric                       << (UpdateToFullSet ? ", full-set" : "") << "] " << &FS
6735ffd83dbSDimitry Andric                       << "\n");
6740b57cec5SDimitry Andric     // Callers of this function may need updating.
6750b57cec5SDimitry Andric     for (auto &CallerID : Callers[Callee])
6760b57cec5SDimitry Andric       WorkList.insert(CallerID);
6770b57cec5SDimitry Andric 
6780b57cec5SDimitry Andric     ++FS.UpdateCount;
6790b57cec5SDimitry Andric   }
6800b57cec5SDimitry Andric }
6810b57cec5SDimitry Andric 
6825ffd83dbSDimitry Andric template <typename CalleeTy>
6835ffd83dbSDimitry Andric void StackSafetyDataFlowAnalysis<CalleeTy>::runDataFlow() {
6845ffd83dbSDimitry Andric   SmallVector<const CalleeTy *, 16> Callees;
6850b57cec5SDimitry Andric   for (auto &F : Functions) {
6860b57cec5SDimitry Andric     Callees.clear();
6875ffd83dbSDimitry Andric     auto &FS = F.second;
6885ffd83dbSDimitry Andric     for (auto &KV : FS.Params)
6895ffd83dbSDimitry Andric       for (auto &CS : KV.second.Calls)
690e8d8bef9SDimitry Andric         Callees.push_back(CS.first.Callee);
6910b57cec5SDimitry Andric 
6920b57cec5SDimitry Andric     llvm::sort(Callees);
6930fca6ea1SDimitry Andric     Callees.erase(llvm::unique(Callees), Callees.end());
6940b57cec5SDimitry Andric 
6950b57cec5SDimitry Andric     for (auto &Callee : Callees)
6960b57cec5SDimitry Andric       Callers[Callee].push_back(F.first);
6970b57cec5SDimitry Andric   }
6980b57cec5SDimitry Andric 
6990b57cec5SDimitry Andric   updateAllNodes();
7000b57cec5SDimitry Andric 
7010b57cec5SDimitry Andric   while (!WorkList.empty()) {
702349cc55cSDimitry Andric     const CalleeTy *Callee = WorkList.pop_back_val();
7030b57cec5SDimitry Andric     updateOneNode(Callee);
7040b57cec5SDimitry Andric   }
7050b57cec5SDimitry Andric }
7060b57cec5SDimitry Andric 
7070b57cec5SDimitry Andric #ifndef NDEBUG
7085ffd83dbSDimitry Andric template <typename CalleeTy>
7095ffd83dbSDimitry Andric void StackSafetyDataFlowAnalysis<CalleeTy>::verifyFixedPoint() {
7100b57cec5SDimitry Andric   WorkList.clear();
7110b57cec5SDimitry Andric   updateAllNodes();
7120b57cec5SDimitry Andric   assert(WorkList.empty());
7130b57cec5SDimitry Andric }
7140b57cec5SDimitry Andric #endif
7150b57cec5SDimitry Andric 
7165ffd83dbSDimitry Andric template <typename CalleeTy>
7175ffd83dbSDimitry Andric const typename StackSafetyDataFlowAnalysis<CalleeTy>::FunctionMap &
7185ffd83dbSDimitry Andric StackSafetyDataFlowAnalysis<CalleeTy>::run() {
7190b57cec5SDimitry Andric   runDataFlow();
7200b57cec5SDimitry Andric   LLVM_DEBUG(verifyFixedPoint());
7215ffd83dbSDimitry Andric   return Functions;
7225ffd83dbSDimitry Andric }
7230b57cec5SDimitry Andric 
724e8d8bef9SDimitry Andric FunctionSummary *findCalleeFunctionSummary(ValueInfo VI, StringRef ModuleId) {
725e8d8bef9SDimitry Andric   if (!VI)
726e8d8bef9SDimitry Andric     return nullptr;
727e8d8bef9SDimitry Andric   auto SummaryList = VI.getSummaryList();
728e8d8bef9SDimitry Andric   GlobalValueSummary* S = nullptr;
729e8d8bef9SDimitry Andric   for (const auto& GVS : SummaryList) {
730e8d8bef9SDimitry Andric     if (!GVS->isLive())
731e8d8bef9SDimitry Andric       continue;
732e8d8bef9SDimitry Andric     if (const AliasSummary *AS = dyn_cast<AliasSummary>(GVS.get()))
733e8d8bef9SDimitry Andric       if (!AS->hasAliasee())
734e8d8bef9SDimitry Andric         continue;
735e8d8bef9SDimitry Andric     if (!isa<FunctionSummary>(GVS->getBaseObject()))
736e8d8bef9SDimitry Andric       continue;
737e8d8bef9SDimitry Andric     if (GlobalValue::isLocalLinkage(GVS->linkage())) {
738e8d8bef9SDimitry Andric       if (GVS->modulePath() == ModuleId) {
739e8d8bef9SDimitry Andric         S = GVS.get();
740e8d8bef9SDimitry Andric         break;
741e8d8bef9SDimitry Andric       }
742e8d8bef9SDimitry Andric     } else if (GlobalValue::isExternalLinkage(GVS->linkage())) {
743e8d8bef9SDimitry Andric       if (S) {
744e8d8bef9SDimitry Andric         ++NumIndexCalleeMultipleExternal;
745e8d8bef9SDimitry Andric         return nullptr;
746e8d8bef9SDimitry Andric       }
747e8d8bef9SDimitry Andric       S = GVS.get();
748e8d8bef9SDimitry Andric     } else if (GlobalValue::isWeakLinkage(GVS->linkage())) {
749e8d8bef9SDimitry Andric       if (S) {
750e8d8bef9SDimitry Andric         ++NumIndexCalleeMultipleWeak;
751e8d8bef9SDimitry Andric         return nullptr;
752e8d8bef9SDimitry Andric       }
753e8d8bef9SDimitry Andric       S = GVS.get();
754e8d8bef9SDimitry Andric     } else if (GlobalValue::isAvailableExternallyLinkage(GVS->linkage()) ||
755e8d8bef9SDimitry Andric                GlobalValue::isLinkOnceLinkage(GVS->linkage())) {
756e8d8bef9SDimitry Andric       if (SummaryList.size() == 1)
757e8d8bef9SDimitry Andric         S = GVS.get();
758e8d8bef9SDimitry Andric       // According thinLTOResolvePrevailingGUID these are unlikely prevailing.
759e8d8bef9SDimitry Andric     } else {
760e8d8bef9SDimitry Andric       ++NumIndexCalleeUnhandled;
761e8d8bef9SDimitry Andric     }
762e8d8bef9SDimitry Andric   };
7635ffd83dbSDimitry Andric   while (S) {
7645ffd83dbSDimitry Andric     if (!S->isLive() || !S->isDSOLocal())
7655ffd83dbSDimitry Andric       return nullptr;
7665ffd83dbSDimitry Andric     if (FunctionSummary *FS = dyn_cast<FunctionSummary>(S))
7675ffd83dbSDimitry Andric       return FS;
7685ffd83dbSDimitry Andric     AliasSummary *AS = dyn_cast<AliasSummary>(S);
769e8d8bef9SDimitry Andric     if (!AS || !AS->hasAliasee())
7705ffd83dbSDimitry Andric       return nullptr;
7715ffd83dbSDimitry Andric     S = AS->getBaseObject();
7725ffd83dbSDimitry Andric     if (S == AS)
7735ffd83dbSDimitry Andric       return nullptr;
7745ffd83dbSDimitry Andric   }
7755ffd83dbSDimitry Andric   return nullptr;
7765ffd83dbSDimitry Andric }
7775ffd83dbSDimitry Andric 
7785ffd83dbSDimitry Andric const Function *findCalleeInModule(const GlobalValue *GV) {
7795ffd83dbSDimitry Andric   while (GV) {
7805ffd83dbSDimitry Andric     if (GV->isDeclaration() || GV->isInterposable() || !GV->isDSOLocal())
7815ffd83dbSDimitry Andric       return nullptr;
7825ffd83dbSDimitry Andric     if (const Function *F = dyn_cast<Function>(GV))
7835ffd83dbSDimitry Andric       return F;
7845ffd83dbSDimitry Andric     const GlobalAlias *A = dyn_cast<GlobalAlias>(GV);
7855ffd83dbSDimitry Andric     if (!A)
7865ffd83dbSDimitry Andric       return nullptr;
787349cc55cSDimitry Andric     GV = A->getAliaseeObject();
7885ffd83dbSDimitry Andric     if (GV == A)
7895ffd83dbSDimitry Andric       return nullptr;
7905ffd83dbSDimitry Andric   }
7915ffd83dbSDimitry Andric   return nullptr;
7925ffd83dbSDimitry Andric }
7935ffd83dbSDimitry Andric 
7945ffd83dbSDimitry Andric const ConstantRange *findParamAccess(const FunctionSummary &FS,
7955ffd83dbSDimitry Andric                                      uint32_t ParamNo) {
7965ffd83dbSDimitry Andric   assert(FS.isLive());
7975ffd83dbSDimitry Andric   assert(FS.isDSOLocal());
798fcaf7f86SDimitry Andric   for (const auto &PS : FS.paramAccesses())
7995ffd83dbSDimitry Andric     if (ParamNo == PS.ParamNo)
8005ffd83dbSDimitry Andric       return &PS.Use;
8015ffd83dbSDimitry Andric   return nullptr;
8025ffd83dbSDimitry Andric }
8035ffd83dbSDimitry Andric 
8045ffd83dbSDimitry Andric void resolveAllCalls(UseInfo<GlobalValue> &Use,
8055ffd83dbSDimitry Andric                      const ModuleSummaryIndex *Index) {
8065ffd83dbSDimitry Andric   ConstantRange FullSet(Use.Range.getBitWidth(), true);
807e8d8bef9SDimitry Andric   // Move Use.Calls to a temp storage and repopulate - don't use std::move as it
808e8d8bef9SDimitry Andric   // leaves Use.Calls in an undefined state.
809e8d8bef9SDimitry Andric   UseInfo<GlobalValue>::CallsTy TmpCalls;
810e8d8bef9SDimitry Andric   std::swap(TmpCalls, Use.Calls);
811e8d8bef9SDimitry Andric   for (const auto &C : TmpCalls) {
812e8d8bef9SDimitry Andric     const Function *F = findCalleeInModule(C.first.Callee);
8135ffd83dbSDimitry Andric     if (F) {
814e8d8bef9SDimitry Andric       Use.Calls.emplace(CallInfo<GlobalValue>(F, C.first.ParamNo), C.second);
8155ffd83dbSDimitry Andric       continue;
8165ffd83dbSDimitry Andric     }
8175ffd83dbSDimitry Andric 
8185ffd83dbSDimitry Andric     if (!Index)
8195ffd83dbSDimitry Andric       return Use.updateRange(FullSet);
820e8d8bef9SDimitry Andric     FunctionSummary *FS =
821e8d8bef9SDimitry Andric         findCalleeFunctionSummary(Index->getValueInfo(C.first.Callee->getGUID()),
822e8d8bef9SDimitry Andric                                   C.first.Callee->getParent()->getModuleIdentifier());
823e8d8bef9SDimitry Andric     ++NumModuleCalleeLookupTotal;
824e8d8bef9SDimitry Andric     if (!FS) {
825e8d8bef9SDimitry Andric       ++NumModuleCalleeLookupFailed;
8265ffd83dbSDimitry Andric       return Use.updateRange(FullSet);
827e8d8bef9SDimitry Andric     }
828e8d8bef9SDimitry Andric     const ConstantRange *Found = findParamAccess(*FS, C.first.ParamNo);
829e8d8bef9SDimitry Andric     if (!Found || Found->isFullSet())
8305ffd83dbSDimitry Andric       return Use.updateRange(FullSet);
8315ffd83dbSDimitry Andric     ConstantRange Access = Found->sextOrTrunc(Use.Range.getBitWidth());
832e8d8bef9SDimitry Andric     if (!Access.isEmptySet())
833e8d8bef9SDimitry Andric       Use.updateRange(addOverflowNever(Access, C.second));
8345ffd83dbSDimitry Andric   }
8355ffd83dbSDimitry Andric }
8365ffd83dbSDimitry Andric 
8375ffd83dbSDimitry Andric GVToSSI createGlobalStackSafetyInfo(
8385ffd83dbSDimitry Andric     std::map<const GlobalValue *, FunctionInfo<GlobalValue>> Functions,
8395ffd83dbSDimitry Andric     const ModuleSummaryIndex *Index) {
8405ffd83dbSDimitry Andric   GVToSSI SSI;
8415ffd83dbSDimitry Andric   if (Functions.empty())
8420b57cec5SDimitry Andric     return SSI;
8435ffd83dbSDimitry Andric 
8445ffd83dbSDimitry Andric   // FIXME: Simplify printing and remove copying here.
8455ffd83dbSDimitry Andric   auto Copy = Functions;
8465ffd83dbSDimitry Andric 
8475ffd83dbSDimitry Andric   for (auto &FnKV : Copy)
848e8d8bef9SDimitry Andric     for (auto &KV : FnKV.second.Params) {
8495ffd83dbSDimitry Andric       resolveAllCalls(KV.second, Index);
850e8d8bef9SDimitry Andric       if (KV.second.Range.isFullSet())
851e8d8bef9SDimitry Andric         KV.second.Calls.clear();
852e8d8bef9SDimitry Andric     }
8535ffd83dbSDimitry Andric 
854349cc55cSDimitry Andric   uint32_t PointerSize =
8550fca6ea1SDimitry Andric       Copy.begin()->first->getDataLayout().getPointerSizeInBits();
8565ffd83dbSDimitry Andric   StackSafetyDataFlowAnalysis<GlobalValue> SSDFA(PointerSize, std::move(Copy));
8575ffd83dbSDimitry Andric 
858fcaf7f86SDimitry Andric   for (const auto &F : SSDFA.run()) {
8595ffd83dbSDimitry Andric     auto FI = F.second;
8605ffd83dbSDimitry Andric     auto &SrcF = Functions[F.first];
8615ffd83dbSDimitry Andric     for (auto &KV : FI.Allocas) {
8625ffd83dbSDimitry Andric       auto &A = KV.second;
8635ffd83dbSDimitry Andric       resolveAllCalls(A, Index);
8645ffd83dbSDimitry Andric       for (auto &C : A.Calls) {
865e8d8bef9SDimitry Andric         A.updateRange(SSDFA.getArgumentAccessRange(C.first.Callee,
866e8d8bef9SDimitry Andric                                                    C.first.ParamNo, C.second));
8675ffd83dbSDimitry Andric       }
8685ffd83dbSDimitry Andric       // FIXME: This is needed only to preserve calls in print() results.
8695ffd83dbSDimitry Andric       A.Calls = SrcF.Allocas.find(KV.first)->second.Calls;
8705ffd83dbSDimitry Andric     }
8715ffd83dbSDimitry Andric     for (auto &KV : FI.Params) {
8725ffd83dbSDimitry Andric       auto &P = KV.second;
8735ffd83dbSDimitry Andric       P.Calls = SrcF.Params.find(KV.first)->second.Calls;
8745ffd83dbSDimitry Andric     }
8755ffd83dbSDimitry Andric     SSI[F.first] = std::move(FI);
8760b57cec5SDimitry Andric   }
8770b57cec5SDimitry Andric 
8785ffd83dbSDimitry Andric   return SSI;
8790b57cec5SDimitry Andric }
8800b57cec5SDimitry Andric 
8810b57cec5SDimitry Andric } // end anonymous namespace
8820b57cec5SDimitry Andric 
8830b57cec5SDimitry Andric StackSafetyInfo::StackSafetyInfo() = default;
8840b57cec5SDimitry Andric 
8855ffd83dbSDimitry Andric StackSafetyInfo::StackSafetyInfo(Function *F,
8865ffd83dbSDimitry Andric                                  std::function<ScalarEvolution &()> GetSE)
8875ffd83dbSDimitry Andric     : F(F), GetSE(GetSE) {}
8885ffd83dbSDimitry Andric 
8895ffd83dbSDimitry Andric StackSafetyInfo::StackSafetyInfo(StackSafetyInfo &&) = default;
8905ffd83dbSDimitry Andric 
8915ffd83dbSDimitry Andric StackSafetyInfo &StackSafetyInfo::operator=(StackSafetyInfo &&) = default;
8920b57cec5SDimitry Andric 
8930b57cec5SDimitry Andric StackSafetyInfo::~StackSafetyInfo() = default;
8940b57cec5SDimitry Andric 
8955ffd83dbSDimitry Andric const StackSafetyInfo::InfoTy &StackSafetyInfo::getInfo() const {
8965ffd83dbSDimitry Andric   if (!Info) {
8975ffd83dbSDimitry Andric     StackSafetyLocalAnalysis SSLA(*F, GetSE());
8985ffd83dbSDimitry Andric     Info.reset(new InfoTy{SSLA.run()});
8995ffd83dbSDimitry Andric   }
9005ffd83dbSDimitry Andric   return *Info;
9015ffd83dbSDimitry Andric }
9025ffd83dbSDimitry Andric 
9035ffd83dbSDimitry Andric void StackSafetyInfo::print(raw_ostream &O) const {
9045ffd83dbSDimitry Andric   getInfo().Info.print(O, F->getName(), dyn_cast<Function>(F));
905349cc55cSDimitry Andric   O << "\n";
9065ffd83dbSDimitry Andric }
9075ffd83dbSDimitry Andric 
9085ffd83dbSDimitry Andric const StackSafetyGlobalInfo::InfoTy &StackSafetyGlobalInfo::getInfo() const {
9095ffd83dbSDimitry Andric   if (!Info) {
9105ffd83dbSDimitry Andric     std::map<const GlobalValue *, FunctionInfo<GlobalValue>> Functions;
9115ffd83dbSDimitry Andric     for (auto &F : M->functions()) {
9125ffd83dbSDimitry Andric       if (!F.isDeclaration()) {
9135ffd83dbSDimitry Andric         auto FI = GetSSI(F).getInfo().Info;
9145ffd83dbSDimitry Andric         Functions.emplace(&F, std::move(FI));
9155ffd83dbSDimitry Andric       }
9165ffd83dbSDimitry Andric     }
9175ffd83dbSDimitry Andric     Info.reset(new InfoTy{
918349cc55cSDimitry Andric         createGlobalStackSafetyInfo(std::move(Functions), Index), {}, {}});
919349cc55cSDimitry Andric 
9205ffd83dbSDimitry Andric     for (auto &FnKV : Info->Info) {
9215ffd83dbSDimitry Andric       for (auto &KV : FnKV.second.Allocas) {
9225ffd83dbSDimitry Andric         ++NumAllocaTotal;
9235ffd83dbSDimitry Andric         const AllocaInst *AI = KV.first;
924349cc55cSDimitry Andric         auto AIRange = getStaticAllocaSizeRange(*AI);
925349cc55cSDimitry Andric         if (AIRange.contains(KV.second.Range)) {
9265ffd83dbSDimitry Andric           Info->SafeAllocas.insert(AI);
9275ffd83dbSDimitry Andric           ++NumAllocaStackSafe;
9285ffd83dbSDimitry Andric         }
9294824e7fdSDimitry Andric         Info->UnsafeAccesses.insert(KV.second.UnsafeAccesses.begin(),
9304824e7fdSDimitry Andric                                     KV.second.UnsafeAccesses.end());
9315ffd83dbSDimitry Andric       }
9325ffd83dbSDimitry Andric     }
933349cc55cSDimitry Andric 
9345ffd83dbSDimitry Andric     if (StackSafetyPrint)
9355ffd83dbSDimitry Andric       print(errs());
9365ffd83dbSDimitry Andric   }
9375ffd83dbSDimitry Andric   return *Info;
9385ffd83dbSDimitry Andric }
9395ffd83dbSDimitry Andric 
9405ffd83dbSDimitry Andric std::vector<FunctionSummary::ParamAccess>
941e8d8bef9SDimitry Andric StackSafetyInfo::getParamAccesses(ModuleSummaryIndex &Index) const {
9425ffd83dbSDimitry Andric   // Implementation transforms internal representation of parameter information
9435ffd83dbSDimitry Andric   // into FunctionSummary format.
9445ffd83dbSDimitry Andric   std::vector<FunctionSummary::ParamAccess> ParamAccesses;
9455ffd83dbSDimitry Andric   for (const auto &KV : getInfo().Info.Params) {
9465ffd83dbSDimitry Andric     auto &PS = KV.second;
9475ffd83dbSDimitry Andric     // Parameter accessed by any or unknown offset, represented as FullSet by
9485ffd83dbSDimitry Andric     // StackSafety, is handled as the parameter for which we have no
9495ffd83dbSDimitry Andric     // StackSafety info at all. So drop it to reduce summary size.
9505ffd83dbSDimitry Andric     if (PS.Range.isFullSet())
9515ffd83dbSDimitry Andric       continue;
9525ffd83dbSDimitry Andric 
9535ffd83dbSDimitry Andric     ParamAccesses.emplace_back(KV.first, PS.Range);
9545ffd83dbSDimitry Andric     FunctionSummary::ParamAccess &Param = ParamAccesses.back();
9555ffd83dbSDimitry Andric 
9565ffd83dbSDimitry Andric     Param.Calls.reserve(PS.Calls.size());
957fcaf7f86SDimitry Andric     for (const auto &C : PS.Calls) {
9585ffd83dbSDimitry Andric       // Parameter forwarded into another function by any or unknown offset
9595ffd83dbSDimitry Andric       // will make ParamAccess::Range as FullSet anyway. So we can drop the
9605ffd83dbSDimitry Andric       // entire parameter like we did above.
9615ffd83dbSDimitry Andric       // TODO(vitalybuka): Return already filtered parameters from getInfo().
962e8d8bef9SDimitry Andric       if (C.second.isFullSet()) {
9635ffd83dbSDimitry Andric         ParamAccesses.pop_back();
9645ffd83dbSDimitry Andric         break;
9655ffd83dbSDimitry Andric       }
966e8d8bef9SDimitry Andric       Param.Calls.emplace_back(C.first.ParamNo,
967e8d8bef9SDimitry Andric                                Index.getOrInsertValueInfo(C.first.Callee),
968e8d8bef9SDimitry Andric                                C.second);
9695ffd83dbSDimitry Andric     }
9705ffd83dbSDimitry Andric   }
971e8d8bef9SDimitry Andric   for (FunctionSummary::ParamAccess &Param : ParamAccesses) {
972e8d8bef9SDimitry Andric     sort(Param.Calls, [](const FunctionSummary::ParamAccess::Call &L,
973e8d8bef9SDimitry Andric                          const FunctionSummary::ParamAccess::Call &R) {
974e8d8bef9SDimitry Andric       return std::tie(L.ParamNo, L.Callee) < std::tie(R.ParamNo, R.Callee);
975e8d8bef9SDimitry Andric     });
976e8d8bef9SDimitry Andric   }
9775ffd83dbSDimitry Andric   return ParamAccesses;
9785ffd83dbSDimitry Andric }
9795ffd83dbSDimitry Andric 
9805ffd83dbSDimitry Andric StackSafetyGlobalInfo::StackSafetyGlobalInfo() = default;
9815ffd83dbSDimitry Andric 
9825ffd83dbSDimitry Andric StackSafetyGlobalInfo::StackSafetyGlobalInfo(
9835ffd83dbSDimitry Andric     Module *M, std::function<const StackSafetyInfo &(Function &F)> GetSSI,
9845ffd83dbSDimitry Andric     const ModuleSummaryIndex *Index)
9855ffd83dbSDimitry Andric     : M(M), GetSSI(GetSSI), Index(Index) {
9865ffd83dbSDimitry Andric   if (StackSafetyRun)
9875ffd83dbSDimitry Andric     getInfo();
9885ffd83dbSDimitry Andric }
9895ffd83dbSDimitry Andric 
9905ffd83dbSDimitry Andric StackSafetyGlobalInfo::StackSafetyGlobalInfo(StackSafetyGlobalInfo &&) =
9915ffd83dbSDimitry Andric     default;
9925ffd83dbSDimitry Andric 
9935ffd83dbSDimitry Andric StackSafetyGlobalInfo &
9945ffd83dbSDimitry Andric StackSafetyGlobalInfo::operator=(StackSafetyGlobalInfo &&) = default;
9955ffd83dbSDimitry Andric 
9965ffd83dbSDimitry Andric StackSafetyGlobalInfo::~StackSafetyGlobalInfo() = default;
9975ffd83dbSDimitry Andric 
9985ffd83dbSDimitry Andric bool StackSafetyGlobalInfo::isSafe(const AllocaInst &AI) const {
9995ffd83dbSDimitry Andric   const auto &Info = getInfo();
10005ffd83dbSDimitry Andric   return Info.SafeAllocas.count(&AI);
10015ffd83dbSDimitry Andric }
10025ffd83dbSDimitry Andric 
1003349cc55cSDimitry Andric bool StackSafetyGlobalInfo::stackAccessIsSafe(const Instruction &I) const {
1004349cc55cSDimitry Andric   const auto &Info = getInfo();
10054824e7fdSDimitry Andric   return Info.UnsafeAccesses.find(&I) == Info.UnsafeAccesses.end();
1006349cc55cSDimitry Andric }
1007349cc55cSDimitry Andric 
10085ffd83dbSDimitry Andric void StackSafetyGlobalInfo::print(raw_ostream &O) const {
10095ffd83dbSDimitry Andric   auto &SSI = getInfo().Info;
10105ffd83dbSDimitry Andric   if (SSI.empty())
10115ffd83dbSDimitry Andric     return;
10125ffd83dbSDimitry Andric   const Module &M = *SSI.begin()->first->getParent();
1013fcaf7f86SDimitry Andric   for (const auto &F : M.functions()) {
10145ffd83dbSDimitry Andric     if (!F.isDeclaration()) {
10155ffd83dbSDimitry Andric       SSI.find(&F)->second.print(O, F.getName(), &F);
1016349cc55cSDimitry Andric       O << "    safe accesses:"
1017349cc55cSDimitry Andric         << "\n";
1018349cc55cSDimitry Andric       for (const auto &I : instructions(F)) {
1019349cc55cSDimitry Andric         const CallInst *Call = dyn_cast<CallInst>(&I);
1020349cc55cSDimitry Andric         if ((isa<StoreInst>(I) || isa<LoadInst>(I) || isa<MemIntrinsic>(I) ||
10215f757f3fSDimitry Andric              isa<AtomicCmpXchgInst>(I) || isa<AtomicRMWInst>(I) ||
1022349cc55cSDimitry Andric              (Call && Call->hasByValArgument())) &&
1023349cc55cSDimitry Andric             stackAccessIsSafe(I)) {
1024349cc55cSDimitry Andric           O << "     " << I << "\n";
1025349cc55cSDimitry Andric         }
1026349cc55cSDimitry Andric       }
10275ffd83dbSDimitry Andric       O << "\n";
10285ffd83dbSDimitry Andric     }
10295ffd83dbSDimitry Andric   }
10305ffd83dbSDimitry Andric }
10315ffd83dbSDimitry Andric 
10325ffd83dbSDimitry Andric LLVM_DUMP_METHOD void StackSafetyGlobalInfo::dump() const { print(dbgs()); }
10330b57cec5SDimitry Andric 
10340b57cec5SDimitry Andric AnalysisKey StackSafetyAnalysis::Key;
10350b57cec5SDimitry Andric 
10360b57cec5SDimitry Andric StackSafetyInfo StackSafetyAnalysis::run(Function &F,
10370b57cec5SDimitry Andric                                          FunctionAnalysisManager &AM) {
10385ffd83dbSDimitry Andric   return StackSafetyInfo(&F, [&AM, &F]() -> ScalarEvolution & {
10395ffd83dbSDimitry Andric     return AM.getResult<ScalarEvolutionAnalysis>(F);
10405ffd83dbSDimitry Andric   });
10410b57cec5SDimitry Andric }
10420b57cec5SDimitry Andric 
10430b57cec5SDimitry Andric PreservedAnalyses StackSafetyPrinterPass::run(Function &F,
10440b57cec5SDimitry Andric                                               FunctionAnalysisManager &AM) {
10450b57cec5SDimitry Andric   OS << "'Stack Safety Local Analysis' for function '" << F.getName() << "'\n";
10460b57cec5SDimitry Andric   AM.getResult<StackSafetyAnalysis>(F).print(OS);
10470b57cec5SDimitry Andric   return PreservedAnalyses::all();
10480b57cec5SDimitry Andric }
10490b57cec5SDimitry Andric 
10500b57cec5SDimitry Andric char StackSafetyInfoWrapperPass::ID = 0;
10510b57cec5SDimitry Andric 
10520b57cec5SDimitry Andric StackSafetyInfoWrapperPass::StackSafetyInfoWrapperPass() : FunctionPass(ID) {
10530b57cec5SDimitry Andric   initializeStackSafetyInfoWrapperPassPass(*PassRegistry::getPassRegistry());
10540b57cec5SDimitry Andric }
10550b57cec5SDimitry Andric 
10560b57cec5SDimitry Andric void StackSafetyInfoWrapperPass::getAnalysisUsage(AnalysisUsage &AU) const {
10575ffd83dbSDimitry Andric   AU.addRequiredTransitive<ScalarEvolutionWrapperPass>();
10580b57cec5SDimitry Andric   AU.setPreservesAll();
10590b57cec5SDimitry Andric }
10600b57cec5SDimitry Andric 
10610b57cec5SDimitry Andric void StackSafetyInfoWrapperPass::print(raw_ostream &O, const Module *M) const {
10620b57cec5SDimitry Andric   SSI.print(O);
10630b57cec5SDimitry Andric }
10640b57cec5SDimitry Andric 
10650b57cec5SDimitry Andric bool StackSafetyInfoWrapperPass::runOnFunction(Function &F) {
10665ffd83dbSDimitry Andric   auto *SE = &getAnalysis<ScalarEvolutionWrapperPass>().getSE();
10675ffd83dbSDimitry Andric   SSI = {&F, [SE]() -> ScalarEvolution & { return *SE; }};
10680b57cec5SDimitry Andric   return false;
10690b57cec5SDimitry Andric }
10700b57cec5SDimitry Andric 
10710b57cec5SDimitry Andric AnalysisKey StackSafetyGlobalAnalysis::Key;
10720b57cec5SDimitry Andric 
10730b57cec5SDimitry Andric StackSafetyGlobalInfo
10740b57cec5SDimitry Andric StackSafetyGlobalAnalysis::run(Module &M, ModuleAnalysisManager &AM) {
10755ffd83dbSDimitry Andric   // FIXME: Lookup Module Summary.
10760b57cec5SDimitry Andric   FunctionAnalysisManager &FAM =
10770b57cec5SDimitry Andric       AM.getResult<FunctionAnalysisManagerModuleProxy>(M).getManager();
10785ffd83dbSDimitry Andric   return {&M,
10795ffd83dbSDimitry Andric           [&FAM](Function &F) -> const StackSafetyInfo & {
10800b57cec5SDimitry Andric             return FAM.getResult<StackSafetyAnalysis>(F);
10815ffd83dbSDimitry Andric           },
10825ffd83dbSDimitry Andric           nullptr};
10830b57cec5SDimitry Andric }
10840b57cec5SDimitry Andric 
10850b57cec5SDimitry Andric PreservedAnalyses StackSafetyGlobalPrinterPass::run(Module &M,
10860b57cec5SDimitry Andric                                                     ModuleAnalysisManager &AM) {
10870b57cec5SDimitry Andric   OS << "'Stack Safety Analysis' for module '" << M.getName() << "'\n";
10885ffd83dbSDimitry Andric   AM.getResult<StackSafetyGlobalAnalysis>(M).print(OS);
10890b57cec5SDimitry Andric   return PreservedAnalyses::all();
10900b57cec5SDimitry Andric }
10910b57cec5SDimitry Andric 
10920b57cec5SDimitry Andric char StackSafetyGlobalInfoWrapperPass::ID = 0;
10930b57cec5SDimitry Andric 
10940b57cec5SDimitry Andric StackSafetyGlobalInfoWrapperPass::StackSafetyGlobalInfoWrapperPass()
10950b57cec5SDimitry Andric     : ModulePass(ID) {
10960b57cec5SDimitry Andric   initializeStackSafetyGlobalInfoWrapperPassPass(
10970b57cec5SDimitry Andric       *PassRegistry::getPassRegistry());
10980b57cec5SDimitry Andric }
10990b57cec5SDimitry Andric 
11005ffd83dbSDimitry Andric StackSafetyGlobalInfoWrapperPass::~StackSafetyGlobalInfoWrapperPass() = default;
11015ffd83dbSDimitry Andric 
11020b57cec5SDimitry Andric void StackSafetyGlobalInfoWrapperPass::print(raw_ostream &O,
11030b57cec5SDimitry Andric                                              const Module *M) const {
11045ffd83dbSDimitry Andric   SSGI.print(O);
11050b57cec5SDimitry Andric }
11060b57cec5SDimitry Andric 
11070b57cec5SDimitry Andric void StackSafetyGlobalInfoWrapperPass::getAnalysisUsage(
11080b57cec5SDimitry Andric     AnalysisUsage &AU) const {
11095ffd83dbSDimitry Andric   AU.setPreservesAll();
11100b57cec5SDimitry Andric   AU.addRequired<StackSafetyInfoWrapperPass>();
11110b57cec5SDimitry Andric }
11120b57cec5SDimitry Andric 
11130b57cec5SDimitry Andric bool StackSafetyGlobalInfoWrapperPass::runOnModule(Module &M) {
11145ffd83dbSDimitry Andric   const ModuleSummaryIndex *ImportSummary = nullptr;
11155ffd83dbSDimitry Andric   if (auto *IndexWrapperPass =
11165ffd83dbSDimitry Andric           getAnalysisIfAvailable<ImmutableModuleSummaryIndexWrapperPass>())
11175ffd83dbSDimitry Andric     ImportSummary = IndexWrapperPass->getIndex();
11185ffd83dbSDimitry Andric 
11195ffd83dbSDimitry Andric   SSGI = {&M,
11205ffd83dbSDimitry Andric           [this](Function &F) -> const StackSafetyInfo & {
11210b57cec5SDimitry Andric             return getAnalysis<StackSafetyInfoWrapperPass>(F).getResult();
11225ffd83dbSDimitry Andric           },
11235ffd83dbSDimitry Andric           ImportSummary};
11240b57cec5SDimitry Andric   return false;
11250b57cec5SDimitry Andric }
11260b57cec5SDimitry Andric 
11275ffd83dbSDimitry Andric bool llvm::needsParamAccessSummary(const Module &M) {
11285ffd83dbSDimitry Andric   if (StackSafetyRun)
11295ffd83dbSDimitry Andric     return true;
1130fcaf7f86SDimitry Andric   for (const auto &F : M.functions())
11315ffd83dbSDimitry Andric     if (F.hasFnAttribute(Attribute::SanitizeMemTag))
11325ffd83dbSDimitry Andric       return true;
11335ffd83dbSDimitry Andric   return false;
11345ffd83dbSDimitry Andric }
11355ffd83dbSDimitry Andric 
11365ffd83dbSDimitry Andric void llvm::generateParamAccessSummary(ModuleSummaryIndex &Index) {
1137e8d8bef9SDimitry Andric   if (!Index.hasParamAccess())
1138e8d8bef9SDimitry Andric     return;
11395ffd83dbSDimitry Andric   const ConstantRange FullSet(FunctionSummary::ParamAccess::RangeWidth, true);
1140e8d8bef9SDimitry Andric 
1141e8d8bef9SDimitry Andric   auto CountParamAccesses = [&](auto &Stat) {
1142e8d8bef9SDimitry Andric     if (!AreStatisticsEnabled())
1143e8d8bef9SDimitry Andric       return;
1144e8d8bef9SDimitry Andric     for (auto &GVS : Index)
1145e8d8bef9SDimitry Andric       for (auto &GV : GVS.second.SummaryList)
1146e8d8bef9SDimitry Andric         if (FunctionSummary *FS = dyn_cast<FunctionSummary>(GV.get()))
1147e8d8bef9SDimitry Andric           Stat += FS->paramAccesses().size();
1148e8d8bef9SDimitry Andric   };
1149e8d8bef9SDimitry Andric 
1150e8d8bef9SDimitry Andric   CountParamAccesses(NumCombinedParamAccessesBefore);
1151e8d8bef9SDimitry Andric 
11525ffd83dbSDimitry Andric   std::map<const FunctionSummary *, FunctionInfo<FunctionSummary>> Functions;
11535ffd83dbSDimitry Andric 
11545ffd83dbSDimitry Andric   // Convert the ModuleSummaryIndex to a FunctionMap
11555ffd83dbSDimitry Andric   for (auto &GVS : Index) {
11565ffd83dbSDimitry Andric     for (auto &GV : GVS.second.SummaryList) {
11575ffd83dbSDimitry Andric       FunctionSummary *FS = dyn_cast<FunctionSummary>(GV.get());
1158e8d8bef9SDimitry Andric       if (!FS || FS->paramAccesses().empty())
11595ffd83dbSDimitry Andric         continue;
11605ffd83dbSDimitry Andric       if (FS->isLive() && FS->isDSOLocal()) {
11615ffd83dbSDimitry Andric         FunctionInfo<FunctionSummary> FI;
1162fcaf7f86SDimitry Andric         for (const auto &PS : FS->paramAccesses()) {
11635ffd83dbSDimitry Andric           auto &US =
11645ffd83dbSDimitry Andric               FI.Params
11655ffd83dbSDimitry Andric                   .emplace(PS.ParamNo, FunctionSummary::ParamAccess::RangeWidth)
11665ffd83dbSDimitry Andric                   .first->second;
11675ffd83dbSDimitry Andric           US.Range = PS.Use;
1168fcaf7f86SDimitry Andric           for (const auto &Call : PS.Calls) {
11695ffd83dbSDimitry Andric             assert(!Call.Offsets.isFullSet());
1170e8d8bef9SDimitry Andric             FunctionSummary *S =
1171e8d8bef9SDimitry Andric                 findCalleeFunctionSummary(Call.Callee, FS->modulePath());
1172e8d8bef9SDimitry Andric             ++NumCombinedCalleeLookupTotal;
11735ffd83dbSDimitry Andric             if (!S) {
1174e8d8bef9SDimitry Andric               ++NumCombinedCalleeLookupFailed;
11755ffd83dbSDimitry Andric               US.Range = FullSet;
11765ffd83dbSDimitry Andric               US.Calls.clear();
11775ffd83dbSDimitry Andric               break;
11785ffd83dbSDimitry Andric             }
1179e8d8bef9SDimitry Andric             US.Calls.emplace(CallInfo<FunctionSummary>(S, Call.ParamNo),
1180e8d8bef9SDimitry Andric                              Call.Offsets);
11815ffd83dbSDimitry Andric           }
11825ffd83dbSDimitry Andric         }
11835ffd83dbSDimitry Andric         Functions.emplace(FS, std::move(FI));
11845ffd83dbSDimitry Andric       }
11855ffd83dbSDimitry Andric       // Reset data for all summaries. Alive and DSO local will be set back from
11865ffd83dbSDimitry Andric       // of data flow results below. Anything else will not be accessed
11875ffd83dbSDimitry Andric       // by ThinLTO backend, so we can save on bitcode size.
11885ffd83dbSDimitry Andric       FS->setParamAccesses({});
11895ffd83dbSDimitry Andric     }
11905ffd83dbSDimitry Andric   }
1191e8d8bef9SDimitry Andric   NumCombinedDataFlowNodes += Functions.size();
11925ffd83dbSDimitry Andric   StackSafetyDataFlowAnalysis<FunctionSummary> SSDFA(
11935ffd83dbSDimitry Andric       FunctionSummary::ParamAccess::RangeWidth, std::move(Functions));
1194fcaf7f86SDimitry Andric   for (const auto &KV : SSDFA.run()) {
11955ffd83dbSDimitry Andric     std::vector<FunctionSummary::ParamAccess> NewParams;
11965ffd83dbSDimitry Andric     NewParams.reserve(KV.second.Params.size());
1197fcaf7f86SDimitry Andric     for (const auto &Param : KV.second.Params) {
1198e8d8bef9SDimitry Andric       // It's not needed as FullSet is processed the same as a missing value.
1199e8d8bef9SDimitry Andric       if (Param.second.Range.isFullSet())
1200e8d8bef9SDimitry Andric         continue;
12015ffd83dbSDimitry Andric       NewParams.emplace_back();
12025ffd83dbSDimitry Andric       FunctionSummary::ParamAccess &New = NewParams.back();
12035ffd83dbSDimitry Andric       New.ParamNo = Param.first;
12045ffd83dbSDimitry Andric       New.Use = Param.second.Range; // Only range is needed.
12055ffd83dbSDimitry Andric     }
12065ffd83dbSDimitry Andric     const_cast<FunctionSummary *>(KV.first)->setParamAccesses(
12075ffd83dbSDimitry Andric         std::move(NewParams));
12085ffd83dbSDimitry Andric   }
1209e8d8bef9SDimitry Andric 
1210e8d8bef9SDimitry Andric   CountParamAccesses(NumCombinedParamAccessesAfter);
12115ffd83dbSDimitry Andric }
12125ffd83dbSDimitry Andric 
12130b57cec5SDimitry Andric static const char LocalPassArg[] = "stack-safety-local";
12140b57cec5SDimitry Andric static const char LocalPassName[] = "Stack Safety Local Analysis";
12150b57cec5SDimitry Andric INITIALIZE_PASS_BEGIN(StackSafetyInfoWrapperPass, LocalPassArg, LocalPassName,
12160b57cec5SDimitry Andric                       false, true)
12170b57cec5SDimitry Andric INITIALIZE_PASS_DEPENDENCY(ScalarEvolutionWrapperPass)
12180b57cec5SDimitry Andric INITIALIZE_PASS_END(StackSafetyInfoWrapperPass, LocalPassArg, LocalPassName,
12190b57cec5SDimitry Andric                     false, true)
12200b57cec5SDimitry Andric 
12210b57cec5SDimitry Andric static const char GlobalPassName[] = "Stack Safety Analysis";
12220b57cec5SDimitry Andric INITIALIZE_PASS_BEGIN(StackSafetyGlobalInfoWrapperPass, DEBUG_TYPE,
12235ffd83dbSDimitry Andric                       GlobalPassName, false, true)
12240b57cec5SDimitry Andric INITIALIZE_PASS_DEPENDENCY(StackSafetyInfoWrapperPass)
12255ffd83dbSDimitry Andric INITIALIZE_PASS_DEPENDENCY(ImmutableModuleSummaryIndexWrapperPass)
12260b57cec5SDimitry Andric INITIALIZE_PASS_END(StackSafetyGlobalInfoWrapperPass, DEBUG_TYPE,
12275ffd83dbSDimitry Andric                     GlobalPassName, false, true)
1228