In earlier patches regarding AnalyzerOptions, a lot of effort went into gathering all config options, and changing the interface so that potential misuse can be eliminited. Up until this point, AnalyzerOptions only evaluated an option when it was querried. For example, if we had a "-no-false-positives" flag, AnalyzerOptions would store an Optional field for it that would be None up until somewhere in the code until the flag's getter function is called. However, now that we're confident that we've gathered all configs, we can evaluate off of them before analysis, so we can emit a error on invalid input even if that prticular flag will not matter in that particular run of the analyzer. Another very big benefit of this is that debug.ConfigDumper will now show the value of all configs every single time. Also, almost all options related class have a similar interface, so uniformity is also a benefit. The implementation for errors on invalid input will be commited shorty. Differential Revision: https://reviews.llvm.org/D53692 llvm-svn: 348031
116 lines
3.7 KiB
C++
116 lines
3.7 KiB
C++
//===-- ModelInjector.cpp ---------------------------------------*- C++ -*-===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "ModelInjector.h"
|
|
#include "clang/AST/Decl.h"
|
|
#include "clang/Basic/IdentifierTable.h"
|
|
#include "clang/Basic/Stack.h"
|
|
#include "clang/Frontend/ASTUnit.h"
|
|
#include "clang/Frontend/CompilerInstance.h"
|
|
#include "clang/Frontend/FrontendAction.h"
|
|
#include "clang/Lex/Preprocessor.h"
|
|
#include "clang/Serialization/ASTReader.h"
|
|
#include "clang/StaticAnalyzer/Frontend/FrontendActions.h"
|
|
#include "llvm/ADT/STLExtras.h"
|
|
#include "llvm/Support/CrashRecoveryContext.h"
|
|
#include "llvm/Support/FileSystem.h"
|
|
#include <utility>
|
|
|
|
using namespace clang;
|
|
using namespace ento;
|
|
|
|
ModelInjector::ModelInjector(CompilerInstance &CI) : CI(CI) {}
|
|
|
|
Stmt *ModelInjector::getBody(const FunctionDecl *D) {
|
|
onBodySynthesis(D);
|
|
return Bodies[D->getName()];
|
|
}
|
|
|
|
Stmt *ModelInjector::getBody(const ObjCMethodDecl *D) {
|
|
onBodySynthesis(D);
|
|
return Bodies[D->getName()];
|
|
}
|
|
|
|
void ModelInjector::onBodySynthesis(const NamedDecl *D) {
|
|
|
|
// FIXME: what about overloads? Declarations can be used as keys but what
|
|
// about file name index? Mangled names may not be suitable for that either.
|
|
if (Bodies.count(D->getName()) != 0)
|
|
return;
|
|
|
|
SourceManager &SM = CI.getSourceManager();
|
|
FileID mainFileID = SM.getMainFileID();
|
|
|
|
AnalyzerOptionsRef analyzerOpts = CI.getAnalyzerOpts();
|
|
llvm::StringRef modelPath = analyzerOpts->ModelPath;
|
|
|
|
llvm::SmallString<128> fileName;
|
|
|
|
if (!modelPath.empty())
|
|
fileName =
|
|
llvm::StringRef(modelPath.str() + "/" + D->getName().str() + ".model");
|
|
else
|
|
fileName = llvm::StringRef(D->getName().str() + ".model");
|
|
|
|
if (!llvm::sys::fs::exists(fileName.str())) {
|
|
Bodies[D->getName()] = nullptr;
|
|
return;
|
|
}
|
|
|
|
auto Invocation = std::make_shared<CompilerInvocation>(CI.getInvocation());
|
|
|
|
FrontendOptions &FrontendOpts = Invocation->getFrontendOpts();
|
|
InputKind IK = InputKind::CXX; // FIXME
|
|
FrontendOpts.Inputs.clear();
|
|
FrontendOpts.Inputs.emplace_back(fileName, IK);
|
|
FrontendOpts.DisableFree = true;
|
|
|
|
Invocation->getDiagnosticOpts().VerifyDiagnostics = 0;
|
|
|
|
// Modules are parsed by a separate CompilerInstance, so this code mimics that
|
|
// behavior for models
|
|
CompilerInstance Instance(CI.getPCHContainerOperations());
|
|
Instance.setInvocation(std::move(Invocation));
|
|
Instance.createDiagnostics(
|
|
new ForwardingDiagnosticConsumer(CI.getDiagnosticClient()),
|
|
/*ShouldOwnClient=*/true);
|
|
|
|
Instance.getDiagnostics().setSourceManager(&SM);
|
|
|
|
Instance.setVirtualFileSystem(&CI.getVirtualFileSystem());
|
|
|
|
// The instance wants to take ownership, however DisableFree frontend option
|
|
// is set to true to avoid double free issues
|
|
Instance.setFileManager(&CI.getFileManager());
|
|
Instance.setSourceManager(&SM);
|
|
Instance.setPreprocessor(CI.getPreprocessorPtr());
|
|
Instance.setASTContext(&CI.getASTContext());
|
|
|
|
Instance.getPreprocessor().InitializeForModelFile();
|
|
|
|
ParseModelFileAction parseModelFile(Bodies);
|
|
|
|
llvm::CrashRecoveryContext CRC;
|
|
|
|
CRC.RunSafelyOnThread([&]() { Instance.ExecuteAction(parseModelFile); },
|
|
DesiredStackSize);
|
|
|
|
Instance.getPreprocessor().FinalizeForModelFile();
|
|
|
|
Instance.resetAndLeakSourceManager();
|
|
Instance.resetAndLeakFileManager();
|
|
Instance.resetAndLeakPreprocessor();
|
|
|
|
// The preprocessor enters to the main file id when parsing is started, so
|
|
// the main file id is changed to the model file during parsing and it needs
|
|
// to be reset to the former main file id after parsing of the model file
|
|
// is done.
|
|
SM.setMainFileID(mainFileID);
|
|
}
|