2009-11-13 11:51:44 +08:00
|
|
|
//===--- CompilerInstance.cpp ---------------------------------------------===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "clang/Frontend/CompilerInstance.h"
|
2009-11-14 10:47:17 +08:00
|
|
|
#include "clang/AST/ASTConsumer.h"
|
2009-11-13 16:20:47 +08:00
|
|
|
#include "clang/AST/ASTContext.h"
|
2011-12-03 07:42:12 +08:00
|
|
|
#include "clang/AST/Decl.h"
|
2009-11-13 11:51:44 +08:00
|
|
|
#include "clang/Basic/Diagnostic.h"
|
2009-11-13 12:12:06 +08:00
|
|
|
#include "clang/Basic/FileManager.h"
|
|
|
|
#include "clang/Basic/SourceManager.h"
|
2009-11-13 11:51:44 +08:00
|
|
|
#include "clang/Basic/TargetInfo.h"
|
2010-01-13 08:48:06 +08:00
|
|
|
#include "clang/Basic/Version.h"
|
2014-06-06 18:36:22 +08:00
|
|
|
#include "clang/Config/config.h"
|
2011-09-26 08:21:47 +08:00
|
|
|
#include "clang/Frontend/ChainedDiagnosticConsumer.h"
|
2010-01-13 08:48:06 +08:00
|
|
|
#include "clang/Frontend/FrontendAction.h"
|
2011-09-13 07:31:24 +08:00
|
|
|
#include "clang/Frontend/FrontendActions.h"
|
2009-11-13 17:36:05 +08:00
|
|
|
#include "clang/Frontend/FrontendDiagnostic.h"
|
2011-04-08 02:31:10 +08:00
|
|
|
#include "clang/Frontend/LogDiagnosticPrinter.h"
|
2011-10-29 08:12:39 +08:00
|
|
|
#include "clang/Frontend/SerializedDiagnosticPrinter.h"
|
2009-11-13 13:52:34 +08:00
|
|
|
#include "clang/Frontend/TextDiagnosticPrinter.h"
|
2009-11-13 13:52:11 +08:00
|
|
|
#include "clang/Frontend/Utils.h"
|
2012-12-04 17:13:33 +08:00
|
|
|
#include "clang/Frontend/VerifyDiagnosticConsumer.h"
|
|
|
|
#include "clang/Lex/HeaderSearch.h"
|
|
|
|
#include "clang/Lex/PTHManager.h"
|
|
|
|
#include "clang/Lex/Preprocessor.h"
|
2009-11-13 17:36:05 +08:00
|
|
|
#include "clang/Sema/CodeCompleteConsumer.h"
|
2012-12-04 17:13:33 +08:00
|
|
|
#include "clang/Sema/Sema.h"
|
|
|
|
#include "clang/Serialization/ASTReader.h"
|
2014-04-23 20:57:01 +08:00
|
|
|
#include "clang/Serialization/GlobalModuleIndex.h"
|
2010-03-31 01:33:59 +08:00
|
|
|
#include "llvm/ADT/Statistic.h"
|
2012-12-04 17:13:33 +08:00
|
|
|
#include "llvm/Support/CrashRecoveryContext.h"
|
2014-06-14 01:20:50 +08:00
|
|
|
#include "llvm/Support/Errc.h"
|
2012-12-04 17:13:33 +08:00
|
|
|
#include "llvm/Support/FileSystem.h"
|
2010-11-30 02:12:39 +08:00
|
|
|
#include "llvm/Support/Host.h"
|
2012-01-30 04:15:24 +08:00
|
|
|
#include "llvm/Support/LockFileManager.h"
|
2012-12-04 17:13:33 +08:00
|
|
|
#include "llvm/Support/MemoryBuffer.h"
|
2010-11-30 02:12:39 +08:00
|
|
|
#include "llvm/Support/Path.h"
|
|
|
|
#include "llvm/Support/Program.h"
|
|
|
|
#include "llvm/Support/Signals.h"
|
2012-12-04 17:13:33 +08:00
|
|
|
#include "llvm/Support/Timer.h"
|
|
|
|
#include "llvm/Support/raw_ostream.h"
|
2013-03-26 05:19:16 +08:00
|
|
|
#include <sys/stat.h>
|
2014-06-13 01:19:42 +08:00
|
|
|
#include <system_error>
|
2013-03-26 05:51:16 +08:00
|
|
|
#include <time.h>
|
2011-10-05 22:53:30 +08:00
|
|
|
|
2009-11-13 11:51:44 +08:00
|
|
|
using namespace clang;
|
|
|
|
|
2014-04-23 20:57:01 +08:00
|
|
|
CompilerInstance::CompilerInstance(bool BuildingModule)
|
|
|
|
: ModuleLoader(BuildingModule),
|
2014-05-22 12:46:25 +08:00
|
|
|
Invocation(new CompilerInvocation()), ModuleManager(nullptr),
|
2014-04-23 20:57:01 +08:00
|
|
|
BuildGlobalModuleIndex(false), HaveFullGlobalModuleIndex(false),
|
|
|
|
ModuleBuildFailed(false) {
|
2010-01-31 05:47:07 +08:00
|
|
|
}
|
2009-11-13 11:51:44 +08:00
|
|
|
|
|
|
|
CompilerInstance::~CompilerInstance() {
|
2012-10-15 03:21:21 +08:00
|
|
|
assert(OutputFiles.empty() && "Still output files in flight?");
|
2010-02-16 09:54:47 +08:00
|
|
|
}
|
|
|
|
|
2010-01-31 05:47:07 +08:00
|
|
|
void CompilerInstance::setInvocation(CompilerInvocation *Value) {
|
2011-03-22 02:40:17 +08:00
|
|
|
Invocation = Value;
|
2010-01-31 05:47:07 +08:00
|
|
|
}
|
|
|
|
|
2013-01-25 08:45:27 +08:00
|
|
|
bool CompilerInstance::shouldBuildGlobalModuleIndex() const {
|
2013-01-25 09:03:03 +08:00
|
|
|
return (BuildGlobalModuleIndex ||
|
2013-03-23 05:26:48 +08:00
|
|
|
(ModuleManager && ModuleManager->isGlobalIndexUnavailable() &&
|
|
|
|
getFrontendOpts().GenerateGlobalModuleIndex)) &&
|
2013-01-25 09:03:03 +08:00
|
|
|
!ModuleBuildFailed;
|
2013-01-25 08:45:27 +08:00
|
|
|
}
|
|
|
|
|
2011-09-26 07:23:43 +08:00
|
|
|
void CompilerInstance::setDiagnostics(DiagnosticsEngine *Value) {
|
2010-04-06 07:52:57 +08:00
|
|
|
Diagnostics = Value;
|
2009-11-14 09:20:40 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void CompilerInstance::setTarget(TargetInfo *Value) {
|
2011-03-22 02:40:17 +08:00
|
|
|
Target = Value;
|
2009-11-14 09:20:40 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void CompilerInstance::setFileManager(FileManager *Value) {
|
2011-03-22 02:40:17 +08:00
|
|
|
FileMgr = Value;
|
2014-02-21 05:59:23 +08:00
|
|
|
if (Value)
|
|
|
|
VirtualFileSystem = Value->getVirtualFileSystem();
|
|
|
|
else
|
|
|
|
VirtualFileSystem.reset();
|
2009-11-14 09:20:40 +08:00
|
|
|
}
|
|
|
|
|
2011-10-08 19:31:46 +08:00
|
|
|
void CompilerInstance::setSourceManager(SourceManager *Value) {
|
2011-03-22 02:40:17 +08:00
|
|
|
SourceMgr = Value;
|
2009-11-14 09:20:40 +08:00
|
|
|
}
|
|
|
|
|
2011-03-22 02:40:17 +08:00
|
|
|
void CompilerInstance::setPreprocessor(Preprocessor *Value) { PP = Value; }
|
2009-11-14 09:20:40 +08:00
|
|
|
|
2011-03-22 02:40:17 +08:00
|
|
|
void CompilerInstance::setASTContext(ASTContext *Value) { Context = Value; }
|
2009-11-14 10:47:17 +08:00
|
|
|
|
2010-08-13 07:31:19 +08:00
|
|
|
void CompilerInstance::setSema(Sema *S) {
|
|
|
|
TheSema.reset(S);
|
|
|
|
}
|
|
|
|
|
2014-08-11 03:56:51 +08:00
|
|
|
void CompilerInstance::setASTConsumer(std::unique_ptr<ASTConsumer> Value) {
|
|
|
|
Consumer = std::move(Value);
|
2009-11-14 09:20:40 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void CompilerInstance::setCodeCompletionConsumer(CodeCompleteConsumer *Value) {
|
|
|
|
CompletionConsumer.reset(Value);
|
|
|
|
}
|
2014-08-11 04:12:39 +08:00
|
|
|
|
|
|
|
std::unique_ptr<Sema> CompilerInstance::takeSema() {
|
|
|
|
return std::move(TheSema);
|
|
|
|
}
|
|
|
|
|
2014-02-27 12:11:59 +08:00
|
|
|
IntrusiveRefCntPtr<ASTReader> CompilerInstance::getModuleManager() const {
|
|
|
|
return ModuleManager;
|
|
|
|
}
|
|
|
|
void CompilerInstance::setModuleManager(IntrusiveRefCntPtr<ASTReader> Reader) {
|
|
|
|
ModuleManager = Reader;
|
|
|
|
}
|
2009-11-14 09:20:40 +08:00
|
|
|
|
2014-06-20 03:36:03 +08:00
|
|
|
std::shared_ptr<ModuleDependencyCollector>
|
|
|
|
CompilerInstance::getModuleDepCollector() const {
|
|
|
|
return ModuleDepCollector;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CompilerInstance::setModuleDepCollector(
|
|
|
|
std::shared_ptr<ModuleDependencyCollector> Collector) {
|
|
|
|
ModuleDepCollector = Collector;
|
|
|
|
}
|
|
|
|
|
2009-11-13 13:52:34 +08:00
|
|
|
// Diagnostics
|
2012-10-24 06:26:28 +08:00
|
|
|
static void SetUpDiagnosticLog(DiagnosticOptions *DiagOpts,
|
2011-04-08 02:59:02 +08:00
|
|
|
const CodeGenOptions *CodeGenOpts,
|
2011-09-26 07:23:43 +08:00
|
|
|
DiagnosticsEngine &Diags) {
|
2014-08-26 02:17:04 +08:00
|
|
|
std::error_code EC;
|
2014-09-16 01:30:56 +08:00
|
|
|
std::unique_ptr<raw_ostream> StreamOwner;
|
2011-07-23 18:55:15 +08:00
|
|
|
raw_ostream *OS = &llvm::errs();
|
2012-10-24 06:26:28 +08:00
|
|
|
if (DiagOpts->DiagnosticLogFile != "-") {
|
2011-04-08 02:31:10 +08:00
|
|
|
// Create the output stream.
|
2014-09-16 01:30:56 +08:00
|
|
|
auto FileOS = llvm::make_unique<llvm::raw_fd_ostream>(
|
2014-08-26 02:17:04 +08:00
|
|
|
DiagOpts->DiagnosticLogFile, EC,
|
2014-09-16 01:30:56 +08:00
|
|
|
llvm::sys::fs::F_Append | llvm::sys::fs::F_Text);
|
2014-08-26 02:17:04 +08:00
|
|
|
if (EC) {
|
2011-04-08 02:31:10 +08:00
|
|
|
Diags.Report(diag::warn_fe_cc_log_diagnostics_failure)
|
2014-08-26 02:17:04 +08:00
|
|
|
<< DiagOpts->DiagnosticLogFile << EC.message();
|
2011-04-08 02:31:10 +08:00
|
|
|
} else {
|
|
|
|
FileOS->SetUnbuffered();
|
|
|
|
FileOS->SetUseAtomicWrites(true);
|
2014-09-16 01:30:56 +08:00
|
|
|
OS = FileOS.get();
|
|
|
|
StreamOwner = std::move(FileOS);
|
2011-04-08 02:31:10 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Chain in the diagnostic client which will log the diagnostics.
|
2014-09-16 01:50:10 +08:00
|
|
|
auto Logger = llvm::make_unique<LogDiagnosticPrinter>(*OS, DiagOpts,
|
|
|
|
std::move(StreamOwner));
|
2011-04-08 02:59:02 +08:00
|
|
|
if (CodeGenOpts)
|
|
|
|
Logger->setDwarfDebugFlags(CodeGenOpts->DwarfDebugFlags);
|
2014-09-16 01:50:10 +08:00
|
|
|
assert(Diags.ownsClient());
|
|
|
|
Diags.setClient(new ChainedDiagnosticConsumer(
|
|
|
|
std::unique_ptr<DiagnosticConsumer>(Diags.takeClient()),
|
|
|
|
std::move(Logger)));
|
2011-04-08 02:31:10 +08:00
|
|
|
}
|
|
|
|
|
2012-10-24 06:26:28 +08:00
|
|
|
static void SetupSerializedDiagnostics(DiagnosticOptions *DiagOpts,
|
2011-10-29 08:12:39 +08:00
|
|
|
DiagnosticsEngine &Diags,
|
|
|
|
StringRef OutputFile) {
|
2014-09-16 01:50:10 +08:00
|
|
|
auto SerializedConsumer =
|
2014-10-24 06:20:11 +08:00
|
|
|
clang::serialized_diags::create(OutputFile, DiagOpts);
|
2011-10-29 08:12:39 +08:00
|
|
|
|
2014-11-13 21:08:27 +08:00
|
|
|
if (Diags.ownsClient()) {
|
|
|
|
Diags.setClient(new ChainedDiagnosticConsumer(
|
|
|
|
std::unique_ptr<DiagnosticConsumer>(Diags.takeClient()),
|
|
|
|
std::move(SerializedConsumer)));
|
|
|
|
} else {
|
|
|
|
Diags.setClient(new ChainedDiagnosticConsumer(
|
|
|
|
Diags.takeClient(), std::move(SerializedConsumer)));
|
|
|
|
}
|
2011-10-29 08:12:39 +08:00
|
|
|
}
|
|
|
|
|
2013-01-20 09:58:28 +08:00
|
|
|
void CompilerInstance::createDiagnostics(DiagnosticConsumer *Client,
|
2013-05-04 07:07:45 +08:00
|
|
|
bool ShouldOwnClient) {
|
2013-01-20 09:58:28 +08:00
|
|
|
Diagnostics = createDiagnostics(&getDiagnosticOpts(), Client,
|
2013-05-04 07:07:45 +08:00
|
|
|
ShouldOwnClient, &getCodeGenOpts());
|
2009-11-13 13:52:34 +08:00
|
|
|
}
|
|
|
|
|
2012-02-20 22:00:23 +08:00
|
|
|
IntrusiveRefCntPtr<DiagnosticsEngine>
|
2012-10-24 06:26:28 +08:00
|
|
|
CompilerInstance::createDiagnostics(DiagnosticOptions *Opts,
|
2011-09-26 07:39:51 +08:00
|
|
|
DiagnosticConsumer *Client,
|
2011-09-13 09:26:44 +08:00
|
|
|
bool ShouldOwnClient,
|
2011-04-08 02:59:02 +08:00
|
|
|
const CodeGenOptions *CodeGenOpts) {
|
2012-02-20 22:00:23 +08:00
|
|
|
IntrusiveRefCntPtr<DiagnosticIDs> DiagID(new DiagnosticIDs());
|
|
|
|
IntrusiveRefCntPtr<DiagnosticsEngine>
|
2012-10-24 06:26:28 +08:00
|
|
|
Diags(new DiagnosticsEngine(DiagID, Opts));
|
2009-11-14 15:53:24 +08:00
|
|
|
|
2009-11-13 13:52:34 +08:00
|
|
|
// Create the diagnostic client for reporting errors or for
|
|
|
|
// implementing -verify.
|
2011-09-29 08:38:00 +08:00
|
|
|
if (Client) {
|
2013-05-04 07:07:45 +08:00
|
|
|
Diags->setClient(Client, ShouldOwnClient);
|
2011-09-29 08:38:00 +08:00
|
|
|
} else
|
2010-11-11 08:39:14 +08:00
|
|
|
Diags->setClient(new TextDiagnosticPrinter(llvm::errs(), Opts));
|
2009-11-14 11:24:39 +08:00
|
|
|
|
|
|
|
// Chain in -verify checker, if requested.
|
2012-10-24 06:26:28 +08:00
|
|
|
if (Opts->VerifyDiagnostics)
|
2011-09-26 08:38:03 +08:00
|
|
|
Diags->setClient(new VerifyDiagnosticConsumer(*Diags));
|
2009-11-13 13:52:34 +08:00
|
|
|
|
2011-04-08 02:31:10 +08:00
|
|
|
// Chain in -diagnostic-log-file dumper, if requested.
|
2012-10-24 06:26:28 +08:00
|
|
|
if (!Opts->DiagnosticLogFile.empty())
|
2011-04-08 02:59:02 +08:00
|
|
|
SetUpDiagnosticLog(Opts, CodeGenOpts, *Diags);
|
2011-10-08 19:31:46 +08:00
|
|
|
|
2012-10-24 06:26:28 +08:00
|
|
|
if (!Opts->DiagnosticSerializationFile.empty())
|
2011-10-29 08:12:39 +08:00
|
|
|
SetupSerializedDiagnostics(Opts, *Diags,
|
2012-10-24 06:26:28 +08:00
|
|
|
Opts->DiagnosticSerializationFile);
|
2011-10-29 08:12:39 +08:00
|
|
|
|
2009-11-13 13:52:34 +08:00
|
|
|
// Configure our handling of diagnostics.
|
2012-10-24 06:26:28 +08:00
|
|
|
ProcessWarningOptions(*Diags, *Opts);
|
2009-11-13 13:52:34 +08:00
|
|
|
|
2010-04-06 07:52:57 +08:00
|
|
|
return Diags;
|
2009-11-13 13:52:34 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// File Manager
|
|
|
|
|
2009-11-13 12:12:06 +08:00
|
|
|
void CompilerInstance::createFileManager() {
|
2014-02-21 05:59:23 +08:00
|
|
|
if (!hasVirtualFileSystem()) {
|
|
|
|
// TODO: choose the virtual file system based on the CompilerInvocation.
|
|
|
|
setVirtualFileSystem(vfs::getRealFileSystem());
|
|
|
|
}
|
|
|
|
FileMgr = new FileManager(getFileSystemOpts(), VirtualFileSystem);
|
2009-11-13 12:12:06 +08:00
|
|
|
}
|
|
|
|
|
2009-11-13 13:52:34 +08:00
|
|
|
// Source Manager
|
|
|
|
|
2010-11-23 16:35:12 +08:00
|
|
|
void CompilerInstance::createSourceManager(FileManager &FileMgr) {
|
2011-03-22 02:40:17 +08:00
|
|
|
SourceMgr = new SourceManager(getDiagnostics(), FileMgr);
|
2009-11-13 12:12:06 +08:00
|
|
|
}
|
2009-11-13 13:52:11 +08:00
|
|
|
|
2014-07-07 14:05:00 +08:00
|
|
|
// Initialize the remapping of files to alternative contents, e.g.,
|
|
|
|
// those specified through other files.
|
|
|
|
static void InitializeFileRemapping(DiagnosticsEngine &Diags,
|
|
|
|
SourceManager &SourceMgr,
|
|
|
|
FileManager &FileMgr,
|
|
|
|
const PreprocessorOptions &InitOpts) {
|
|
|
|
// Remap files in the source manager (with buffers).
|
2014-07-07 15:47:20 +08:00
|
|
|
for (const auto &RB : InitOpts.RemappedFileBuffers) {
|
2014-07-07 14:05:00 +08:00
|
|
|
// Create the file entry for the file that we're mapping from.
|
|
|
|
const FileEntry *FromFile =
|
2014-07-07 15:47:20 +08:00
|
|
|
FileMgr.getVirtualFile(RB.first, RB.second->getBufferSize(), 0);
|
2014-07-07 14:05:00 +08:00
|
|
|
if (!FromFile) {
|
2014-07-07 15:47:20 +08:00
|
|
|
Diags.Report(diag::err_fe_remap_missing_from_file) << RB.first;
|
2014-07-07 14:05:00 +08:00
|
|
|
if (!InitOpts.RetainRemappedFileBuffers)
|
2014-07-07 15:47:20 +08:00
|
|
|
delete RB.second;
|
2014-07-07 14:05:00 +08:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Override the contents of the "from" file with the contents of
|
|
|
|
// the "to" file.
|
2014-07-07 15:47:20 +08:00
|
|
|
SourceMgr.overrideFileContents(FromFile, RB.second,
|
2014-07-07 14:05:00 +08:00
|
|
|
InitOpts.RetainRemappedFileBuffers);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remap files in the source manager (with other files).
|
2014-07-07 15:47:20 +08:00
|
|
|
for (const auto &RF : InitOpts.RemappedFiles) {
|
2014-07-07 14:05:00 +08:00
|
|
|
// Find the file that we're mapping to.
|
2014-07-07 15:47:20 +08:00
|
|
|
const FileEntry *ToFile = FileMgr.getFile(RF.second);
|
2014-07-07 14:05:00 +08:00
|
|
|
if (!ToFile) {
|
2014-07-07 15:47:20 +08:00
|
|
|
Diags.Report(diag::err_fe_remap_missing_to_file) << RF.first << RF.second;
|
2014-07-07 14:05:00 +08:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create the file entry for the file that we're mapping from.
|
|
|
|
const FileEntry *FromFile =
|
2014-07-07 15:47:20 +08:00
|
|
|
FileMgr.getVirtualFile(RF.first, ToFile->getSize(), 0);
|
2014-07-07 14:05:00 +08:00
|
|
|
if (!FromFile) {
|
2014-07-07 15:47:20 +08:00
|
|
|
Diags.Report(diag::err_fe_remap_missing_from_file) << RF.first;
|
2014-07-07 14:05:00 +08:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Override the contents of the "from" file with the contents of
|
|
|
|
// the "to" file.
|
|
|
|
SourceMgr.overrideFileContents(FromFile, ToFile);
|
|
|
|
}
|
|
|
|
|
|
|
|
SourceMgr.setOverridenFilesKeepOriginalName(
|
|
|
|
InitOpts.RemappedFilesKeepOriginalName);
|
|
|
|
}
|
|
|
|
|
2009-11-13 13:52:34 +08:00
|
|
|
// Preprocessor
|
|
|
|
|
2014-03-07 15:47:58 +08:00
|
|
|
void CompilerInstance::createPreprocessor(TranslationUnitKind TUKind) {
|
2011-08-27 07:56:07 +08:00
|
|
|
const PreprocessorOptions &PPOpts = getPreprocessorOpts();
|
2011-10-08 19:31:46 +08:00
|
|
|
|
2009-11-13 13:52:11 +08:00
|
|
|
// Create a PTH manager if we are using some form of a token cache.
|
2014-05-22 12:46:25 +08:00
|
|
|
PTHManager *PTHMgr = nullptr;
|
2009-11-17 13:52:41 +08:00
|
|
|
if (!PPOpts.TokenCache.empty())
|
2011-08-27 07:56:07 +08:00
|
|
|
PTHMgr = PTHManager::Create(PPOpts.TokenCache, getDiagnostics());
|
2011-10-08 19:31:46 +08:00
|
|
|
|
2009-11-13 13:52:11 +08:00
|
|
|
// Create the Preprocessor.
|
2012-10-25 00:19:39 +08:00
|
|
|
HeaderSearch *HeaderInfo = new HeaderSearch(&getHeaderSearchOpts(),
|
Use the same SourceManager for ModuleMaps and compilations.
This allows using virtual file mappings on the original SourceManager to
map in virtual module.map files. Without this patch, the ModuleMap
search will find a module.map file (as the FileEntry exists in the
FileManager), but will be unable to get the content from the
SourceManager (as ModuleMap previously created its own SourceManager).
Two problems needed to be fixed which this patch exposed:
1. Storing the inferred module map
When writing out a module, the ASTWriter stores the names of the files
in the main source manager; when loading the AST again, the ASTReader
errs out if such a file is found missing, unless it is overridden.
Previously CompilerInstance's compileModule method would store the
inferred module map to a temporary file; the problem with this approach
is that now that the module map is handled by the main source manager,
the ASTWriter stores the name of the temporary module map as source to
the compilation; later, when the module is loaded, the temporary file
has already been deleted, which leads to a compilation error. This patch
changes the inferred module map to instead inject a virtual file into
the source manager. This both saves some disk IO, and works with how the
ASTWriter/ASTReader handle overridden source files.
2. Changing test input in test/Modules/Inputs/*
Now that the module map file is handled by the main source manager, the
VerifyDiagnosticConsumer will not ignore diagnostics created while
parsing the module map file. The module test test/Modules/renamed.m uses
-I test/Modules/Inputs and triggers recursive loading of all module maps
in test/Modules/Inputs, some of which had conflicting names, thus
leading errors while parsing the module maps. Those diagnostics already
occur on trunk, but before this patch they would not break the test, as
they were ignored by the VerifyDiagnosticConsumer. This patch thus
changes the module maps that have been recently introduced which broke
the invariant of compatible modules maps in test/Modules/Inputs.
llvm-svn: 193314
2013-10-24 15:51:24 +08:00
|
|
|
getSourceManager(),
|
2011-12-31 12:05:44 +08:00
|
|
|
getDiagnostics(),
|
2012-01-30 14:01:29 +08:00
|
|
|
getLangOpts(),
|
|
|
|
&getTarget());
|
2014-05-02 11:43:38 +08:00
|
|
|
PP = new Preprocessor(&getPreprocessorOpts(), getDiagnostics(), getLangOpts(),
|
2011-08-27 07:56:07 +08:00
|
|
|
getSourceManager(), *HeaderInfo, *this, PTHMgr,
|
2014-05-02 11:43:38 +08:00
|
|
|
/*OwnsHeaderSearch=*/true, TUKind);
|
2014-05-02 11:43:30 +08:00
|
|
|
PP->Initialize(getTarget());
|
2011-10-08 19:31:46 +08:00
|
|
|
|
2009-11-13 13:52:11 +08:00
|
|
|
// Note that this is different then passing PTHMgr to Preprocessor's ctor.
|
|
|
|
// That argument is used as the IdentifierInfoLookup argument to
|
|
|
|
// IdentifierTable's ctor.
|
|
|
|
if (PTHMgr) {
|
2011-08-27 07:56:07 +08:00
|
|
|
PTHMgr->setPreprocessor(&*PP);
|
2009-11-13 13:52:11 +08:00
|
|
|
PP->setPTHManager(PTHMgr);
|
|
|
|
}
|
2011-10-08 19:31:46 +08:00
|
|
|
|
2010-03-20 00:15:56 +08:00
|
|
|
if (PPOpts.DetailedRecord)
|
2012-12-04 15:27:05 +08:00
|
|
|
PP->createPreprocessingRecord();
|
2011-10-08 19:31:46 +08:00
|
|
|
|
2014-07-07 14:05:00 +08:00
|
|
|
// Apply remappings to the source manager.
|
|
|
|
InitializeFileRemapping(PP->getDiagnostics(), PP->getSourceManager(),
|
|
|
|
PP->getFileManager(), PPOpts);
|
|
|
|
|
|
|
|
// Predefine macros and configure the preprocessor.
|
|
|
|
InitializePreprocessor(*PP, PPOpts, getFrontendOpts());
|
|
|
|
|
|
|
|
// Initialize the header search object.
|
|
|
|
ApplyHeaderSearchOptions(PP->getHeaderSearchInfo(), getHeaderSearchOpts(),
|
|
|
|
PP->getLangOpts(), PP->getTargetInfo().getTriple());
|
2011-10-08 19:31:46 +08:00
|
|
|
|
2013-01-30 09:52:57 +08:00
|
|
|
PP->setPreprocessedOutput(getPreprocessorOutputOpts().ShowCPP);
|
|
|
|
|
2011-09-14 07:15:45 +08:00
|
|
|
// Set up the module path, including the hash for the
|
|
|
|
// module-creation options.
|
2012-02-05 10:13:05 +08:00
|
|
|
SmallString<256> SpecificModuleCache(
|
2011-09-14 07:15:45 +08:00
|
|
|
getHeaderSearchOpts().ModuleCachePath);
|
|
|
|
if (!getHeaderSearchOpts().DisableModuleHash)
|
2011-10-08 19:31:46 +08:00
|
|
|
llvm::sys::path::append(SpecificModuleCache,
|
2011-09-14 07:15:45 +08:00
|
|
|
getInvocation().getModuleHash());
|
2011-12-09 01:01:29 +08:00
|
|
|
PP->getHeaderSearchInfo().setModuleCachePath(SpecificModuleCache);
|
2011-10-08 19:31:46 +08:00
|
|
|
|
2009-11-13 13:52:11 +08:00
|
|
|
// Handle generating dependencies, if requested.
|
2011-08-27 07:56:07 +08:00
|
|
|
const DependencyOutputOptions &DepOpts = getDependencyOutputOpts();
|
2009-11-13 13:52:11 +08:00
|
|
|
if (!DepOpts.OutputFile.empty())
|
2014-03-07 14:40:32 +08:00
|
|
|
TheDependencyFileGenerator.reset(
|
|
|
|
DependencyFileGenerator::CreateAndAttachToPreprocessor(*PP, DepOpts));
|
2012-02-03 07:45:13 +08:00
|
|
|
if (!DepOpts.DOTOutputFile.empty())
|
|
|
|
AttachDependencyGraphGen(*PP, DepOpts.DOTOutputFile,
|
2012-02-02 08:54:52 +08:00
|
|
|
getHeaderSearchOpts().Sysroot);
|
2011-10-08 19:31:46 +08:00
|
|
|
|
2014-07-01 04:04:14 +08:00
|
|
|
for (auto &Listener : DependencyCollectors)
|
|
|
|
Listener->attachToPreprocessor(*PP);
|
|
|
|
|
2014-06-20 03:36:03 +08:00
|
|
|
// If we don't have a collector, but we are collecting module dependencies,
|
|
|
|
// then we're the top level compiler instance and need to create one.
|
|
|
|
if (!ModuleDepCollector && !DepOpts.ModuleDependencyOutputDir.empty())
|
|
|
|
ModuleDepCollector = std::make_shared<ModuleDependencyCollector>(
|
|
|
|
DepOpts.ModuleDependencyOutputDir);
|
2013-08-09 08:32:23 +08:00
|
|
|
|
2011-02-02 23:41:17 +08:00
|
|
|
// Handle generating header include information, if requested.
|
|
|
|
if (DepOpts.ShowHeaderIncludes)
|
|
|
|
AttachHeaderIncludeGen(*PP);
|
2011-02-03 05:11:31 +08:00
|
|
|
if (!DepOpts.HeaderIncludeOutputFile.empty()) {
|
2011-07-23 18:55:15 +08:00
|
|
|
StringRef OutputPath = DepOpts.HeaderIncludeOutputFile;
|
2011-02-03 05:11:31 +08:00
|
|
|
if (OutputPath == "-")
|
|
|
|
OutputPath = "";
|
2011-03-22 03:37:38 +08:00
|
|
|
AttachHeaderIncludeGen(*PP, /*ShowAllHeaders=*/true, OutputPath,
|
|
|
|
/*ShowDepth=*/false);
|
2011-02-03 05:11:31 +08:00
|
|
|
}
|
2013-08-09 08:32:23 +08:00
|
|
|
|
|
|
|
if (DepOpts.PrintShowIncludes) {
|
|
|
|
AttachHeaderIncludeGen(*PP, /*ShowAllHeaders=*/false, /*OutputPath=*/"",
|
|
|
|
/*ShowDepth=*/true, /*MSStyle=*/true);
|
|
|
|
}
|
2009-11-13 13:52:11 +08:00
|
|
|
}
|
2009-11-13 16:20:47 +08:00
|
|
|
|
|
|
|
// ASTContext
|
|
|
|
|
|
|
|
void CompilerInstance::createASTContext() {
|
|
|
|
Preprocessor &PP = getPreprocessor();
|
2011-03-22 02:40:17 +08:00
|
|
|
Context = new ASTContext(getLangOpts(), PP.getSourceManager(),
|
2014-05-03 11:46:04 +08:00
|
|
|
PP.getIdentifierTable(), PP.getSelectorTable(),
|
|
|
|
PP.getBuiltinInfo());
|
|
|
|
Context->InitBuiltinTypes(getTarget());
|
2009-11-13 16:20:47 +08:00
|
|
|
}
|
2009-11-13 16:21:10 +08:00
|
|
|
|
|
|
|
// ExternalASTSource
|
|
|
|
|
2014-05-08 12:26:47 +08:00
|
|
|
void CompilerInstance::createPCHExternalASTSource(
|
|
|
|
StringRef Path, bool DisablePCHValidation, bool AllowPCHWithCompilerErrors,
|
|
|
|
void *DeserializationListener, bool OwnDeserializationListener) {
|
2014-02-27 12:11:59 +08:00
|
|
|
IntrusiveRefCntPtr<ExternalASTSource> Source;
|
2010-10-06 00:15:19 +08:00
|
|
|
bool Preamble = getPreprocessorOpts().PrecompiledPreambleBytes.first != 0;
|
2014-05-08 12:26:47 +08:00
|
|
|
Source = createPCHExternalASTSource(
|
|
|
|
Path, getHeaderSearchOpts().Sysroot, DisablePCHValidation,
|
|
|
|
AllowPCHWithCompilerErrors, getPreprocessor(), getASTContext(),
|
|
|
|
DeserializationListener, OwnDeserializationListener, Preamble,
|
|
|
|
getFrontendOpts().UseGlobalModuleIndex);
|
2014-07-05 11:08:06 +08:00
|
|
|
ModuleManager = static_cast<ASTReader*>(Source.get());
|
2009-11-13 16:21:10 +08:00
|
|
|
getASTContext().setExternalSource(Source);
|
|
|
|
}
|
|
|
|
|
2014-05-08 12:26:47 +08:00
|
|
|
ExternalASTSource *CompilerInstance::createPCHExternalASTSource(
|
|
|
|
StringRef Path, const std::string &Sysroot, bool DisablePCHValidation,
|
|
|
|
bool AllowPCHWithCompilerErrors, Preprocessor &PP, ASTContext &Context,
|
|
|
|
void *DeserializationListener, bool OwnDeserializationListener,
|
|
|
|
bool Preamble, bool UseGlobalModuleIndex) {
|
2014-03-12 08:06:17 +08:00
|
|
|
HeaderSearchOptions &HSOpts = PP.getHeaderSearchInfo().getHeaderSearchOpts();
|
|
|
|
|
2014-03-08 04:03:18 +08:00
|
|
|
std::unique_ptr<ASTReader> Reader;
|
2011-09-02 08:26:20 +08:00
|
|
|
Reader.reset(new ASTReader(PP, Context,
|
2011-07-23 00:35:34 +08:00
|
|
|
Sysroot.empty() ? "" : Sysroot.c_str(),
|
2012-11-01 04:59:50 +08:00
|
|
|
DisablePCHValidation,
|
2013-01-25 08:45:27 +08:00
|
|
|
AllowPCHWithCompilerErrors,
|
2014-02-08 01:31:11 +08:00
|
|
|
/*AllowConfigurationMismatch*/false,
|
2014-03-12 08:06:17 +08:00
|
|
|
HSOpts.ModulesValidateSystemHeaders,
|
2013-01-25 08:45:27 +08:00
|
|
|
UseGlobalModuleIndex));
|
2009-11-13 16:21:10 +08:00
|
|
|
|
2010-07-30 08:29:29 +08:00
|
|
|
Reader->setDeserializationListener(
|
2014-05-08 12:26:47 +08:00
|
|
|
static_cast<ASTDeserializationListener *>(DeserializationListener),
|
|
|
|
/*TakeOwnership=*/OwnDeserializationListener);
|
2011-10-08 19:31:46 +08:00
|
|
|
switch (Reader->ReadAST(Path,
|
|
|
|
Preamble ? serialization::MK_Preamble
|
2012-10-23 07:51:00 +08:00
|
|
|
: serialization::MK_PCH,
|
2012-11-16 02:57:22 +08:00
|
|
|
SourceLocation(),
|
2014-02-08 01:31:11 +08:00
|
|
|
ASTReader::ARR_None)) {
|
2010-08-19 07:56:43 +08:00
|
|
|
case ASTReader::Success:
|
2009-11-13 16:21:10 +08:00
|
|
|
// Set the predefines buffer as suggested by the PCH reader. Typically, the
|
|
|
|
// predefines buffer will be empty.
|
|
|
|
PP.setPredefines(Reader->getSuggestedPredefines());
|
2014-03-08 03:33:25 +08:00
|
|
|
return Reader.release();
|
2009-11-13 16:21:10 +08:00
|
|
|
|
2010-08-19 07:56:43 +08:00
|
|
|
case ASTReader::Failure:
|
2009-11-13 16:21:10 +08:00
|
|
|
// Unrecoverable failure: don't even try to process the input file.
|
|
|
|
break;
|
|
|
|
|
2013-03-19 08:28:20 +08:00
|
|
|
case ASTReader::Missing:
|
2012-10-23 06:50:17 +08:00
|
|
|
case ASTReader::OutOfDate:
|
|
|
|
case ASTReader::VersionMismatch:
|
|
|
|
case ASTReader::ConfigurationMismatch:
|
|
|
|
case ASTReader::HadErrors:
|
2009-11-13 16:21:10 +08:00
|
|
|
// No suitable PCH file could be found. Return an error.
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2014-05-22 12:46:25 +08:00
|
|
|
return nullptr;
|
2009-11-13 16:21:10 +08:00
|
|
|
}
|
2009-11-13 17:36:05 +08:00
|
|
|
|
|
|
|
// Code Completion
|
|
|
|
|
2011-10-08 19:31:46 +08:00
|
|
|
static bool EnableCodeCompletion(Preprocessor &PP,
|
2010-08-05 00:47:14 +08:00
|
|
|
const std::string &Filename,
|
|
|
|
unsigned Line,
|
|
|
|
unsigned Column) {
|
|
|
|
// Tell the source manager to chop off the given file at a specific
|
|
|
|
// line and column.
|
2010-11-23 16:35:12 +08:00
|
|
|
const FileEntry *Entry = PP.getFileManager().getFile(Filename);
|
2010-08-05 00:47:14 +08:00
|
|
|
if (!Entry) {
|
|
|
|
PP.getDiagnostics().Report(diag::err_fe_invalid_code_complete_file)
|
|
|
|
<< Filename;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Truncate the named file at the given line/column.
|
|
|
|
PP.SetCodeCompletionPoint(Entry, Line, Column);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2009-11-13 17:36:05 +08:00
|
|
|
void CompilerInstance::createCodeCompletionConsumer() {
|
|
|
|
const ParsedSourceLocation &Loc = getFrontendOpts().CodeCompletionAt;
|
2010-08-05 00:47:14 +08:00
|
|
|
if (!CompletionConsumer) {
|
2012-04-12 18:31:12 +08:00
|
|
|
setCodeCompletionConsumer(
|
2010-08-05 00:47:14 +08:00
|
|
|
createCodeCompletionConsumer(getPreprocessor(),
|
|
|
|
Loc.FileName, Loc.Line, Loc.Column,
|
2012-07-03 01:35:10 +08:00
|
|
|
getFrontendOpts().CodeCompleteOpts,
|
2010-08-05 00:47:14 +08:00
|
|
|
llvm::outs()));
|
|
|
|
if (!CompletionConsumer)
|
|
|
|
return;
|
|
|
|
} else if (EnableCodeCompletion(getPreprocessor(), Loc.FileName,
|
|
|
|
Loc.Line, Loc.Column)) {
|
2014-05-22 12:46:25 +08:00
|
|
|
setCodeCompletionConsumer(nullptr);
|
2010-03-16 14:04:47 +08:00
|
|
|
return;
|
2010-08-05 00:47:14 +08:00
|
|
|
}
|
2009-12-01 13:55:20 +08:00
|
|
|
|
|
|
|
if (CompletionConsumer->isOutputBinary() &&
|
2013-06-13 04:44:26 +08:00
|
|
|
llvm::sys::ChangeStdoutToBinary()) {
|
2009-12-01 13:55:20 +08:00
|
|
|
getPreprocessor().getDiagnostics().Report(diag::err_fe_stdout_binary);
|
2014-05-22 12:46:25 +08:00
|
|
|
setCodeCompletionConsumer(nullptr);
|
2009-12-01 13:55:20 +08:00
|
|
|
}
|
2009-11-13 17:36:05 +08:00
|
|
|
}
|
|
|
|
|
2009-11-29 17:57:35 +08:00
|
|
|
void CompilerInstance::createFrontendTimer() {
|
|
|
|
FrontendTimer.reset(new llvm::Timer("Clang front-end timer"));
|
|
|
|
}
|
|
|
|
|
2009-11-13 17:36:05 +08:00
|
|
|
CodeCompleteConsumer *
|
|
|
|
CompilerInstance::createCodeCompletionConsumer(Preprocessor &PP,
|
|
|
|
const std::string &Filename,
|
|
|
|
unsigned Line,
|
|
|
|
unsigned Column,
|
2012-07-03 01:35:10 +08:00
|
|
|
const CodeCompleteOptions &Opts,
|
2011-07-23 18:55:15 +08:00
|
|
|
raw_ostream &OS) {
|
2010-08-05 00:47:14 +08:00
|
|
|
if (EnableCodeCompletion(PP, Filename, Line, Column))
|
2014-05-22 12:46:25 +08:00
|
|
|
return nullptr;
|
2009-11-13 17:36:05 +08:00
|
|
|
|
|
|
|
// Set up the creation routine for code-completion.
|
2012-07-03 01:35:10 +08:00
|
|
|
return new PrintingCodeCompleteConsumer(Opts, OS);
|
2009-11-13 17:36:05 +08:00
|
|
|
}
|
2009-11-13 18:37:48 +08:00
|
|
|
|
2011-08-26 06:30:56 +08:00
|
|
|
void CompilerInstance::createSema(TranslationUnitKind TUKind,
|
2010-08-13 07:31:19 +08:00
|
|
|
CodeCompleteConsumer *CompletionConsumer) {
|
|
|
|
TheSema.reset(new Sema(getPreprocessor(), getASTContext(), getASTConsumer(),
|
2011-08-26 06:30:56 +08:00
|
|
|
TUKind, CompletionConsumer));
|
2010-08-13 07:31:19 +08:00
|
|
|
}
|
|
|
|
|
2009-11-13 18:37:48 +08:00
|
|
|
// Output Files
|
|
|
|
|
2014-08-11 14:53:11 +08:00
|
|
|
void CompilerInstance::addOutputFile(const OutputFile &OutFile) {
|
2010-09-18 01:38:48 +08:00
|
|
|
assert(OutFile.OS && "Attempt to add empty stream to output list!");
|
2014-08-11 14:53:11 +08:00
|
|
|
OutputFiles.push_back(OutFile);
|
2009-11-13 18:37:48 +08:00
|
|
|
}
|
|
|
|
|
2010-03-06 20:07:48 +08:00
|
|
|
void CompilerInstance::clearOutputFiles(bool EraseFiles) {
|
2010-09-18 01:38:48 +08:00
|
|
|
for (std::list<OutputFile>::iterator
|
2009-11-13 18:37:48 +08:00
|
|
|
it = OutputFiles.begin(), ie = OutputFiles.end(); it != ie; ++it) {
|
2014-08-11 14:53:11 +08:00
|
|
|
delete it->OS;
|
2010-09-18 01:38:48 +08:00
|
|
|
if (!it->TempFilename.empty()) {
|
2011-03-07 06:25:35 +08:00
|
|
|
if (EraseFiles) {
|
2014-01-11 05:32:14 +08:00
|
|
|
llvm::sys::fs::remove(it->TempFilename);
|
2011-03-07 06:25:35 +08:00
|
|
|
} else {
|
2012-02-05 10:13:05 +08:00
|
|
|
SmallString<128> NewOutFile(it->Filename);
|
2011-03-07 06:25:35 +08:00
|
|
|
|
2010-11-04 06:45:23 +08:00
|
|
|
// If '-working-directory' was passed, the output filename should be
|
|
|
|
// relative to that.
|
2011-03-14 09:13:54 +08:00
|
|
|
FileMgr->FixupRelativePath(NewOutFile);
|
2014-06-12 22:02:15 +08:00
|
|
|
if (std::error_code ec =
|
|
|
|
llvm::sys::fs::rename(it->TempFilename, NewOutFile.str())) {
|
2012-05-17 04:55:58 +08:00
|
|
|
getDiagnostics().Report(diag::err_unable_to_rename_temp)
|
2011-03-07 06:25:35 +08:00
|
|
|
<< it->TempFilename << it->Filename << ec.message();
|
|
|
|
|
2014-01-11 05:32:14 +08:00
|
|
|
llvm::sys::fs::remove(it->TempFilename);
|
2010-09-18 01:38:48 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (!it->Filename.empty() && EraseFiles)
|
2013-06-26 12:32:59 +08:00
|
|
|
llvm::sys::fs::remove(it->Filename);
|
2011-10-08 19:31:46 +08:00
|
|
|
|
2009-11-13 18:37:48 +08:00
|
|
|
}
|
|
|
|
OutputFiles.clear();
|
|
|
|
}
|
|
|
|
|
2009-11-14 02:32:08 +08:00
|
|
|
llvm::raw_fd_ostream *
|
|
|
|
CompilerInstance::createDefaultOutputFile(bool Binary,
|
2011-07-23 18:55:15 +08:00
|
|
|
StringRef InFile,
|
|
|
|
StringRef Extension) {
|
2009-11-14 02:32:08 +08:00
|
|
|
return createOutputFile(getFrontendOpts().OutputFile, Binary,
|
2012-03-03 08:36:06 +08:00
|
|
|
/*RemoveFileOnSignal=*/true, InFile, Extension,
|
|
|
|
/*UseTemporary=*/true);
|
2009-11-14 02:32:08 +08:00
|
|
|
}
|
|
|
|
|
2014-06-04 01:23:34 +08:00
|
|
|
llvm::raw_null_ostream *CompilerInstance::createNullOutputFile() {
|
2014-08-11 14:53:11 +08:00
|
|
|
llvm::raw_null_ostream *OS = new llvm::raw_null_ostream();
|
|
|
|
addOutputFile(OutputFile("", "", OS));
|
|
|
|
return OS;
|
2014-06-04 01:23:34 +08:00
|
|
|
}
|
|
|
|
|
2009-11-14 02:32:08 +08:00
|
|
|
llvm::raw_fd_ostream *
|
2011-07-23 18:55:15 +08:00
|
|
|
CompilerInstance::createOutputFile(StringRef OutputPath,
|
2011-02-01 06:00:42 +08:00
|
|
|
bool Binary, bool RemoveFileOnSignal,
|
2011-07-23 18:55:15 +08:00
|
|
|
StringRef InFile,
|
2011-07-28 08:45:10 +08:00
|
|
|
StringRef Extension,
|
2012-03-03 08:36:02 +08:00
|
|
|
bool UseTemporary,
|
|
|
|
bool CreateMissingDirectories) {
|
2014-08-26 02:17:04 +08:00
|
|
|
std::string OutputPathName, TempPathName;
|
|
|
|
std::error_code EC;
|
|
|
|
llvm::raw_fd_ostream *OS = createOutputFile(
|
|
|
|
OutputPath, EC, Binary, RemoveFileOnSignal, InFile, Extension,
|
|
|
|
UseTemporary, CreateMissingDirectories, &OutputPathName, &TempPathName);
|
2009-11-14 02:32:08 +08:00
|
|
|
if (!OS) {
|
2014-08-26 02:17:04 +08:00
|
|
|
getDiagnostics().Report(diag::err_fe_unable_to_open_output) << OutputPath
|
|
|
|
<< EC.message();
|
2014-05-22 12:46:25 +08:00
|
|
|
return nullptr;
|
2009-11-14 02:32:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Add the output file -- but don't try to remove "-", since this means we are
|
|
|
|
// using stdin.
|
2010-09-18 01:38:48 +08:00
|
|
|
addOutputFile(OutputFile((OutputPathName != "-") ? OutputPathName : "",
|
2014-08-11 14:53:11 +08:00
|
|
|
TempPathName, OS));
|
2009-11-14 02:32:08 +08:00
|
|
|
|
2014-08-11 14:53:11 +08:00
|
|
|
return OS;
|
2009-11-14 02:32:08 +08:00
|
|
|
}
|
|
|
|
|
2014-08-26 02:17:04 +08:00
|
|
|
llvm::raw_fd_ostream *CompilerInstance::createOutputFile(
|
|
|
|
StringRef OutputPath, std::error_code &Error, bool Binary,
|
|
|
|
bool RemoveFileOnSignal, StringRef InFile, StringRef Extension,
|
|
|
|
bool UseTemporary, bool CreateMissingDirectories,
|
|
|
|
std::string *ResultPathName, std::string *TempPathName) {
|
2012-03-03 08:36:02 +08:00
|
|
|
assert((!CreateMissingDirectories || UseTemporary) &&
|
|
|
|
"CreateMissingDirectories is only allowed when using temporary files");
|
|
|
|
|
2010-09-18 01:38:48 +08:00
|
|
|
std::string OutFile, TempFile;
|
2009-11-14 02:32:08 +08:00
|
|
|
if (!OutputPath.empty()) {
|
|
|
|
OutFile = OutputPath;
|
|
|
|
} else if (InFile == "-") {
|
|
|
|
OutFile = "-";
|
|
|
|
} else if (!Extension.empty()) {
|
2013-06-26 12:32:59 +08:00
|
|
|
SmallString<128> Path(InFile);
|
|
|
|
llvm::sys::path::replace_extension(Path, Extension);
|
2009-11-14 02:32:08 +08:00
|
|
|
OutFile = Path.str();
|
|
|
|
} else {
|
|
|
|
OutFile = "-";
|
|
|
|
}
|
2011-07-28 08:45:10 +08:00
|
|
|
|
2014-03-08 04:03:18 +08:00
|
|
|
std::unique_ptr<llvm::raw_fd_ostream> OS;
|
2011-07-28 08:45:10 +08:00
|
|
|
std::string OSFile;
|
|
|
|
|
2013-06-28 02:26:26 +08:00
|
|
|
if (UseTemporary) {
|
|
|
|
if (OutFile == "-")
|
|
|
|
UseTemporary = false;
|
|
|
|
else {
|
|
|
|
llvm::sys::fs::file_status Status;
|
|
|
|
llvm::sys::fs::status(OutputPath, Status);
|
|
|
|
if (llvm::sys::fs::exists(Status)) {
|
|
|
|
// Fail early if we can't write to the final destination.
|
|
|
|
if (!llvm::sys::fs::can_write(OutputPath))
|
2014-05-22 12:46:25 +08:00
|
|
|
return nullptr;
|
2013-06-28 02:26:26 +08:00
|
|
|
|
|
|
|
// Don't use a temporary if the output is a special file. This handles
|
|
|
|
// things like '-o /dev/null'
|
|
|
|
if (!llvm::sys::fs::is_regular_file(Status))
|
|
|
|
UseTemporary = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (UseTemporary) {
|
|
|
|
// Create a temporary file.
|
|
|
|
SmallString<128> TempPath;
|
|
|
|
TempPath = OutFile;
|
|
|
|
TempPath += "-%%%%%%%%";
|
|
|
|
int fd;
|
2014-06-12 22:02:15 +08:00
|
|
|
std::error_code EC =
|
2013-07-06 05:13:58 +08:00
|
|
|
llvm::sys::fs::createUniqueFile(TempPath.str(), fd, TempPath);
|
2013-06-28 11:49:04 +08:00
|
|
|
|
|
|
|
if (CreateMissingDirectories &&
|
2014-06-14 01:20:50 +08:00
|
|
|
EC == llvm::errc::no_such_file_or_directory) {
|
2013-06-28 11:49:04 +08:00
|
|
|
StringRef Parent = llvm::sys::path::parent_path(OutputPath);
|
|
|
|
EC = llvm::sys::fs::create_directories(Parent);
|
|
|
|
if (!EC) {
|
2013-07-06 05:13:58 +08:00
|
|
|
EC = llvm::sys::fs::createUniqueFile(TempPath.str(), fd, TempPath);
|
2013-06-28 11:49:04 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!EC) {
|
2014-08-11 14:53:11 +08:00
|
|
|
OS.reset(new llvm::raw_fd_ostream(fd, /*shouldClose=*/true));
|
2013-06-28 02:26:26 +08:00
|
|
|
OSFile = TempFile = TempPath.str();
|
2010-09-18 01:38:48 +08:00
|
|
|
}
|
2013-06-28 02:26:26 +08:00
|
|
|
// If we failed to create the temporary, fallback to writing to the file
|
|
|
|
// directly. This handles the corner case where we cannot write to the
|
|
|
|
// directory, but can write to the file.
|
2010-09-18 01:38:48 +08:00
|
|
|
}
|
|
|
|
|
2011-07-28 08:45:10 +08:00
|
|
|
if (!OS) {
|
|
|
|
OSFile = OutFile;
|
2014-08-11 14:53:11 +08:00
|
|
|
OS.reset(new llvm::raw_fd_ostream(
|
2014-08-26 02:17:04 +08:00
|
|
|
OSFile, Error,
|
2014-08-11 14:53:11 +08:00
|
|
|
(Binary ? llvm::sys::fs::F_None : llvm::sys::fs::F_Text)));
|
2014-08-26 02:17:04 +08:00
|
|
|
if (Error)
|
2014-05-22 12:46:25 +08:00
|
|
|
return nullptr;
|
2011-07-28 08:45:10 +08:00
|
|
|
}
|
2009-11-14 02:32:08 +08:00
|
|
|
|
2010-09-18 01:38:48 +08:00
|
|
|
// Make sure the out stream file gets removed if we crash.
|
2011-02-01 06:00:42 +08:00
|
|
|
if (RemoveFileOnSignal)
|
2013-06-14 05:02:40 +08:00
|
|
|
llvm::sys::RemoveFileOnSignal(OSFile);
|
2010-09-18 01:38:48 +08:00
|
|
|
|
2009-11-14 02:32:08 +08:00
|
|
|
if (ResultPathName)
|
|
|
|
*ResultPathName = OutFile;
|
2010-09-18 01:38:48 +08:00
|
|
|
if (TempPathName)
|
|
|
|
*TempPathName = TempFile;
|
2009-11-14 02:32:08 +08:00
|
|
|
|
2014-08-11 14:53:11 +08:00
|
|
|
return OS.release();
|
2009-11-14 02:32:08 +08:00
|
|
|
}
|
2009-11-14 15:53:04 +08:00
|
|
|
|
|
|
|
// Initialization Utilities
|
|
|
|
|
2012-11-10 03:40:33 +08:00
|
|
|
bool CompilerInstance::InitializeSourceManager(const FrontendInputFile &Input){
|
|
|
|
return InitializeSourceManager(Input, getDiagnostics(),
|
2012-01-28 03:52:33 +08:00
|
|
|
getFileManager(), getSourceManager(),
|
|
|
|
getFrontendOpts());
|
2009-11-14 15:53:04 +08:00
|
|
|
}
|
|
|
|
|
2012-11-10 03:40:33 +08:00
|
|
|
bool CompilerInstance::InitializeSourceManager(const FrontendInputFile &Input,
|
2011-09-26 07:23:43 +08:00
|
|
|
DiagnosticsEngine &Diags,
|
2009-11-14 15:53:04 +08:00
|
|
|
FileManager &FileMgr,
|
|
|
|
SourceManager &SourceMgr,
|
|
|
|
const FrontendOptions &Opts) {
|
2012-11-10 03:40:33 +08:00
|
|
|
SrcMgr::CharacteristicKind
|
2012-11-10 03:40:39 +08:00
|
|
|
Kind = Input.isSystem() ? SrcMgr::C_System : SrcMgr::C_User;
|
2012-11-10 03:40:33 +08:00
|
|
|
|
2012-11-10 03:40:45 +08:00
|
|
|
if (Input.isBuffer()) {
|
2014-08-29 15:59:55 +08:00
|
|
|
SourceMgr.setMainFileID(SourceMgr.createFileID(
|
|
|
|
std::unique_ptr<llvm::MemoryBuffer>(Input.getBuffer()), Kind));
|
2012-11-10 03:40:45 +08:00
|
|
|
assert(!SourceMgr.getMainFileID().isInvalid() &&
|
|
|
|
"Couldn't establish MainFileID!");
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
StringRef InputFile = Input.getFile();
|
|
|
|
|
2011-09-20 04:40:35 +08:00
|
|
|
// Figure out where to get and map in the main file.
|
|
|
|
if (InputFile != "-") {
|
2013-08-12 21:46:52 +08:00
|
|
|
const FileEntry *File = FileMgr.getFile(InputFile, /*OpenFile=*/true);
|
2010-10-27 05:13:51 +08:00
|
|
|
if (!File) {
|
2009-11-14 15:53:04 +08:00
|
|
|
Diags.Report(diag::err_fe_error_reading) << InputFile;
|
|
|
|
return false;
|
|
|
|
}
|
2012-11-06 06:53:33 +08:00
|
|
|
|
|
|
|
// The natural SourceManager infrastructure can't currently handle named
|
|
|
|
// pipes, but we would at least like to accept them for the main
|
2013-08-12 21:46:52 +08:00
|
|
|
// file. Detect them here, read them with the volatile flag so FileMgr will
|
|
|
|
// pick up the correct size, and simply override their contents as we do for
|
|
|
|
// STDIN.
|
2012-11-06 06:53:33 +08:00
|
|
|
if (File->isNamedPipe()) {
|
2014-10-27 06:44:13 +08:00
|
|
|
auto MB = FileMgr.getBufferForFile(File, /*isVolatile=*/true);
|
|
|
|
if (MB) {
|
2013-08-12 21:46:52 +08:00
|
|
|
// Create a new virtual file that will have the correct size.
|
2014-10-27 06:44:13 +08:00
|
|
|
File = FileMgr.getVirtualFile(InputFile, (*MB)->getBufferSize(), 0);
|
|
|
|
SourceMgr.overrideFileContents(File, std::move(*MB));
|
2013-08-12 21:46:52 +08:00
|
|
|
} else {
|
2014-10-27 06:44:13 +08:00
|
|
|
Diags.Report(diag::err_cannot_open_file) << InputFile
|
|
|
|
<< MB.getError().message();
|
2012-11-06 06:53:33 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2012-11-27 08:04:16 +08:00
|
|
|
|
2014-05-21 09:12:41 +08:00
|
|
|
SourceMgr.setMainFileID(
|
|
|
|
SourceMgr.createFileID(File, SourceLocation(), Kind));
|
2009-11-14 15:53:04 +08:00
|
|
|
} else {
|
2014-07-07 01:43:24 +08:00
|
|
|
llvm::ErrorOr<std::unique_ptr<llvm::MemoryBuffer>> SBOrErr =
|
|
|
|
llvm::MemoryBuffer::getSTDIN();
|
|
|
|
if (std::error_code EC = SBOrErr.getError()) {
|
|
|
|
Diags.Report(diag::err_fe_error_reading_stdin) << EC.message();
|
2009-11-14 15:53:04 +08:00
|
|
|
return false;
|
|
|
|
}
|
2014-07-07 01:43:24 +08:00
|
|
|
std::unique_ptr<llvm::MemoryBuffer> SB = std::move(SBOrErr.get());
|
|
|
|
|
2010-10-27 07:21:25 +08:00
|
|
|
const FileEntry *File = FileMgr.getVirtualFile(SB->getBufferIdentifier(),
|
2010-11-23 16:35:12 +08:00
|
|
|
SB->getBufferSize(), 0);
|
2014-05-21 09:12:41 +08:00
|
|
|
SourceMgr.setMainFileID(
|
|
|
|
SourceMgr.createFileID(File, SourceLocation(), Kind));
|
2014-08-28 04:54:45 +08:00
|
|
|
SourceMgr.overrideFileContents(File, std::move(SB));
|
2009-11-14 15:53:04 +08:00
|
|
|
}
|
|
|
|
|
2010-10-27 05:13:51 +08:00
|
|
|
assert(!SourceMgr.getMainFileID().isInvalid() &&
|
|
|
|
"Couldn't establish MainFileID!");
|
2009-11-14 15:53:04 +08:00
|
|
|
return true;
|
|
|
|
}
|
2010-01-13 08:48:06 +08:00
|
|
|
|
|
|
|
// High-Level Operations
|
|
|
|
|
|
|
|
bool CompilerInstance::ExecuteAction(FrontendAction &Act) {
|
|
|
|
assert(hasDiagnostics() && "Diagnostics engine is not initialized!");
|
|
|
|
assert(!getFrontendOpts().ShowHelp && "Client must handle '-help'!");
|
|
|
|
assert(!getFrontendOpts().ShowVersion && "Client must handle '-version'!");
|
|
|
|
|
|
|
|
// FIXME: Take this as an argument, once all the APIs we used have moved to
|
|
|
|
// taking it as an input instead of hard-coding llvm::errs.
|
2011-07-23 18:55:15 +08:00
|
|
|
raw_ostream &OS = llvm::errs();
|
2010-01-13 08:48:06 +08:00
|
|
|
|
|
|
|
// Create the target instance.
|
2014-07-06 13:26:44 +08:00
|
|
|
setTarget(TargetInfo::CreateTargetInfo(getDiagnostics(),
|
|
|
|
getInvocation().TargetOpts));
|
2010-01-13 08:48:06 +08:00
|
|
|
if (!hasTarget())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Inform the target of the language options.
|
|
|
|
//
|
|
|
|
// FIXME: We shouldn't need to do this, the target should be immutable once
|
|
|
|
// created. This complexity should be lifted elsewhere.
|
2014-07-06 13:14:24 +08:00
|
|
|
getTarget().adjust(getLangOpts());
|
2010-01-13 08:48:06 +08:00
|
|
|
|
2012-04-17 05:03:30 +08:00
|
|
|
// rewriter project will change target built-in bool type from its default.
|
|
|
|
if (getFrontendOpts().ProgramAction == frontend::RewriteObjC)
|
|
|
|
getTarget().noSignedCharForObjCBool();
|
|
|
|
|
2010-01-13 08:48:06 +08:00
|
|
|
// Validate/process some options.
|
|
|
|
if (getHeaderSearchOpts().Verbose)
|
|
|
|
OS << "clang -cc1 version " CLANG_VERSION_STRING
|
2014-06-06 18:36:22 +08:00
|
|
|
<< " based upon " << BACKEND_PACKAGE_STRING
|
2011-11-02 05:33:06 +08:00
|
|
|
<< " default target " << llvm::sys::getDefaultTargetTriple() << "\n";
|
2010-01-13 08:48:06 +08:00
|
|
|
|
|
|
|
if (getFrontendOpts().ShowTimers)
|
|
|
|
createFrontendTimer();
|
|
|
|
|
2010-03-31 01:33:59 +08:00
|
|
|
if (getFrontendOpts().ShowStats)
|
|
|
|
llvm::EnableStatistics();
|
2011-10-08 19:31:46 +08:00
|
|
|
|
2010-01-13 08:48:06 +08:00
|
|
|
for (unsigned i = 0, e = getFrontendOpts().Inputs.size(); i != e; ++i) {
|
Add support for the static analyzer to synthesize function implementations from external model files.
Currently the analyzer lazily models some functions using 'BodyFarm',
which constructs a fake function implementation that the analyzer
can simulate that approximates the semantics of the function when
it is called. BodyFarm does this by constructing the AST for
such definitions on-the-fly. One strength of BodyFarm
is that all symbols and types referenced by synthesized function
bodies are contextual adapted to the containing translation unit.
The downside is that these ASTs are hardcoded in Clang's own
source code.
A more scalable model is to allow these models to be defined as source
code in separate "model" files and have the analyzer use those
definitions lazily when a function body is needed. Among other things,
it will allow more customization of the analyzer for specific APIs
and platforms.
This patch provides the initial infrastructure for this feature.
It extends BodyFarm to use an abstract API 'CodeInjector' that can be
used to synthesize function bodies. That 'CodeInjector' is
implemented using a new 'ModelInjector' in libFrontend, which lazily
parses a model file and injects the ASTs into the current translation
unit.
Models are currently found by specifying a 'model-path' as an
analyzer option; if no path is specified the CodeInjector is not
used, thus defaulting to the current behavior in the analyzer.
Models currently contain a single function definition, and can
be found by finding the file <function name>.model. This is an
initial starting point for something more rich, but it bootstraps
this feature for future evolution.
This patch was contributed by Gábor Horváth as part of his
Google Summer of Code project.
Some notes:
- This introduces the notion of a "model file" into
FrontendAction and the Preprocessor. This nomenclature
is specific to the static analyzer, but possibly could be
generalized. Essentially these are sources pulled in
exogenously from the principal translation.
Preprocessor gets a 'InitializeForModelFile' and
'FinalizeForModelFile' which could possibly be hoisted out
of Preprocessor if Preprocessor exposed a new API to
change the PragmaHandlers and some other internal pieces. This
can be revisited.
FrontendAction gets a 'isModelParsingAction()' predicate function
used to allow a new FrontendAction to recycle the Preprocessor
and ASTContext. This name could probably be made something
more general (i.e., not tied to 'model files') at the expense
of losing the intent of why it exists. This can be revisited.
- This is a moderate sized patch; it has gone through some amount of
offline code review. Most of the changes to the non-analyzer
parts are fairly small, and would make little sense without
the analyzer changes.
- Most of the analyzer changes are plumbing, with the interesting
behavior being introduced by ModelInjector.cpp and
ModelConsumer.cpp.
- The new functionality introduced by this change is off-by-default.
It requires an analyzer config option to enable.
llvm-svn: 216550
2014-08-27 23:14:15 +08:00
|
|
|
// Reset the ID tables if we are reusing the SourceManager and parsing
|
|
|
|
// regular files.
|
|
|
|
if (hasSourceManager() && !Act.isModelParsingAction())
|
2010-06-08 07:23:50 +08:00
|
|
|
getSourceManager().clearIDTables();
|
2010-01-13 08:48:06 +08:00
|
|
|
|
2012-01-21 00:28:04 +08:00
|
|
|
if (Act.BeginSourceFile(*this, getFrontendOpts().Inputs[i])) {
|
2010-01-13 08:48:06 +08:00
|
|
|
Act.Execute();
|
|
|
|
Act.EndSourceFile();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-12-07 13:52:12 +08:00
|
|
|
// Notify the diagnostic client that all files were processed.
|
|
|
|
getDiagnostics().getClient()->finish();
|
|
|
|
|
2010-04-08 02:47:42 +08:00
|
|
|
if (getDiagnosticOpts().ShowCarets) {
|
2010-11-19 04:06:46 +08:00
|
|
|
// We can have multiple diagnostics sharing one diagnostic client.
|
|
|
|
// Get the total number of warnings/errors from the client.
|
|
|
|
unsigned NumWarnings = getDiagnostics().getClient()->getNumWarnings();
|
|
|
|
unsigned NumErrors = getDiagnostics().getClient()->getNumErrors();
|
2011-10-08 19:31:46 +08:00
|
|
|
|
2010-04-08 02:47:42 +08:00
|
|
|
if (NumWarnings)
|
|
|
|
OS << NumWarnings << " warning" << (NumWarnings == 1 ? "" : "s");
|
|
|
|
if (NumWarnings && NumErrors)
|
|
|
|
OS << " and ";
|
|
|
|
if (NumErrors)
|
|
|
|
OS << NumErrors << " error" << (NumErrors == 1 ? "" : "s");
|
|
|
|
if (NumWarnings || NumErrors)
|
|
|
|
OS << " generated.\n";
|
|
|
|
}
|
2010-01-13 08:48:06 +08:00
|
|
|
|
2010-06-08 07:23:50 +08:00
|
|
|
if (getFrontendOpts().ShowStats && hasFileManager()) {
|
2010-01-13 08:48:06 +08:00
|
|
|
getFileManager().PrintStats();
|
|
|
|
OS << "\n";
|
|
|
|
}
|
|
|
|
|
2010-11-19 05:13:57 +08:00
|
|
|
return !getDiagnostics().getClient()->getNumErrors();
|
2010-01-13 08:48:06 +08:00
|
|
|
}
|
|
|
|
|
2011-09-13 07:31:24 +08:00
|
|
|
/// \brief Determine the appropriate source input kind based on language
|
|
|
|
/// options.
|
|
|
|
static InputKind getSourceInputKindFromOptions(const LangOptions &LangOpts) {
|
|
|
|
if (LangOpts.OpenCL)
|
|
|
|
return IK_OpenCL;
|
|
|
|
if (LangOpts.CUDA)
|
|
|
|
return IK_CUDA;
|
|
|
|
if (LangOpts.ObjC1)
|
|
|
|
return LangOpts.CPlusPlus? IK_ObjCXX : IK_ObjC;
|
|
|
|
return LangOpts.CPlusPlus? IK_CXX : IK_C;
|
|
|
|
}
|
|
|
|
|
2011-11-30 03:06:37 +08:00
|
|
|
/// \brief Compile a module file for the given module, using the options
|
2014-07-20 00:29:28 +08:00
|
|
|
/// provided by the importing compiler instance. Returns true if the module
|
|
|
|
/// was built without errors.
|
|
|
|
static bool compileModuleImpl(CompilerInstance &ImportingInstance,
|
|
|
|
SourceLocation ImportLoc,
|
|
|
|
Module *Module,
|
|
|
|
StringRef ModuleFileName) {
|
2011-11-30 03:06:37 +08:00
|
|
|
ModuleMap &ModMap
|
|
|
|
= ImportingInstance.getPreprocessor().getHeaderSearchInfo().getModuleMap();
|
|
|
|
|
2011-09-13 07:31:24 +08:00
|
|
|
// Construct a compiler invocation for creating this module.
|
2012-02-20 22:00:23 +08:00
|
|
|
IntrusiveRefCntPtr<CompilerInvocation> Invocation
|
2011-09-13 07:31:24 +08:00
|
|
|
(new CompilerInvocation(ImportingInstance.getInvocation()));
|
2011-10-08 19:31:46 +08:00
|
|
|
|
2011-11-30 05:59:16 +08:00
|
|
|
PreprocessorOptions &PPOpts = Invocation->getPreprocessorOpts();
|
|
|
|
|
2011-09-16 04:53:28 +08:00
|
|
|
// For any options that aren't intended to affect how a module is built,
|
|
|
|
// reset them to their default values.
|
2011-11-18 07:01:24 +08:00
|
|
|
Invocation->getLangOpts()->resetNonModularOptions();
|
2011-11-30 05:59:16 +08:00
|
|
|
PPOpts.resetNonModularOptions();
|
2011-10-08 19:31:46 +08:00
|
|
|
|
2013-02-07 08:21:12 +08:00
|
|
|
// Remove any macro definitions that are explicitly ignored by the module.
|
|
|
|
// They aren't supposed to affect how the module is built anyway.
|
|
|
|
const HeaderSearchOptions &HSOpts = Invocation->getHeaderSearchOpts();
|
2014-03-01 22:48:57 +08:00
|
|
|
PPOpts.Macros.erase(
|
|
|
|
std::remove_if(PPOpts.Macros.begin(), PPOpts.Macros.end(),
|
|
|
|
[&HSOpts](const std::pair<std::string, bool> &def) {
|
|
|
|
StringRef MacroDef = def.first;
|
|
|
|
return HSOpts.ModulesIgnoreMacros.count(MacroDef.split('=').first) > 0;
|
|
|
|
}),
|
|
|
|
PPOpts.Macros.end());
|
2013-02-07 08:21:12 +08:00
|
|
|
|
2011-11-16 03:35:01 +08:00
|
|
|
// Note the name of the module we're building.
|
2011-11-30 02:31:39 +08:00
|
|
|
Invocation->getLangOpts()->CurrentModule = Module->getTopLevelModuleName();
|
2011-11-16 03:35:01 +08:00
|
|
|
|
2012-11-30 07:55:25 +08:00
|
|
|
// Make sure that the failed-module structure has been allocated in
|
|
|
|
// the importing instance, and propagate the pointer to the newly-created
|
|
|
|
// instance.
|
|
|
|
PreprocessorOptions &ImportingPPOpts
|
|
|
|
= ImportingInstance.getInvocation().getPreprocessorOpts();
|
|
|
|
if (!ImportingPPOpts.FailedModules)
|
|
|
|
ImportingPPOpts.FailedModules = new PreprocessorOptions::FailedModulesSet;
|
|
|
|
PPOpts.FailedModules = ImportingPPOpts.FailedModules;
|
|
|
|
|
2011-11-30 05:59:16 +08:00
|
|
|
// If there is a module map file, build the module using the module map.
|
2011-09-16 04:53:28 +08:00
|
|
|
// Set up the inputs/outputs so that we build the module from its umbrella
|
|
|
|
// header.
|
2011-09-13 07:31:24 +08:00
|
|
|
FrontendOptions &FrontendOpts = Invocation->getFrontendOpts();
|
2011-09-14 07:15:45 +08:00
|
|
|
FrontendOpts.OutputFile = ModuleFileName.str();
|
2011-09-13 07:31:24 +08:00
|
|
|
FrontendOpts.DisableFree = false;
|
2013-01-25 08:45:27 +08:00
|
|
|
FrontendOpts.GenerateGlobalModuleIndex = false;
|
2011-09-13 07:31:24 +08:00
|
|
|
FrontendOpts.Inputs.clear();
|
2011-11-30 05:59:16 +08:00
|
|
|
InputKind IK = getSourceInputKindFromOptions(*Invocation->getLangOpts());
|
2011-10-08 19:31:46 +08:00
|
|
|
|
2011-11-30 05:59:16 +08:00
|
|
|
// Don't free the remapped file buffers; they are owned by our caller.
|
|
|
|
PPOpts.RetainRemappedFileBuffers = true;
|
|
|
|
|
|
|
|
Invocation->getDiagnosticOpts().VerifyDiagnostics = 0;
|
2011-09-14 07:20:27 +08:00
|
|
|
assert(ImportingInstance.getInvocation().getModuleHash() ==
|
2011-11-30 05:59:16 +08:00
|
|
|
Invocation->getModuleHash() && "Module hash mismatch!");
|
|
|
|
|
2011-09-13 07:31:24 +08:00
|
|
|
// Construct a compiler instance that will be used to actually create the
|
|
|
|
// module.
|
2014-04-23 20:57:01 +08:00
|
|
|
CompilerInstance Instance(/*BuildingModule=*/true);
|
2011-09-13 07:31:24 +08:00
|
|
|
Instance.setInvocation(&*Invocation);
|
2013-05-04 06:58:43 +08:00
|
|
|
|
|
|
|
Instance.createDiagnostics(new ForwardingDiagnosticConsumer(
|
|
|
|
ImportingInstance.getDiagnosticClient()),
|
2013-05-04 07:07:45 +08:00
|
|
|
/*ShouldOwnClient=*/true);
|
2012-12-01 02:38:50 +08:00
|
|
|
|
2014-02-21 05:59:23 +08:00
|
|
|
Instance.setVirtualFileSystem(&ImportingInstance.getVirtualFileSystem());
|
|
|
|
|
2012-12-01 06:11:57 +08:00
|
|
|
// Note that this module is part of the module build stack, so that we
|
2012-12-01 02:38:50 +08:00
|
|
|
// can detect cycles in the module graph.
|
2014-03-01 05:16:07 +08:00
|
|
|
Instance.setFileManager(&ImportingInstance.getFileManager());
|
2012-12-01 02:38:50 +08:00
|
|
|
Instance.createSourceManager(Instance.getFileManager());
|
|
|
|
SourceManager &SourceMgr = Instance.getSourceManager();
|
2012-12-01 06:11:57 +08:00
|
|
|
SourceMgr.setModuleBuildStack(
|
|
|
|
ImportingInstance.getSourceManager().getModuleBuildStack());
|
|
|
|
SourceMgr.pushModuleBuildStack(Module->getTopLevelModuleName(),
|
2012-12-01 02:38:50 +08:00
|
|
|
FullSourceLoc(ImportLoc, ImportingInstance.getSourceManager()));
|
|
|
|
|
2014-06-20 03:36:03 +08:00
|
|
|
// If we're collecting module dependencies, we need to share a collector
|
|
|
|
// between all of the module CompilerInstances.
|
|
|
|
Instance.setModuleDepCollector(ImportingInstance.getModuleDepCollector());
|
|
|
|
|
Use the same SourceManager for ModuleMaps and compilations.
This allows using virtual file mappings on the original SourceManager to
map in virtual module.map files. Without this patch, the ModuleMap
search will find a module.map file (as the FileEntry exists in the
FileManager), but will be unable to get the content from the
SourceManager (as ModuleMap previously created its own SourceManager).
Two problems needed to be fixed which this patch exposed:
1. Storing the inferred module map
When writing out a module, the ASTWriter stores the names of the files
in the main source manager; when loading the AST again, the ASTReader
errs out if such a file is found missing, unless it is overridden.
Previously CompilerInstance's compileModule method would store the
inferred module map to a temporary file; the problem with this approach
is that now that the module map is handled by the main source manager,
the ASTWriter stores the name of the temporary module map as source to
the compilation; later, when the module is loaded, the temporary file
has already been deleted, which leads to a compilation error. This patch
changes the inferred module map to instead inject a virtual file into
the source manager. This both saves some disk IO, and works with how the
ASTWriter/ASTReader handle overridden source files.
2. Changing test input in test/Modules/Inputs/*
Now that the module map file is handled by the main source manager, the
VerifyDiagnosticConsumer will not ignore diagnostics created while
parsing the module map file. The module test test/Modules/renamed.m uses
-I test/Modules/Inputs and triggers recursive loading of all module maps
in test/Modules/Inputs, some of which had conflicting names, thus
leading errors while parsing the module maps. Those diagnostics already
occur on trunk, but before this patch they would not break the test, as
they were ignored by the VerifyDiagnosticConsumer. This patch thus
changes the module maps that have been recently introduced which broke
the invariant of compatible modules maps in test/Modules/Inputs.
llvm-svn: 193314
2013-10-24 15:51:24 +08:00
|
|
|
// Get or create the module map that we'll use to build this module.
|
|
|
|
std::string InferredModuleMapContent;
|
|
|
|
if (const FileEntry *ModuleMapFile =
|
|
|
|
ModMap.getContainingModuleMapFile(Module)) {
|
|
|
|
// Use the module map where this module resides.
|
|
|
|
FrontendOpts.Inputs.push_back(
|
|
|
|
FrontendInputFile(ModuleMapFile->getName(), IK));
|
|
|
|
} else {
|
|
|
|
llvm::raw_string_ostream OS(InferredModuleMapContent);
|
|
|
|
Module->print(OS);
|
|
|
|
OS.flush();
|
|
|
|
FrontendOpts.Inputs.push_back(
|
|
|
|
FrontendInputFile("__inferred_module.map", IK));
|
|
|
|
|
2014-08-28 04:03:29 +08:00
|
|
|
std::unique_ptr<llvm::MemoryBuffer> ModuleMapBuffer =
|
Use the same SourceManager for ModuleMaps and compilations.
This allows using virtual file mappings on the original SourceManager to
map in virtual module.map files. Without this patch, the ModuleMap
search will find a module.map file (as the FileEntry exists in the
FileManager), but will be unable to get the content from the
SourceManager (as ModuleMap previously created its own SourceManager).
Two problems needed to be fixed which this patch exposed:
1. Storing the inferred module map
When writing out a module, the ASTWriter stores the names of the files
in the main source manager; when loading the AST again, the ASTReader
errs out if such a file is found missing, unless it is overridden.
Previously CompilerInstance's compileModule method would store the
inferred module map to a temporary file; the problem with this approach
is that now that the module map is handled by the main source manager,
the ASTWriter stores the name of the temporary module map as source to
the compilation; later, when the module is loaded, the temporary file
has already been deleted, which leads to a compilation error. This patch
changes the inferred module map to instead inject a virtual file into
the source manager. This both saves some disk IO, and works with how the
ASTWriter/ASTReader handle overridden source files.
2. Changing test input in test/Modules/Inputs/*
Now that the module map file is handled by the main source manager, the
VerifyDiagnosticConsumer will not ignore diagnostics created while
parsing the module map file. The module test test/Modules/renamed.m uses
-I test/Modules/Inputs and triggers recursive loading of all module maps
in test/Modules/Inputs, some of which had conflicting names, thus
leading errors while parsing the module maps. Those diagnostics already
occur on trunk, but before this patch they would not break the test, as
they were ignored by the VerifyDiagnosticConsumer. This patch thus
changes the module maps that have been recently introduced which broke
the invariant of compatible modules maps in test/Modules/Inputs.
llvm-svn: 193314
2013-10-24 15:51:24 +08:00
|
|
|
llvm::MemoryBuffer::getMemBuffer(InferredModuleMapContent);
|
|
|
|
ModuleMapFile = Instance.getFileManager().getVirtualFile(
|
|
|
|
"__inferred_module.map", InferredModuleMapContent.size(), 0);
|
2014-08-28 04:54:45 +08:00
|
|
|
SourceMgr.overrideFileContents(ModuleMapFile, std::move(ModuleMapBuffer));
|
Use the same SourceManager for ModuleMaps and compilations.
This allows using virtual file mappings on the original SourceManager to
map in virtual module.map files. Without this patch, the ModuleMap
search will find a module.map file (as the FileEntry exists in the
FileManager), but will be unable to get the content from the
SourceManager (as ModuleMap previously created its own SourceManager).
Two problems needed to be fixed which this patch exposed:
1. Storing the inferred module map
When writing out a module, the ASTWriter stores the names of the files
in the main source manager; when loading the AST again, the ASTReader
errs out if such a file is found missing, unless it is overridden.
Previously CompilerInstance's compileModule method would store the
inferred module map to a temporary file; the problem with this approach
is that now that the module map is handled by the main source manager,
the ASTWriter stores the name of the temporary module map as source to
the compilation; later, when the module is loaded, the temporary file
has already been deleted, which leads to a compilation error. This patch
changes the inferred module map to instead inject a virtual file into
the source manager. This both saves some disk IO, and works with how the
ASTWriter/ASTReader handle overridden source files.
2. Changing test input in test/Modules/Inputs/*
Now that the module map file is handled by the main source manager, the
VerifyDiagnosticConsumer will not ignore diagnostics created while
parsing the module map file. The module test test/Modules/renamed.m uses
-I test/Modules/Inputs and triggers recursive loading of all module maps
in test/Modules/Inputs, some of which had conflicting names, thus
leading errors while parsing the module maps. Those diagnostics already
occur on trunk, but before this patch they would not break the test, as
they were ignored by the VerifyDiagnosticConsumer. This patch thus
changes the module maps that have been recently introduced which broke
the invariant of compatible modules maps in test/Modules/Inputs.
llvm-svn: 193314
2013-10-24 15:51:24 +08:00
|
|
|
}
|
2012-12-01 02:38:50 +08:00
|
|
|
|
2014-08-09 08:57:23 +08:00
|
|
|
// Construct a module-generating action. Passing through the module map is
|
2014-04-15 02:00:01 +08:00
|
|
|
// safe because the FileManager is shared between the compiler instances.
|
2014-08-09 08:57:23 +08:00
|
|
|
GenerateModuleAction CreateModuleAction(
|
|
|
|
ModMap.getModuleMapFileForUniquing(Module), Module->IsSystem);
|
2014-10-14 10:08:30 +08:00
|
|
|
|
|
|
|
ImportingInstance.getDiagnostics().Report(ImportLoc,
|
|
|
|
diag::remark_module_build)
|
|
|
|
<< Module->Name << ModuleFileName;
|
|
|
|
|
2011-10-04 08:21:21 +08:00
|
|
|
// Execute the action to actually build the module in-place. Use a separate
|
|
|
|
// thread so that we get a stack large enough.
|
|
|
|
const unsigned ThreadStackSize = 8 << 20;
|
|
|
|
llvm::CrashRecoveryContext CRC;
|
2014-03-05 05:50:01 +08:00
|
|
|
CRC.RunSafelyOnThread([&]() { Instance.ExecuteAction(CreateModuleAction); },
|
|
|
|
ThreadStackSize);
|
2013-05-04 06:58:43 +08:00
|
|
|
|
2014-10-14 10:08:30 +08:00
|
|
|
ImportingInstance.getDiagnostics().Report(ImportLoc,
|
|
|
|
diag::remark_module_build_done)
|
|
|
|
<< Module->Name;
|
|
|
|
|
2011-11-30 05:59:16 +08:00
|
|
|
// Delete the temporary module map file.
|
|
|
|
// FIXME: Even though we're executing under crash protection, it would still
|
|
|
|
// be nice to do this with RemoveFileOnSignal when we can. However, that
|
|
|
|
// doesn't make sense for all clients, so clean this up manually.
|
2012-10-15 03:50:53 +08:00
|
|
|
Instance.clearOutputFiles(/*EraseFiles=*/true);
|
2013-01-24 06:38:11 +08:00
|
|
|
|
|
|
|
// We've rebuilt a module. If we're allowed to generate or update the global
|
|
|
|
// module index, record that fact in the importing compiler instance.
|
2013-01-25 08:45:27 +08:00
|
|
|
if (ImportingInstance.getFrontendOpts().GenerateGlobalModuleIndex) {
|
2013-01-24 06:38:11 +08:00
|
|
|
ImportingInstance.setBuildGlobalModuleIndex(true);
|
|
|
|
}
|
2014-07-20 00:29:28 +08:00
|
|
|
|
|
|
|
return !Instance.getDiagnostics().hasErrorOccurred();
|
2011-10-08 19:31:46 +08:00
|
|
|
}
|
2011-09-13 07:31:24 +08:00
|
|
|
|
2014-06-18 06:35:27 +08:00
|
|
|
static bool compileAndLoadModule(CompilerInstance &ImportingInstance,
|
|
|
|
SourceLocation ImportLoc,
|
2014-07-20 00:29:28 +08:00
|
|
|
SourceLocation ModuleNameLoc, Module *Module,
|
2014-06-18 06:35:27 +08:00
|
|
|
StringRef ModuleFileName) {
|
2014-09-27 06:42:23 +08:00
|
|
|
DiagnosticsEngine &Diags = ImportingInstance.getDiagnostics();
|
|
|
|
|
2014-07-20 00:29:28 +08:00
|
|
|
auto diagnoseBuildFailure = [&] {
|
2014-09-27 06:42:23 +08:00
|
|
|
Diags.Report(ModuleNameLoc, diag::err_module_not_built)
|
2014-07-20 00:29:28 +08:00
|
|
|
<< Module->Name << SourceRange(ImportLoc, ModuleNameLoc);
|
|
|
|
};
|
|
|
|
|
2014-04-06 11:21:44 +08:00
|
|
|
// FIXME: have LockFileManager return an error_code so that we can
|
|
|
|
// avoid the mkdir when the directory already exists.
|
|
|
|
StringRef Dir = llvm::sys::path::parent_path(ModuleFileName);
|
|
|
|
llvm::sys::fs::create_directories(Dir);
|
|
|
|
|
|
|
|
while (1) {
|
2014-06-18 06:35:27 +08:00
|
|
|
unsigned ModuleLoadCapabilities = ASTReader::ARR_Missing;
|
2014-04-06 11:21:44 +08:00
|
|
|
llvm::LockFileManager Locked(ModuleFileName);
|
|
|
|
switch (Locked) {
|
|
|
|
case llvm::LockFileManager::LFS_Error:
|
2014-09-27 06:42:23 +08:00
|
|
|
Diags.Report(ModuleNameLoc, diag::err_module_lock_failure)
|
|
|
|
<< Module->Name;
|
2014-06-18 06:35:27 +08:00
|
|
|
return false;
|
2014-04-06 11:21:44 +08:00
|
|
|
|
|
|
|
case llvm::LockFileManager::LFS_Owned:
|
2014-06-18 06:35:27 +08:00
|
|
|
// We're responsible for building the module ourselves.
|
2014-07-20 00:29:28 +08:00
|
|
|
if (!compileModuleImpl(ImportingInstance, ModuleNameLoc, Module,
|
|
|
|
ModuleFileName)) {
|
|
|
|
diagnoseBuildFailure();
|
|
|
|
return false;
|
|
|
|
}
|
2014-04-06 11:21:44 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case llvm::LockFileManager::LFS_Shared:
|
|
|
|
// Someone else is responsible for building the module. Wait for them to
|
|
|
|
// finish.
|
|
|
|
if (Locked.waitForUnlock() == llvm::LockFileManager::Res_OwnerDied)
|
|
|
|
continue; // try again to get the lock.
|
2014-06-18 06:35:27 +08:00
|
|
|
ModuleLoadCapabilities |= ASTReader::ARR_OutOfDate;
|
|
|
|
break;
|
2014-04-06 11:21:44 +08:00
|
|
|
}
|
|
|
|
|
2014-06-18 06:35:27 +08:00
|
|
|
// Try to read the module file, now that we've compiled it.
|
|
|
|
ASTReader::ASTReadResult ReadResult =
|
|
|
|
ImportingInstance.getModuleManager()->ReadAST(
|
2014-10-22 10:05:46 +08:00
|
|
|
ModuleFileName, serialization::MK_ImplicitModule, ImportLoc,
|
2014-06-18 06:35:27 +08:00
|
|
|
ModuleLoadCapabilities);
|
|
|
|
|
|
|
|
if (ReadResult == ASTReader::OutOfDate &&
|
|
|
|
Locked == llvm::LockFileManager::LFS_Shared) {
|
|
|
|
// The module may be out of date in the presence of file system races,
|
|
|
|
// or if one of its imports depends on header search paths that are not
|
|
|
|
// consistent with this ImportingInstance. Try again...
|
|
|
|
continue;
|
|
|
|
} else if (ReadResult == ASTReader::Missing) {
|
2014-07-20 00:29:28 +08:00
|
|
|
diagnoseBuildFailure();
|
2014-09-27 06:42:23 +08:00
|
|
|
} else if (ReadResult != ASTReader::Success &&
|
|
|
|
!Diags.hasErrorOccurred()) {
|
|
|
|
// The ASTReader didn't diagnose the error, so conservatively report it.
|
|
|
|
diagnoseBuildFailure();
|
2014-06-18 06:35:27 +08:00
|
|
|
}
|
|
|
|
return ReadResult == ASTReader::Success;
|
2014-04-06 11:21:44 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-03-20 08:22:05 +08:00
|
|
|
/// \brief Diagnose differences between the current definition of the given
|
|
|
|
/// configuration macro and the definition provided on the command line.
|
|
|
|
static void checkConfigMacro(Preprocessor &PP, StringRef ConfigMacro,
|
|
|
|
Module *Mod, SourceLocation ImportLoc) {
|
|
|
|
IdentifierInfo *Id = PP.getIdentifierInfo(ConfigMacro);
|
|
|
|
SourceManager &SourceMgr = PP.getSourceManager();
|
|
|
|
|
|
|
|
// If this identifier has never had a macro definition, then it could
|
|
|
|
// not have changed.
|
|
|
|
if (!Id->hadMacroDefinition())
|
|
|
|
return;
|
|
|
|
|
|
|
|
// If this identifier does not currently have a macro definition,
|
|
|
|
// check whether it had one on the command line.
|
|
|
|
if (!Id->hasMacroDefinition()) {
|
2013-03-27 01:17:01 +08:00
|
|
|
MacroDirective::DefInfo LatestDef =
|
|
|
|
PP.getMacroDirectiveHistory(Id)->getDefinition();
|
|
|
|
for (MacroDirective::DefInfo Def = LatestDef; Def;
|
|
|
|
Def = Def.getPreviousDefinition()) {
|
|
|
|
FileID FID = SourceMgr.getFileID(Def.getLocation());
|
2013-03-20 08:22:05 +08:00
|
|
|
if (FID.isInvalid())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// We only care about the predefines buffer.
|
2013-04-03 11:16:36 +08:00
|
|
|
if (FID != PP.getPredefinesFileID())
|
2013-03-20 08:22:05 +08:00
|
|
|
continue;
|
|
|
|
|
|
|
|
// This macro was defined on the command line, then #undef'd later.
|
|
|
|
// Complain.
|
|
|
|
PP.Diag(ImportLoc, diag::warn_module_config_macro_undef)
|
|
|
|
<< true << ConfigMacro << Mod->getFullModuleName();
|
2013-03-27 01:17:01 +08:00
|
|
|
if (LatestDef.isUndefined())
|
|
|
|
PP.Diag(LatestDef.getUndefLocation(), diag::note_module_def_undef_here)
|
2013-03-20 08:22:05 +08:00
|
|
|
<< true;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Okay: no definition in the predefines buffer.
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// This identifier has a macro definition. Check whether we had a definition
|
|
|
|
// on the command line.
|
2013-03-27 01:17:01 +08:00
|
|
|
MacroDirective::DefInfo LatestDef =
|
|
|
|
PP.getMacroDirectiveHistory(Id)->getDefinition();
|
|
|
|
MacroDirective::DefInfo PredefinedDef;
|
|
|
|
for (MacroDirective::DefInfo Def = LatestDef; Def;
|
|
|
|
Def = Def.getPreviousDefinition()) {
|
|
|
|
FileID FID = SourceMgr.getFileID(Def.getLocation());
|
2013-03-20 08:22:05 +08:00
|
|
|
if (FID.isInvalid())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// We only care about the predefines buffer.
|
2013-04-03 11:16:36 +08:00
|
|
|
if (FID != PP.getPredefinesFileID())
|
2013-03-20 08:22:05 +08:00
|
|
|
continue;
|
|
|
|
|
2013-03-27 01:17:01 +08:00
|
|
|
PredefinedDef = Def;
|
2013-03-20 08:22:05 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If there was no definition for this macro in the predefines buffer,
|
|
|
|
// complain.
|
2013-03-27 01:17:01 +08:00
|
|
|
if (!PredefinedDef ||
|
|
|
|
(!PredefinedDef.getLocation().isValid() &&
|
|
|
|
PredefinedDef.getUndefLocation().isValid())) {
|
2013-03-20 08:22:05 +08:00
|
|
|
PP.Diag(ImportLoc, diag::warn_module_config_macro_undef)
|
|
|
|
<< false << ConfigMacro << Mod->getFullModuleName();
|
2013-03-27 01:17:01 +08:00
|
|
|
PP.Diag(LatestDef.getLocation(), diag::note_module_def_undef_here)
|
2013-03-20 08:22:05 +08:00
|
|
|
<< false;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the current macro definition is the same as the predefined macro
|
|
|
|
// definition, it's okay.
|
2013-03-27 01:17:01 +08:00
|
|
|
if (LatestDef.getMacroInfo() == PredefinedDef.getMacroInfo() ||
|
2013-04-04 01:39:30 +08:00
|
|
|
LatestDef.getMacroInfo()->isIdenticalTo(*PredefinedDef.getMacroInfo(),PP,
|
|
|
|
/*Syntactically=*/true))
|
2013-03-20 08:22:05 +08:00
|
|
|
return;
|
|
|
|
|
|
|
|
// The macro definitions differ.
|
|
|
|
PP.Diag(ImportLoc, diag::warn_module_config_macro_undef)
|
|
|
|
<< false << ConfigMacro << Mod->getFullModuleName();
|
2013-03-27 01:17:01 +08:00
|
|
|
PP.Diag(LatestDef.getLocation(), diag::note_module_def_undef_here)
|
2013-03-20 08:22:05 +08:00
|
|
|
<< false;
|
|
|
|
}
|
|
|
|
|
2013-03-26 05:19:16 +08:00
|
|
|
/// \brief Write a new timestamp file with the given path.
|
|
|
|
static void writeTimestampFile(StringRef TimestampFile) {
|
2014-08-26 02:17:04 +08:00
|
|
|
std::error_code EC;
|
|
|
|
llvm::raw_fd_ostream Out(TimestampFile.str(), EC, llvm::sys::fs::F_None);
|
2013-03-26 05:19:16 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief Prune the module cache of modules that haven't been accessed in
|
|
|
|
/// a long time.
|
|
|
|
static void pruneModuleCache(const HeaderSearchOptions &HSOpts) {
|
|
|
|
struct stat StatBuf;
|
|
|
|
llvm::SmallString<128> TimestampFile;
|
|
|
|
TimestampFile = HSOpts.ModuleCachePath;
|
|
|
|
llvm::sys::path::append(TimestampFile, "modules.timestamp");
|
|
|
|
|
|
|
|
// Try to stat() the timestamp file.
|
|
|
|
if (::stat(TimestampFile.c_str(), &StatBuf)) {
|
|
|
|
// If the timestamp file wasn't there, create one now.
|
|
|
|
if (errno == ENOENT) {
|
|
|
|
writeTimestampFile(TimestampFile);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check whether the time stamp is older than our pruning interval.
|
|
|
|
// If not, do nothing.
|
|
|
|
time_t TimeStampModTime = StatBuf.st_mtime;
|
2014-05-22 12:46:25 +08:00
|
|
|
time_t CurrentTime = time(nullptr);
|
2013-03-30 01:39:43 +08:00
|
|
|
if (CurrentTime - TimeStampModTime <= time_t(HSOpts.ModuleCachePruneInterval))
|
2013-03-26 05:19:16 +08:00
|
|
|
return;
|
|
|
|
|
|
|
|
// Write a new timestamp file so that nobody else attempts to prune.
|
|
|
|
// There is a benign race condition here, if two Clang instances happen to
|
|
|
|
// notice at the same time that the timestamp is out-of-date.
|
|
|
|
writeTimestampFile(TimestampFile);
|
|
|
|
|
|
|
|
// Walk the entire module cache, looking for unused module files and module
|
|
|
|
// indices.
|
2014-06-12 22:02:15 +08:00
|
|
|
std::error_code EC;
|
2013-03-26 05:19:16 +08:00
|
|
|
SmallString<128> ModuleCachePathNative;
|
|
|
|
llvm::sys::path::native(HSOpts.ModuleCachePath, ModuleCachePathNative);
|
|
|
|
for (llvm::sys::fs::directory_iterator
|
|
|
|
Dir(ModuleCachePathNative.str(), EC), DirEnd;
|
|
|
|
Dir != DirEnd && !EC; Dir.increment(EC)) {
|
|
|
|
// If we don't have a directory, there's nothing to look into.
|
2013-07-17 12:23:07 +08:00
|
|
|
if (!llvm::sys::fs::is_directory(Dir->path()))
|
2013-03-26 05:19:16 +08:00
|
|
|
continue;
|
|
|
|
|
|
|
|
// Walk all of the files within this directory.
|
|
|
|
for (llvm::sys::fs::directory_iterator File(Dir->path(), EC), FileEnd;
|
|
|
|
File != FileEnd && !EC; File.increment(EC)) {
|
|
|
|
// We only care about module and global module index files.
|
Add an option to allow Clang verify source files for a module only once during
the build
When Clang loads the module, it verifies the user source files that the module
was built from. If any file was changed, the module is rebuilt. There are two
problems with this:
1. correctness: we don't verify system files (there are too many of them, and
stat'ing all of them would take a lot of time);
2. performance: the same module file is verified again and again during a
single build.
This change allows the build system to optimize source file verification. The
idea is based on the fact that while the project is being built, the source
files don't change. This allows us to verify the module only once during a
single build session. The build system passes a flag,
-fbuild-session-timestamp=, to inform Clang of the time when the build started.
The build system also requests to enable this feature by passing
-fmodules-validate-once-per-build-session. If these flags are not passed, the
behavior is not changed. When Clang verifies the module the first time, it
writes out a timestamp file. Then, when Clang loads the module the second
time, it finds a timestamp file, so it can compare the verification timestamp
of the module with the time when the build started. If the verification
timestamp is too old, the module is verified again, and the timestamp file is
updated.
llvm-svn: 201224
2014-02-12 18:33:14 +08:00
|
|
|
StringRef Extension = llvm::sys::path::extension(File->path());
|
|
|
|
if (Extension != ".pcm" && Extension != ".timestamp" &&
|
|
|
|
llvm::sys::path::filename(File->path()) != "modules.idx")
|
2013-03-26 05:19:16 +08:00
|
|
|
continue;
|
|
|
|
|
|
|
|
// Look at this file. If we can't stat it, there's nothing interesting
|
|
|
|
// there.
|
Add an option to allow Clang verify source files for a module only once during
the build
When Clang loads the module, it verifies the user source files that the module
was built from. If any file was changed, the module is rebuilt. There are two
problems with this:
1. correctness: we don't verify system files (there are too many of them, and
stat'ing all of them would take a lot of time);
2. performance: the same module file is verified again and again during a
single build.
This change allows the build system to optimize source file verification. The
idea is based on the fact that while the project is being built, the source
files don't change. This allows us to verify the module only once during a
single build session. The build system passes a flag,
-fbuild-session-timestamp=, to inform Clang of the time when the build started.
The build system also requests to enable this feature by passing
-fmodules-validate-once-per-build-session. If these flags are not passed, the
behavior is not changed. When Clang verifies the module the first time, it
writes out a timestamp file. Then, when Clang loads the module the second
time, it finds a timestamp file, so it can compare the verification timestamp
of the module with the time when the build started. If the verification
timestamp is too old, the module is verified again, and the timestamp file is
updated.
llvm-svn: 201224
2014-02-12 18:33:14 +08:00
|
|
|
if (::stat(File->path().c_str(), &StatBuf))
|
2013-03-26 05:19:16 +08:00
|
|
|
continue;
|
|
|
|
|
|
|
|
// If the file has been used recently enough, leave it there.
|
|
|
|
time_t FileAccessTime = StatBuf.st_atime;
|
2013-03-30 01:39:43 +08:00
|
|
|
if (CurrentTime - FileAccessTime <=
|
|
|
|
time_t(HSOpts.ModuleCachePruneAfter)) {
|
2013-03-26 05:19:16 +08:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remove the file.
|
Add an option to allow Clang verify source files for a module only once during
the build
When Clang loads the module, it verifies the user source files that the module
was built from. If any file was changed, the module is rebuilt. There are two
problems with this:
1. correctness: we don't verify system files (there are too many of them, and
stat'ing all of them would take a lot of time);
2. performance: the same module file is verified again and again during a
single build.
This change allows the build system to optimize source file verification. The
idea is based on the fact that while the project is being built, the source
files don't change. This allows us to verify the module only once during a
single build session. The build system passes a flag,
-fbuild-session-timestamp=, to inform Clang of the time when the build started.
The build system also requests to enable this feature by passing
-fmodules-validate-once-per-build-session. If these flags are not passed, the
behavior is not changed. When Clang verifies the module the first time, it
writes out a timestamp file. Then, when Clang loads the module the second
time, it finds a timestamp file, so it can compare the verification timestamp
of the module with the time when the build started. If the verification
timestamp is too old, the module is verified again, and the timestamp file is
updated.
llvm-svn: 201224
2014-02-12 18:33:14 +08:00
|
|
|
llvm::sys::fs::remove(File->path());
|
|
|
|
|
|
|
|
// Remove the timestamp file.
|
|
|
|
std::string TimpestampFilename = File->path() + ".timestamp";
|
|
|
|
llvm::sys::fs::remove(TimpestampFilename);
|
2013-03-26 05:19:16 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// If we removed all of the files in the directory, remove the directory
|
|
|
|
// itself.
|
Add an option to allow Clang verify source files for a module only once during
the build
When Clang loads the module, it verifies the user source files that the module
was built from. If any file was changed, the module is rebuilt. There are two
problems with this:
1. correctness: we don't verify system files (there are too many of them, and
stat'ing all of them would take a lot of time);
2. performance: the same module file is verified again and again during a
single build.
This change allows the build system to optimize source file verification. The
idea is based on the fact that while the project is being built, the source
files don't change. This allows us to verify the module only once during a
single build session. The build system passes a flag,
-fbuild-session-timestamp=, to inform Clang of the time when the build started.
The build system also requests to enable this feature by passing
-fmodules-validate-once-per-build-session. If these flags are not passed, the
behavior is not changed. When Clang verifies the module the first time, it
writes out a timestamp file. Then, when Clang loads the module the second
time, it finds a timestamp file, so it can compare the verification timestamp
of the module with the time when the build started. If the verification
timestamp is too old, the module is verified again, and the timestamp file is
updated.
llvm-svn: 201224
2014-02-12 18:33:14 +08:00
|
|
|
if (llvm::sys::fs::directory_iterator(Dir->path(), EC) ==
|
|
|
|
llvm::sys::fs::directory_iterator() && !EC)
|
2014-01-11 05:32:14 +08:00
|
|
|
llvm::sys::fs::remove(Dir->path());
|
2013-03-26 05:19:16 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-23 20:57:01 +08:00
|
|
|
void CompilerInstance::createModuleManager() {
|
|
|
|
if (!ModuleManager) {
|
|
|
|
if (!hasASTContext())
|
|
|
|
createASTContext();
|
|
|
|
|
|
|
|
// If we're not recursively building a module, check whether we
|
|
|
|
// need to prune the module cache.
|
|
|
|
if (getSourceManager().getModuleBuildStack().empty() &&
|
|
|
|
getHeaderSearchOpts().ModuleCachePruneInterval > 0 &&
|
|
|
|
getHeaderSearchOpts().ModuleCachePruneAfter > 0) {
|
|
|
|
pruneModuleCache(getHeaderSearchOpts());
|
|
|
|
}
|
|
|
|
|
|
|
|
HeaderSearchOptions &HSOpts = getHeaderSearchOpts();
|
|
|
|
std::string Sysroot = HSOpts.Sysroot;
|
|
|
|
const PreprocessorOptions &PPOpts = getPreprocessorOpts();
|
|
|
|
ModuleManager = new ASTReader(getPreprocessor(), *Context,
|
|
|
|
Sysroot.empty() ? "" : Sysroot.c_str(),
|
|
|
|
PPOpts.DisablePCHValidation,
|
|
|
|
/*AllowASTWithCompilerErrors=*/false,
|
|
|
|
/*AllowConfigurationMismatch=*/false,
|
|
|
|
HSOpts.ModulesValidateSystemHeaders,
|
|
|
|
getFrontendOpts().UseGlobalModuleIndex);
|
|
|
|
if (hasASTConsumer()) {
|
|
|
|
ModuleManager->setDeserializationListener(
|
|
|
|
getASTConsumer().GetASTDeserializationListener());
|
|
|
|
getASTContext().setASTMutationListener(
|
|
|
|
getASTConsumer().GetASTMutationListener());
|
|
|
|
}
|
|
|
|
getASTContext().setExternalSource(ModuleManager);
|
|
|
|
if (hasSema())
|
|
|
|
ModuleManager->InitializeSema(getSema());
|
|
|
|
if (hasASTConsumer())
|
|
|
|
ModuleManager->StartTranslationUnit(&getASTConsumer());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-10-28 07:01:16 +08:00
|
|
|
bool CompilerInstance::loadModuleFile(StringRef FileName) {
|
|
|
|
// Helper to recursively read the module names for all modules we're adding.
|
|
|
|
// We mark these as known and redirect any attempt to load that module to
|
|
|
|
// the files we were handed.
|
|
|
|
struct ReadModuleNames : ASTReaderListener {
|
|
|
|
CompilerInstance &CI;
|
|
|
|
std::vector<StringRef> ModuleFileStack;
|
|
|
|
bool Failed;
|
|
|
|
bool TopFileIsModule;
|
|
|
|
|
|
|
|
ReadModuleNames(CompilerInstance &CI)
|
|
|
|
: CI(CI), Failed(false), TopFileIsModule(false) {}
|
|
|
|
|
|
|
|
bool needsImportVisitation() const override { return true; }
|
|
|
|
|
|
|
|
void visitImport(StringRef FileName) override {
|
|
|
|
ModuleFileStack.push_back(FileName);
|
|
|
|
if (ASTReader::readASTFileControlBlock(FileName, CI.getFileManager(),
|
|
|
|
*this)) {
|
|
|
|
CI.getDiagnostics().Report(SourceLocation(),
|
|
|
|
diag::err_module_file_not_found)
|
|
|
|
<< FileName;
|
|
|
|
// FIXME: Produce a note stack explaining how we got here.
|
|
|
|
Failed = true;
|
|
|
|
}
|
|
|
|
ModuleFileStack.pop_back();
|
|
|
|
}
|
2014-10-22 10:05:46 +08:00
|
|
|
|
2014-10-28 07:01:16 +08:00
|
|
|
void ReadModuleName(StringRef ModuleName) override {
|
|
|
|
if (ModuleFileStack.size() == 1)
|
|
|
|
TopFileIsModule = true;
|
|
|
|
|
|
|
|
auto &ModuleFile = CI.ModuleFileOverrides[ModuleName];
|
|
|
|
if (!ModuleFile.empty() && ModuleFile != ModuleFileStack.back())
|
|
|
|
CI.getDiagnostics().Report(SourceLocation(),
|
|
|
|
diag::err_conflicting_module_files)
|
|
|
|
<< ModuleName << ModuleFile << ModuleFileStack.back();
|
|
|
|
ModuleFile = ModuleFileStack.back();
|
|
|
|
}
|
|
|
|
} RMN(*this);
|
|
|
|
|
|
|
|
RMN.visitImport(FileName);
|
2014-10-22 10:05:46 +08:00
|
|
|
|
2014-10-28 07:01:16 +08:00
|
|
|
if (RMN.Failed)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// If we never found a module name for the top file, then it's not a module,
|
|
|
|
// it's a PCH or preamble or something.
|
|
|
|
if (!RMN.TopFileIsModule) {
|
|
|
|
getDiagnostics().Report(SourceLocation(), diag::err_module_file_not_module)
|
2014-10-22 10:05:46 +08:00
|
|
|
<< FileName;
|
2014-10-28 07:01:16 +08:00
|
|
|
return false;
|
2014-10-22 10:05:46 +08:00
|
|
|
}
|
2014-10-28 07:01:16 +08:00
|
|
|
|
|
|
|
return true;
|
2014-10-22 10:05:46 +08:00
|
|
|
}
|
|
|
|
|
2012-11-30 07:55:25 +08:00
|
|
|
ModuleLoadResult
|
|
|
|
CompilerInstance::loadModule(SourceLocation ImportLoc,
|
|
|
|
ModuleIdPath Path,
|
|
|
|
Module::NameVisibilityKind Visibility,
|
|
|
|
bool IsInclusionDirective) {
|
2013-10-19 06:48:20 +08:00
|
|
|
// Determine what file we're searching from.
|
|
|
|
StringRef ModuleName = Path[0].first->getName();
|
|
|
|
SourceLocation ModuleNameLoc = Path[0].second;
|
|
|
|
|
2011-11-30 12:26:53 +08:00
|
|
|
// If we've already handled this import, just return the cached result.
|
|
|
|
// This one-element cache is important to eliminate redundant diagnostics
|
|
|
|
// when both the preprocessor and parser see the same import declaration.
|
2011-12-02 01:11:21 +08:00
|
|
|
if (!ImportLoc.isInvalid() && LastModuleImportLoc == ImportLoc) {
|
|
|
|
// Make the named module visible.
|
2014-07-23 23:30:23 +08:00
|
|
|
if (LastModuleImportResult && ModuleName != getLangOpts().CurrentModule &&
|
|
|
|
ModuleName != getLangOpts().ImplementationOfModule)
|
2013-02-02 00:36:12 +08:00
|
|
|
ModuleManager->makeModuleVisible(LastModuleImportResult, Visibility,
|
2013-03-21 05:10:35 +08:00
|
|
|
ImportLoc, /*Complain=*/false);
|
2011-12-01 01:33:56 +08:00
|
|
|
return LastModuleImportResult;
|
2011-12-02 01:11:21 +08:00
|
|
|
}
|
2011-10-08 19:31:46 +08:00
|
|
|
|
2014-05-22 12:46:25 +08:00
|
|
|
clang::Module *Module = nullptr;
|
2013-10-19 06:48:20 +08:00
|
|
|
|
2011-11-30 12:03:44 +08:00
|
|
|
// If we don't already have information on this module, load the module now.
|
2011-12-01 07:21:26 +08:00
|
|
|
llvm::DenseMap<const IdentifierInfo *, clang::Module *>::iterator Known
|
2011-12-01 01:33:56 +08:00
|
|
|
= KnownModules.find(Path[0].first);
|
2011-12-09 01:01:29 +08:00
|
|
|
if (Known != KnownModules.end()) {
|
|
|
|
// Retrieve the cached top-level module.
|
|
|
|
Module = Known->second;
|
2014-07-23 23:30:23 +08:00
|
|
|
} else if (ModuleName == getLangOpts().CurrentModule ||
|
|
|
|
ModuleName == getLangOpts().ImplementationOfModule) {
|
2011-12-09 01:01:29 +08:00
|
|
|
// This is the module we're building.
|
2014-05-05 13:31:33 +08:00
|
|
|
Module = PP->getHeaderSearchInfo().lookupModule(ModuleName);
|
2011-12-09 01:01:29 +08:00
|
|
|
Known = KnownModules.insert(std::make_pair(Path[0].first, Module)).first;
|
|
|
|
} else {
|
2011-11-30 12:03:44 +08:00
|
|
|
// Search for a module with the given name.
|
2012-01-30 01:08:11 +08:00
|
|
|
Module = PP->getHeaderSearchInfo().lookupModule(ModuleName);
|
2014-01-23 07:19:39 +08:00
|
|
|
if (!Module) {
|
|
|
|
getDiagnostics().Report(ModuleNameLoc, diag::err_module_not_found)
|
|
|
|
<< ModuleName
|
|
|
|
<< SourceRange(ImportLoc, ModuleNameLoc);
|
|
|
|
ModuleBuildFailed = true;
|
|
|
|
return ModuleLoadResult();
|
|
|
|
}
|
|
|
|
|
2014-10-28 07:01:16 +08:00
|
|
|
auto Override = ModuleFileOverrides.find(ModuleName);
|
|
|
|
bool Explicit = Override != ModuleFileOverrides.end();
|
|
|
|
|
2014-04-29 04:58:58 +08:00
|
|
|
std::string ModuleFileName =
|
2014-10-28 07:01:16 +08:00
|
|
|
Explicit ? Override->second
|
|
|
|
: PP->getHeaderSearchInfo().getModuleFileName(Module);
|
2012-11-30 07:55:25 +08:00
|
|
|
|
2011-11-30 12:03:44 +08:00
|
|
|
// If we don't already have an ASTReader, create one now.
|
2014-04-23 20:57:01 +08:00
|
|
|
if (!ModuleManager)
|
|
|
|
createModuleManager();
|
2011-10-08 19:31:46 +08:00
|
|
|
|
2014-03-07 14:40:32 +08:00
|
|
|
if (TheDependencyFileGenerator)
|
|
|
|
TheDependencyFileGenerator->AttachToASTReader(*ModuleManager);
|
|
|
|
|
2014-06-20 03:36:03 +08:00
|
|
|
if (ModuleDepCollector)
|
|
|
|
ModuleDepCollector->attachToASTReader(*ModuleManager);
|
|
|
|
|
2014-07-01 04:04:14 +08:00
|
|
|
for (auto &Listener : DependencyCollectors)
|
|
|
|
Listener->attachToASTReader(*ModuleManager);
|
|
|
|
|
2013-03-19 08:28:20 +08:00
|
|
|
// Try to load the module file.
|
2014-10-28 07:01:16 +08:00
|
|
|
unsigned ARRFlags =
|
|
|
|
Explicit ? 0 : ASTReader::ARR_OutOfDate | ASTReader::ARR_Missing;
|
2014-10-22 10:05:46 +08:00
|
|
|
switch (ModuleManager->ReadAST(ModuleFileName,
|
2014-10-28 07:01:16 +08:00
|
|
|
Explicit ? serialization::MK_ExplicitModule
|
|
|
|
: serialization::MK_ImplicitModule,
|
|
|
|
ImportLoc, ARRFlags)) {
|
2011-11-30 12:03:44 +08:00
|
|
|
case ASTReader::Success:
|
|
|
|
break;
|
2011-08-27 07:56:07 +08:00
|
|
|
|
2013-09-17 08:51:29 +08:00
|
|
|
case ASTReader::OutOfDate:
|
2013-03-19 08:28:20 +08:00
|
|
|
case ASTReader::Missing: {
|
2014-10-28 07:01:16 +08:00
|
|
|
if (Explicit) {
|
|
|
|
// ReadAST has already complained for us.
|
|
|
|
ModuleLoader::HadFatalFailure = true;
|
|
|
|
KnownModules[Path[0].first] = nullptr;
|
|
|
|
return ModuleLoadResult();
|
|
|
|
}
|
|
|
|
|
2013-09-17 08:51:29 +08:00
|
|
|
// The module file is missing or out-of-date. Build it.
|
2014-01-23 07:19:39 +08:00
|
|
|
assert(Module && "missing module file");
|
2013-03-19 08:28:20 +08:00
|
|
|
// Check whether there is a cycle in the module graph.
|
|
|
|
ModuleBuildStack ModPath = getSourceManager().getModuleBuildStack();
|
|
|
|
ModuleBuildStack::iterator Pos = ModPath.begin(), PosEnd = ModPath.end();
|
|
|
|
for (; Pos != PosEnd; ++Pos) {
|
|
|
|
if (Pos->first == ModuleName)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Pos != PosEnd) {
|
|
|
|
SmallString<256> CyclePath;
|
|
|
|
for (; Pos != PosEnd; ++Pos) {
|
|
|
|
CyclePath += Pos->first;
|
|
|
|
CyclePath += " -> ";
|
|
|
|
}
|
|
|
|
CyclePath += ModuleName;
|
|
|
|
|
|
|
|
getDiagnostics().Report(ModuleNameLoc, diag::err_module_cycle)
|
|
|
|
<< ModuleName << CyclePath;
|
|
|
|
return ModuleLoadResult();
|
|
|
|
}
|
2012-11-30 07:55:25 +08:00
|
|
|
|
|
|
|
// Check whether we have already attempted to build this module (but
|
|
|
|
// failed).
|
|
|
|
if (getPreprocessorOpts().FailedModules &&
|
|
|
|
getPreprocessorOpts().FailedModules->hasAlreadyFailed(ModuleName)) {
|
|
|
|
getDiagnostics().Report(ModuleNameLoc, diag::err_module_not_built)
|
|
|
|
<< ModuleName
|
|
|
|
<< SourceRange(ImportLoc, ModuleNameLoc);
|
2013-01-25 08:45:27 +08:00
|
|
|
ModuleBuildFailed = true;
|
2012-11-30 07:55:25 +08:00
|
|
|
return ModuleLoadResult();
|
|
|
|
}
|
|
|
|
|
2014-06-18 06:35:27 +08:00
|
|
|
// Try to compile and then load the module.
|
|
|
|
if (!compileAndLoadModule(*this, ImportLoc, ModuleNameLoc, Module,
|
|
|
|
ModuleFileName)) {
|
2014-09-27 06:42:23 +08:00
|
|
|
assert(getDiagnostics().hasErrorOccurred() &&
|
|
|
|
"undiagnosed error in compileAndLoadModule");
|
2013-01-10 10:04:18 +08:00
|
|
|
if (getPreprocessorOpts().FailedModules)
|
|
|
|
getPreprocessorOpts().FailedModules->addFailed(ModuleName);
|
2014-05-22 12:46:25 +08:00
|
|
|
KnownModules[Path[0].first] = nullptr;
|
2013-01-25 08:45:27 +08:00
|
|
|
ModuleBuildFailed = true;
|
2012-11-30 07:55:25 +08:00
|
|
|
return ModuleLoadResult();
|
2012-11-08 01:46:15 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Okay, we've rebuilt and now loaded the module.
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2012-10-23 06:50:17 +08:00
|
|
|
case ASTReader::VersionMismatch:
|
|
|
|
case ASTReader::ConfigurationMismatch:
|
|
|
|
case ASTReader::HadErrors:
|
2013-05-24 13:44:08 +08:00
|
|
|
ModuleLoader::HadFatalFailure = true;
|
2011-11-30 12:03:44 +08:00
|
|
|
// FIXME: The ASTReader will already have complained, but can we showhorn
|
|
|
|
// that diagnostic information into a more useful form?
|
2014-05-22 12:46:25 +08:00
|
|
|
KnownModules[Path[0].first] = nullptr;
|
2012-11-30 07:55:25 +08:00
|
|
|
return ModuleLoadResult();
|
2011-10-08 19:31:46 +08:00
|
|
|
|
2011-11-30 12:03:44 +08:00
|
|
|
case ASTReader::Failure:
|
2013-05-24 13:44:08 +08:00
|
|
|
ModuleLoader::HadFatalFailure = true;
|
2011-12-01 01:33:56 +08:00
|
|
|
// Already complained, but note now that we failed.
|
2014-05-22 12:46:25 +08:00
|
|
|
KnownModules[Path[0].first] = nullptr;
|
2013-01-25 08:45:27 +08:00
|
|
|
ModuleBuildFailed = true;
|
2012-11-30 07:55:25 +08:00
|
|
|
return ModuleLoadResult();
|
2011-11-30 12:03:44 +08:00
|
|
|
}
|
2012-09-29 09:06:04 +08:00
|
|
|
|
2011-12-01 01:33:56 +08:00
|
|
|
// Cache the result of this top-level module lookup for later.
|
|
|
|
Known = KnownModules.insert(std::make_pair(Path[0].first, Module)).first;
|
2011-08-27 07:56:07 +08:00
|
|
|
}
|
2011-11-30 12:03:44 +08:00
|
|
|
|
2011-12-01 01:33:56 +08:00
|
|
|
// If we never found the module, fail.
|
|
|
|
if (!Module)
|
2012-11-30 07:55:25 +08:00
|
|
|
return ModuleLoadResult();
|
2011-12-01 01:33:56 +08:00
|
|
|
|
2011-11-30 12:03:44 +08:00
|
|
|
// Verify that the rest of the module path actually corresponds to
|
|
|
|
// a submodule.
|
2011-12-01 01:33:56 +08:00
|
|
|
if (Path.size() > 1) {
|
2011-11-30 12:03:44 +08:00
|
|
|
for (unsigned I = 1, N = Path.size(); I != N; ++I) {
|
|
|
|
StringRef Name = Path[I].first->getName();
|
2012-01-05 07:32:19 +08:00
|
|
|
clang::Module *Sub = Module->findSubmodule(Name);
|
2011-11-30 12:03:44 +08:00
|
|
|
|
2012-01-05 07:32:19 +08:00
|
|
|
if (!Sub) {
|
2011-11-30 12:03:44 +08:00
|
|
|
// Attempt to perform typo correction to find a module name that works.
|
2013-01-13 03:30:44 +08:00
|
|
|
SmallVector<StringRef, 2> Best;
|
2011-11-30 12:03:44 +08:00
|
|
|
unsigned BestEditDistance = (std::numeric_limits<unsigned>::max)();
|
|
|
|
|
2012-01-05 07:32:19 +08:00
|
|
|
for (clang::Module::submodule_iterator J = Module->submodule_begin(),
|
|
|
|
JEnd = Module->submodule_end();
|
2011-12-01 03:41:21 +08:00
|
|
|
J != JEnd; ++J) {
|
2012-01-05 07:32:19 +08:00
|
|
|
unsigned ED = Name.edit_distance((*J)->Name,
|
2011-11-30 12:03:44 +08:00
|
|
|
/*AllowReplacements=*/true,
|
|
|
|
BestEditDistance);
|
|
|
|
if (ED <= BestEditDistance) {
|
2012-01-05 07:32:19 +08:00
|
|
|
if (ED < BestEditDistance) {
|
2011-11-30 12:03:44 +08:00
|
|
|
Best.clear();
|
2012-01-05 07:32:19 +08:00
|
|
|
BestEditDistance = ED;
|
|
|
|
}
|
|
|
|
|
|
|
|
Best.push_back((*J)->Name);
|
2011-11-30 12:03:44 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// If there was a clear winner, user it.
|
|
|
|
if (Best.size() == 1) {
|
|
|
|
getDiagnostics().Report(Path[I].second,
|
|
|
|
diag::err_no_submodule_suggest)
|
2011-12-01 01:33:56 +08:00
|
|
|
<< Path[I].first << Module->getFullModuleName() << Best[0]
|
2011-11-30 12:03:44 +08:00
|
|
|
<< SourceRange(Path[0].second, Path[I-1].second)
|
|
|
|
<< FixItHint::CreateReplacement(SourceRange(Path[I].second),
|
|
|
|
Best[0]);
|
2012-01-05 07:32:19 +08:00
|
|
|
|
|
|
|
Sub = Module->findSubmodule(Best[0]);
|
2011-11-30 12:03:44 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-01-05 07:32:19 +08:00
|
|
|
if (!Sub) {
|
2011-11-30 12:03:44 +08:00
|
|
|
// No submodule by this name. Complain, and don't look for further
|
|
|
|
// submodules.
|
|
|
|
getDiagnostics().Report(Path[I].second, diag::err_no_submodule)
|
2011-12-01 01:33:56 +08:00
|
|
|
<< Path[I].first << Module->getFullModuleName()
|
2011-11-30 12:03:44 +08:00
|
|
|
<< SourceRange(Path[0].second, Path[I-1].second);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2012-01-05 07:32:19 +08:00
|
|
|
Module = Sub;
|
2011-11-30 12:03:44 +08:00
|
|
|
}
|
|
|
|
}
|
2014-07-23 23:30:23 +08:00
|
|
|
|
|
|
|
// Don't make the module visible if we are in the implementation.
|
|
|
|
if (ModuleName == getLangOpts().ImplementationOfModule)
|
|
|
|
return ModuleLoadResult(Module, false);
|
2011-11-30 12:03:44 +08:00
|
|
|
|
2011-12-09 01:01:29 +08:00
|
|
|
// Make the named module visible, if it's not already part of the module
|
|
|
|
// we are parsing.
|
2011-12-20 08:28:52 +08:00
|
|
|
if (ModuleName != getLangOpts().CurrentModule) {
|
|
|
|
if (!Module->IsFromModuleFile) {
|
|
|
|
// We have an umbrella header or directory that doesn't actually include
|
|
|
|
// all of the headers within the directory it covers. Complain about
|
|
|
|
// this missing submodule and recover by forgetting that we ever saw
|
|
|
|
// this submodule.
|
|
|
|
// FIXME: Should we detect this at module load time? It seems fairly
|
|
|
|
// expensive (and rare).
|
|
|
|
getDiagnostics().Report(ImportLoc, diag::warn_missing_submodule)
|
|
|
|
<< Module->getFullModuleName()
|
|
|
|
<< SourceRange(Path.front().second, Path.back().second);
|
2014-05-22 12:46:25 +08:00
|
|
|
|
|
|
|
return ModuleLoadResult(nullptr, true);
|
2011-12-20 08:28:52 +08:00
|
|
|
}
|
2011-12-31 12:05:44 +08:00
|
|
|
|
|
|
|
// Check whether this module is available.
|
2013-10-29 06:18:19 +08:00
|
|
|
clang::Module::Requirement Requirement;
|
2013-12-17 18:31:37 +08:00
|
|
|
clang::Module::HeaderDirective MissingHeader;
|
|
|
|
if (!Module->isAvailable(getLangOpts(), getTarget(), Requirement,
|
|
|
|
MissingHeader)) {
|
|
|
|
if (MissingHeader.FileNameLoc.isValid()) {
|
|
|
|
getDiagnostics().Report(MissingHeader.FileNameLoc,
|
|
|
|
diag::err_module_header_missing)
|
|
|
|
<< MissingHeader.IsUmbrella << MissingHeader.FileName;
|
|
|
|
} else {
|
|
|
|
getDiagnostics().Report(ImportLoc, diag::err_module_unavailable)
|
|
|
|
<< Module->getFullModuleName()
|
|
|
|
<< Requirement.second << Requirement.first
|
|
|
|
<< SourceRange(Path.front().second, Path.back().second);
|
|
|
|
}
|
2011-12-31 12:05:44 +08:00
|
|
|
LastModuleImportLoc = ImportLoc;
|
2012-11-30 07:55:25 +08:00
|
|
|
LastModuleImportResult = ModuleLoadResult();
|
|
|
|
return ModuleLoadResult();
|
2011-12-31 12:05:44 +08:00
|
|
|
}
|
|
|
|
|
2013-03-21 05:10:35 +08:00
|
|
|
ModuleManager->makeModuleVisible(Module, Visibility, ImportLoc,
|
|
|
|
/*Complain=*/true);
|
2011-12-20 08:28:52 +08:00
|
|
|
}
|
2013-03-20 08:22:05 +08:00
|
|
|
|
|
|
|
// Check for any configuration macros that have changed.
|
|
|
|
clang::Module *TopModule = Module->getTopLevelModule();
|
|
|
|
for (unsigned I = 0, N = TopModule->ConfigMacros.size(); I != N; ++I) {
|
|
|
|
checkConfigMacro(getPreprocessor(), TopModule->ConfigMacros[I],
|
|
|
|
Module, ImportLoc);
|
|
|
|
}
|
|
|
|
|
2011-12-03 07:42:12 +08:00
|
|
|
// If this module import was due to an inclusion directive, create an
|
|
|
|
// implicit import declaration to capture it in the AST.
|
|
|
|
if (IsInclusionDirective && hasASTContext()) {
|
|
|
|
TranslationUnitDecl *TU = getASTContext().getTranslationUnitDecl();
|
2012-10-03 09:58:37 +08:00
|
|
|
ImportDecl *ImportD = ImportDecl::CreateImplicit(getASTContext(), TU,
|
|
|
|
ImportLoc, Module,
|
|
|
|
Path.back().second);
|
|
|
|
TU->addDecl(ImportD);
|
|
|
|
if (Consumer)
|
|
|
|
Consumer->HandleImplicitImportDecl(ImportD);
|
2011-12-03 07:42:12 +08:00
|
|
|
}
|
2011-11-30 12:03:44 +08:00
|
|
|
|
2011-11-30 12:26:53 +08:00
|
|
|
LastModuleImportLoc = ImportLoc;
|
2012-11-30 07:55:25 +08:00
|
|
|
LastModuleImportResult = ModuleLoadResult(Module, false);
|
|
|
|
return LastModuleImportResult;
|
2011-08-27 07:56:07 +08:00
|
|
|
}
|
2013-01-12 09:29:50 +08:00
|
|
|
|
|
|
|
void CompilerInstance::makeModuleVisible(Module *Mod,
|
2013-02-02 00:36:12 +08:00
|
|
|
Module::NameVisibilityKind Visibility,
|
2013-03-21 05:10:35 +08:00
|
|
|
SourceLocation ImportLoc,
|
|
|
|
bool Complain){
|
|
|
|
ModuleManager->makeModuleVisible(Mod, Visibility, ImportLoc, Complain);
|
2013-01-12 09:29:50 +08:00
|
|
|
}
|
|
|
|
|
2014-04-23 20:57:01 +08:00
|
|
|
GlobalModuleIndex *CompilerInstance::loadGlobalModuleIndex(
|
|
|
|
SourceLocation TriggerLoc) {
|
|
|
|
if (!ModuleManager)
|
|
|
|
createModuleManager();
|
|
|
|
// Can't do anything if we don't have the module manager.
|
|
|
|
if (!ModuleManager)
|
2014-05-22 12:46:25 +08:00
|
|
|
return nullptr;
|
2014-04-23 20:57:01 +08:00
|
|
|
// Get an existing global index. This loads it if not already
|
|
|
|
// loaded.
|
|
|
|
ModuleManager->loadGlobalIndex();
|
|
|
|
GlobalModuleIndex *GlobalIndex = ModuleManager->getGlobalIndex();
|
|
|
|
// If the global index doesn't exist, create it.
|
|
|
|
if (!GlobalIndex && shouldBuildGlobalModuleIndex() && hasFileManager() &&
|
|
|
|
hasPreprocessor()) {
|
|
|
|
llvm::sys::fs::create_directories(
|
|
|
|
getPreprocessor().getHeaderSearchInfo().getModuleCachePath());
|
|
|
|
GlobalModuleIndex::writeIndex(
|
|
|
|
getFileManager(),
|
|
|
|
getPreprocessor().getHeaderSearchInfo().getModuleCachePath());
|
|
|
|
ModuleManager->resetForReload();
|
|
|
|
ModuleManager->loadGlobalIndex();
|
|
|
|
GlobalIndex = ModuleManager->getGlobalIndex();
|
|
|
|
}
|
|
|
|
// For finding modules needing to be imported for fixit messages,
|
|
|
|
// we need to make the global index cover all modules, so we do that here.
|
|
|
|
if (!HaveFullGlobalModuleIndex && GlobalIndex && !buildingModule()) {
|
|
|
|
ModuleMap &MMap = getPreprocessor().getHeaderSearchInfo().getModuleMap();
|
|
|
|
bool RecreateIndex = false;
|
|
|
|
for (ModuleMap::module_iterator I = MMap.module_begin(),
|
|
|
|
E = MMap.module_end(); I != E; ++I) {
|
|
|
|
Module *TheModule = I->second;
|
|
|
|
const FileEntry *Entry = TheModule->getASTFile();
|
|
|
|
if (!Entry) {
|
|
|
|
SmallVector<std::pair<IdentifierInfo *, SourceLocation>, 2> Path;
|
|
|
|
Path.push_back(std::make_pair(
|
|
|
|
getPreprocessor().getIdentifierInfo(TheModule->Name), TriggerLoc));
|
|
|
|
std::reverse(Path.begin(), Path.end());
|
|
|
|
// Load a module as hidden. This also adds it to the global index.
|
2014-04-23 21:45:44 +08:00
|
|
|
loadModule(TheModule->DefinitionLoc, Path,
|
2014-04-23 20:57:01 +08:00
|
|
|
Module::Hidden, false);
|
|
|
|
RecreateIndex = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (RecreateIndex) {
|
|
|
|
GlobalModuleIndex::writeIndex(
|
|
|
|
getFileManager(),
|
|
|
|
getPreprocessor().getHeaderSearchInfo().getModuleCachePath());
|
|
|
|
ModuleManager->resetForReload();
|
|
|
|
ModuleManager->loadGlobalIndex();
|
|
|
|
GlobalIndex = ModuleManager->getGlobalIndex();
|
|
|
|
}
|
|
|
|
HaveFullGlobalModuleIndex = true;
|
|
|
|
}
|
|
|
|
return GlobalIndex;
|
|
|
|
}
|
2014-04-24 03:04:32 +08:00
|
|
|
|
|
|
|
// Check global module index for missing imports.
|
|
|
|
bool
|
|
|
|
CompilerInstance::lookupMissingImports(StringRef Name,
|
|
|
|
SourceLocation TriggerLoc) {
|
|
|
|
// Look for the symbol in non-imported modules, but only if an error
|
|
|
|
// actually occurred.
|
|
|
|
if (!buildingModule()) {
|
|
|
|
// Load global module index, or retrieve a previously loaded one.
|
|
|
|
GlobalModuleIndex *GlobalIndex = loadGlobalModuleIndex(
|
|
|
|
TriggerLoc);
|
|
|
|
|
|
|
|
// Only if we have a global index.
|
|
|
|
if (GlobalIndex) {
|
|
|
|
GlobalModuleIndex::HitSet FoundModules;
|
|
|
|
|
|
|
|
// Find the modules that reference the identifier.
|
|
|
|
// Note that this only finds top-level modules.
|
|
|
|
// We'll let diagnoseTypo find the actual declaration module.
|
|
|
|
if (GlobalIndex->lookupIdentifier(Name, FoundModules))
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
Provide a BuryPointer for unique_ptrs.
In theory, it'd be nice if we could move to a case where all buried
pointers were buried via unique_ptr to demonstrate that the program had
finished with the value (that we could really have cleanly deallocated
it) but instead chose to bury it.
I think the main reason that's not possible right now is the various
IntrusiveRefCntPtrs in the Frontend, sharing ownership for a variety of
compiler bits (see the various similar
"CompilerInstance::releaseAndLeak*" functions). I have yet to figure out
their correct ownership semantics - but perhaps, even if the
intrusiveness can be removed, the shared ownership may yet remain and
that would lead to a non-unique burying as is there today. (though we
could model that a little better - by passing in a shared_ptr, etc -
rather than needing the two step that's currently used in those other
releaseAndLeak* functions)
This might be a bit more robust if BuryPointer took the boolean:
BuryPointer(bool, unique_ptr<T>)
and the choice to bury was made internally - that way, even when
DisableFree was not set, the unique_ptr would still be null in the
caller and there'd be no chance of accidentally having a different
codepath where the value is used after burial in !DisableFree, but it
becomes null only in DisableFree, etc...
llvm-svn: 216742
2014-08-30 00:53:14 +08:00
|
|
|
void CompilerInstance::resetAndLeakSema() { BuryPointer(takeSema()); }
|