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