From a70949807d0475443cc3d085137f4a0ba83ce854 Mon Sep 17 00:00:00 2001 From: Shaobo Date: Tue, 16 Nov 2021 15:05:59 -0800 Subject: [PATCH] Added a pass that warns about loops It also prints a loop bound if it can be determined via LLVM's ScalarEvolution class. Specifically, it's equal to the `ConstantMaxBackedgeTakenCount`. Partially addressed #760 --- CMakeLists.txt | 2 ++ include/smack/LoopInfo.h | 21 ++++++++++++++ include/smack/SmackWarnings.h | 2 ++ lib/smack/LoopInfo.cpp | 54 +++++++++++++++++++++++++++++++++++ lib/smack/SmackWarnings.cpp | 6 ++++ tools/llvm2bpl/llvm2bpl.cpp | 3 ++ 6 files changed, 88 insertions(+) create mode 100644 include/smack/LoopInfo.h create mode 100644 lib/smack/LoopInfo.cpp diff --git a/CMakeLists.txt b/CMakeLists.txt index 1a70d7692..3bfa8bbc8 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -124,6 +124,7 @@ add_library(smackTranslator STATIC include/smack/SplitAggregateValue.h include/smack/Prelude.h include/smack/SmackWarnings.h + include/smack/LoopInfo.h lib/smack/AddTiming.cpp lib/smack/BoogieAst.cpp lib/smack/BplFilePrinter.cpp @@ -151,6 +152,7 @@ add_library(smackTranslator STATIC lib/smack/SplitAggregateValue.cpp lib/smack/Prelude.cpp lib/smack/SmackWarnings.cpp + lib/smack/LoopInfo.cpp ) add_executable(llvm2bpl diff --git a/include/smack/LoopInfo.h b/include/smack/LoopInfo.h new file mode 100644 index 000000000..e9bd7a2d9 --- /dev/null +++ b/include/smack/LoopInfo.h @@ -0,0 +1,21 @@ +// +// This file is distributed under the MIT License. See LICENSE for details. +// +#ifndef LOOPINFO_H +#define LOOPINFO_H + +#include "llvm/Pass.h" + +namespace smack { + +class LoopInfo : public llvm::FunctionPass { +public: + static char ID; // Pass identification, replacement for typeid + LoopInfo() : llvm::FunctionPass(ID) {} + virtual llvm::StringRef getPassName() const override; + virtual bool runOnFunction(llvm::Function &F) override; + virtual void getAnalysisUsage(llvm::AnalysisUsage &) const override; +}; +} // namespace smack + +#endif // LOOPINFO_H diff --git a/include/smack/SmackWarnings.h b/include/smack/SmackWarnings.h index 5c3e5d940..11cb3b8bd 100644 --- a/include/smack/SmackWarnings.h +++ b/include/smack/SmackWarnings.h @@ -34,6 +34,8 @@ class SmackWarnings { static void warnApproximate(std::string name, Block *currBlock, const llvm::Instruction *i); + static void warnLoop(std::string description); + static void warnOverApproximate(std::string name, UnsetFlagsT unsetFlags, Block *currBlock, const llvm::Instruction *i, FlagRelation rel); diff --git a/lib/smack/LoopInfo.cpp b/lib/smack/LoopInfo.cpp new file mode 100644 index 000000000..ec9894c7e --- /dev/null +++ b/lib/smack/LoopInfo.cpp @@ -0,0 +1,54 @@ +// +// This file is distributed under the MIT License. See LICENSE for details. +// + +#define DEBUG_TYPE "smack-loop-info" +#include "smack/LoopInfo.h" +#include "smack/Debug.h" +#include "smack/SmackWarnings.h" +#include "llvm/Analysis/LoopInfo.h" +#include "llvm/Analysis/ScalarEvolution.h" +#include "llvm/Analysis/ScalarEvolutionExpressions.h" +#include "llvm/Support/raw_ostream.h" + +namespace smack { + +using namespace llvm; + +void LoopInfo::getAnalysisUsage(AnalysisUsage &AU) const { + AU.setPreservesAll(); + AU.addRequired(); + AU.addRequired(); +} + +bool LoopInfo::runOnFunction(Function &F) { + auto &loopInfo = getAnalysis().getLoopInfo(); + auto &SE = getAnalysis().getSE(); + for (auto LI = loopInfo.begin(), LIEnd = loopInfo.end(); LI != LIEnd; ++LI) { + auto L = *LI; + auto lr = L->getLocRange(); + auto sl = lr.getStart(); + auto el = lr.getEnd(); + std::string description; + raw_string_ostream o(description); + o << "found loop from line " << sl.getLine() << " to line " << el.getLine() + << " in function " << F.getName() << " with "; + // TODO: figure out why induction variable won't work + // auto bs = L->getInductionVariable(SE); + if (auto C = + dyn_cast(SE.getConstantMaxBackedgeTakenCount(L))) { + auto CI = C->getValue()->getValue().getZExtValue(); + o << "known loop bound " << CI; + } else + o << "unknown loop bound"; + SmackWarnings::warnLoop(o.str()); + } + return false; +} + +char LoopInfo::ID = 0; + +StringRef LoopInfo::getPassName() const { + return "Get Loop Information for the purpose of sound analysis"; +} +} // namespace smack diff --git a/lib/smack/SmackWarnings.cpp b/lib/smack/SmackWarnings.cpp index 835b30c7c..267b4d09c 100644 --- a/lib/smack/SmackWarnings.cpp +++ b/lib/smack/SmackWarnings.cpp @@ -54,6 +54,12 @@ std::string SmackWarnings::getFlagStr(UnsetFlagsT flags) { return ret + "}"; } +void SmackWarnings::warnLoop(std::string description) { + processApproximate(description + "; please check loop unrolling bound " + "otherwise there may be missed bugs", + {}, nullptr, nullptr, FlagRelation::And); +} + void SmackWarnings::warnApproximate(std::string name, Block *currBlock, const Instruction *i) { processApproximate( diff --git a/tools/llvm2bpl/llvm2bpl.cpp b/tools/llvm2bpl/llvm2bpl.cpp index 0502cfcde..5833b81af 100644 --- a/tools/llvm2bpl/llvm2bpl.cpp +++ b/tools/llvm2bpl/llvm2bpl.cpp @@ -34,6 +34,7 @@ #include "smack/ExtractContracts.h" #include "smack/InitializePasses.h" #include "smack/IntegerOverflowChecker.h" +#include "smack/LoopInfo.h" #include "smack/MemorySafetyChecker.h" #include "smack/Naming.h" #include "smack/NormalizeLoops.h" @@ -213,6 +214,8 @@ int main(int argc, char **argv) { // pass_manager.add(new smack::SimplifyLibCalls()); pass_manager.add(new llvm::Devirtualize()); pass_manager.add(new smack::SplitAggregateValue()); + pass_manager.add(llvm::createLoopSimplifyPass()); + pass_manager.add(new smack::LoopInfo()); if (smack::SmackOptions::MemorySafety) { pass_manager.add(new smack::MemorySafetyChecker());