|  | //===--- 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" | 
|  | #include "clang/Sema/Sema.h" | 
|  | #include "clang/AST/ASTConsumer.h" | 
|  | #include "clang/AST/ASTContext.h" | 
|  | #include "clang/Basic/Diagnostic.h" | 
|  | #include "clang/Basic/FileManager.h" | 
|  | #include "clang/Basic/SourceManager.h" | 
|  | #include "clang/Basic/TargetInfo.h" | 
|  | #include "clang/Basic/Version.h" | 
|  | #include "clang/Lex/HeaderSearch.h" | 
|  | #include "clang/Lex/Preprocessor.h" | 
|  | #include "clang/Lex/PTHManager.h" | 
|  | #include "clang/Frontend/ChainedDiagnosticConsumer.h" | 
|  | #include "clang/Frontend/FrontendAction.h" | 
|  | #include "clang/Frontend/FrontendActions.h" | 
|  | #include "clang/Frontend/FrontendDiagnostic.h" | 
|  | #include "clang/Frontend/LogDiagnosticPrinter.h" | 
|  | #include "clang/Frontend/SerializedDiagnosticPrinter.h" | 
|  | #include "clang/Frontend/TextDiagnosticPrinter.h" | 
|  | #include "clang/Frontend/VerifyDiagnosticConsumer.h" | 
|  | #include "clang/Frontend/Utils.h" | 
|  | #include "clang/Serialization/ASTReader.h" | 
|  | #include "clang/Sema/CodeCompleteConsumer.h" | 
|  | #include "llvm/Support/FileSystem.h" | 
|  | #include "llvm/Support/MemoryBuffer.h" | 
|  | #include "llvm/Support/raw_ostream.h" | 
|  | #include "llvm/ADT/Statistic.h" | 
|  | #include "llvm/Support/Timer.h" | 
|  | #include "llvm/Support/Host.h" | 
|  | #include "llvm/Support/Path.h" | 
|  | #include "llvm/Support/Program.h" | 
|  | #include "llvm/Support/Signals.h" | 
|  | #include "llvm/Support/system_error.h" | 
|  | #include "llvm/Support/CrashRecoveryContext.h" | 
|  | #include "llvm/Config/config.h" | 
|  |  | 
|  | // Support for FileLockManager | 
|  | #include <fstream> | 
|  | #include <sys/types.h> | 
|  | #include <sys/stat.h> | 
|  |  | 
|  | #if LLVM_ON_WIN32 | 
|  | #include <windows.h> | 
|  | #endif | 
|  | #if LLVM_ON_UNIX | 
|  | #include <unistd.h> | 
|  | #endif | 
|  |  | 
|  | using namespace clang; | 
|  |  | 
|  | CompilerInstance::CompilerInstance() | 
|  | : Invocation(new CompilerInvocation()), ModuleManager(0) { | 
|  | } | 
|  |  | 
|  | CompilerInstance::~CompilerInstance() { | 
|  | } | 
|  |  | 
|  | void CompilerInstance::setInvocation(CompilerInvocation *Value) { | 
|  | Invocation = Value; | 
|  | } | 
|  |  | 
|  | void CompilerInstance::setDiagnostics(DiagnosticsEngine *Value) { | 
|  | Diagnostics = Value; | 
|  | } | 
|  |  | 
|  | void CompilerInstance::setTarget(TargetInfo *Value) { | 
|  | Target = Value; | 
|  | } | 
|  |  | 
|  | void CompilerInstance::setFileManager(FileManager *Value) { | 
|  | FileMgr = Value; | 
|  | } | 
|  |  | 
|  | void CompilerInstance::setSourceManager(SourceManager *Value) { | 
|  | SourceMgr = Value; | 
|  | } | 
|  |  | 
|  | void CompilerInstance::setPreprocessor(Preprocessor *Value) { PP = Value; } | 
|  |  | 
|  | void CompilerInstance::setASTContext(ASTContext *Value) { Context = Value; } | 
|  |  | 
|  | void CompilerInstance::setSema(Sema *S) { | 
|  | TheSema.reset(S); | 
|  | } | 
|  |  | 
|  | void CompilerInstance::setASTConsumer(ASTConsumer *Value) { | 
|  | Consumer.reset(Value); | 
|  | } | 
|  |  | 
|  | void CompilerInstance::setCodeCompletionConsumer(CodeCompleteConsumer *Value) { | 
|  | CompletionConsumer.reset(Value); | 
|  | } | 
|  |  | 
|  | // Diagnostics | 
|  | static void SetUpBuildDumpLog(const DiagnosticOptions &DiagOpts, | 
|  | unsigned argc, const char* const *argv, | 
|  | DiagnosticsEngine &Diags) { | 
|  | std::string ErrorInfo; | 
|  | llvm::OwningPtr<raw_ostream> OS( | 
|  | new llvm::raw_fd_ostream(DiagOpts.DumpBuildInformation.c_str(), ErrorInfo)); | 
|  | if (!ErrorInfo.empty()) { | 
|  | Diags.Report(diag::err_fe_unable_to_open_logfile) | 
|  | << DiagOpts.DumpBuildInformation << ErrorInfo; | 
|  | return; | 
|  | } | 
|  |  | 
|  | (*OS) << "clang -cc1 command line arguments: "; | 
|  | for (unsigned i = 0; i != argc; ++i) | 
|  | (*OS) << argv[i] << ' '; | 
|  | (*OS) << '\n'; | 
|  |  | 
|  | // Chain in a diagnostic client which will log the diagnostics. | 
|  | DiagnosticConsumer *Logger = | 
|  | new TextDiagnosticPrinter(*OS.take(), DiagOpts, /*OwnsOutputStream=*/true); | 
|  | Diags.setClient(new ChainedDiagnosticConsumer(Diags.takeClient(), Logger)); | 
|  | } | 
|  |  | 
|  | static void SetUpDiagnosticLog(const DiagnosticOptions &DiagOpts, | 
|  | const CodeGenOptions *CodeGenOpts, | 
|  | DiagnosticsEngine &Diags) { | 
|  | std::string ErrorInfo; | 
|  | bool OwnsStream = false; | 
|  | raw_ostream *OS = &llvm::errs(); | 
|  | if (DiagOpts.DiagnosticLogFile != "-") { | 
|  | // Create the output stream. | 
|  | llvm::raw_fd_ostream *FileOS( | 
|  | new llvm::raw_fd_ostream(DiagOpts.DiagnosticLogFile.c_str(), | 
|  | ErrorInfo, llvm::raw_fd_ostream::F_Append)); | 
|  | if (!ErrorInfo.empty()) { | 
|  | Diags.Report(diag::warn_fe_cc_log_diagnostics_failure) | 
|  | << DiagOpts.DumpBuildInformation << ErrorInfo; | 
|  | } else { | 
|  | FileOS->SetUnbuffered(); | 
|  | FileOS->SetUseAtomicWrites(true); | 
|  | OS = FileOS; | 
|  | OwnsStream = true; | 
|  | } | 
|  | } | 
|  |  | 
|  | // Chain in the diagnostic client which will log the diagnostics. | 
|  | LogDiagnosticPrinter *Logger = new LogDiagnosticPrinter(*OS, DiagOpts, | 
|  | OwnsStream); | 
|  | if (CodeGenOpts) | 
|  | Logger->setDwarfDebugFlags(CodeGenOpts->DwarfDebugFlags); | 
|  | Diags.setClient(new ChainedDiagnosticConsumer(Diags.takeClient(), Logger)); | 
|  | } | 
|  |  | 
|  | static void SetupSerializedDiagnostics(const DiagnosticOptions &DiagOpts, | 
|  | DiagnosticsEngine &Diags, | 
|  | StringRef OutputFile) { | 
|  | std::string ErrorInfo; | 
|  | llvm::OwningPtr<llvm::raw_fd_ostream> OS; | 
|  | OS.reset(new llvm::raw_fd_ostream(OutputFile.str().c_str(), ErrorInfo, | 
|  | llvm::raw_fd_ostream::F_Binary)); | 
|  |  | 
|  | if (!ErrorInfo.empty()) { | 
|  | Diags.Report(diag::warn_fe_serialized_diag_failure) | 
|  | << OutputFile << ErrorInfo; | 
|  | return; | 
|  | } | 
|  |  | 
|  | DiagnosticConsumer *SerializedConsumer = | 
|  | clang::serialized_diags::create(OS.take(), Diags); | 
|  |  | 
|  |  | 
|  | Diags.setClient(new ChainedDiagnosticConsumer(Diags.takeClient(), | 
|  | SerializedConsumer)); | 
|  | } | 
|  |  | 
|  | void CompilerInstance::createDiagnostics(int Argc, const char* const *Argv, | 
|  | DiagnosticConsumer *Client, | 
|  | bool ShouldOwnClient, | 
|  | bool ShouldCloneClient) { | 
|  | Diagnostics = createDiagnostics(getDiagnosticOpts(), Argc, Argv, Client, | 
|  | ShouldOwnClient, ShouldCloneClient, | 
|  | &getCodeGenOpts()); | 
|  | } | 
|  |  | 
|  | llvm::IntrusiveRefCntPtr<DiagnosticsEngine> | 
|  | CompilerInstance::createDiagnostics(const DiagnosticOptions &Opts, | 
|  | int Argc, const char* const *Argv, | 
|  | DiagnosticConsumer *Client, | 
|  | bool ShouldOwnClient, | 
|  | bool ShouldCloneClient, | 
|  | const CodeGenOptions *CodeGenOpts) { | 
|  | llvm::IntrusiveRefCntPtr<DiagnosticIDs> DiagID(new DiagnosticIDs()); | 
|  | llvm::IntrusiveRefCntPtr<DiagnosticsEngine> | 
|  | Diags(new DiagnosticsEngine(DiagID)); | 
|  |  | 
|  | // Create the diagnostic client for reporting errors or for | 
|  | // implementing -verify. | 
|  | if (Client) { | 
|  | if (ShouldCloneClient) | 
|  | Diags->setClient(Client->clone(*Diags), ShouldOwnClient); | 
|  | else | 
|  | Diags->setClient(Client, ShouldOwnClient); | 
|  | } else | 
|  | Diags->setClient(new TextDiagnosticPrinter(llvm::errs(), Opts)); | 
|  |  | 
|  | // Chain in -verify checker, if requested. | 
|  | if (Opts.VerifyDiagnostics) | 
|  | Diags->setClient(new VerifyDiagnosticConsumer(*Diags)); | 
|  |  | 
|  | // Chain in -diagnostic-log-file dumper, if requested. | 
|  | if (!Opts.DiagnosticLogFile.empty()) | 
|  | SetUpDiagnosticLog(Opts, CodeGenOpts, *Diags); | 
|  |  | 
|  | if (!Opts.DumpBuildInformation.empty()) | 
|  | SetUpBuildDumpLog(Opts, Argc, Argv, *Diags); | 
|  |  | 
|  | if (!Opts.DiagnosticSerializationFile.empty()) | 
|  | SetupSerializedDiagnostics(Opts, *Diags, | 
|  | Opts.DiagnosticSerializationFile); | 
|  |  | 
|  | // Configure our handling of diagnostics. | 
|  | ProcessWarningOptions(*Diags, Opts); | 
|  |  | 
|  | return Diags; | 
|  | } | 
|  |  | 
|  | // File Manager | 
|  |  | 
|  | void CompilerInstance::createFileManager() { | 
|  | FileMgr = new FileManager(getFileSystemOpts()); | 
|  | } | 
|  |  | 
|  | // Source Manager | 
|  |  | 
|  | void CompilerInstance::createSourceManager(FileManager &FileMgr) { | 
|  | SourceMgr = new SourceManager(getDiagnostics(), FileMgr); | 
|  | } | 
|  |  | 
|  | // Preprocessor | 
|  |  | 
|  | void CompilerInstance::createPreprocessor() { | 
|  | const PreprocessorOptions &PPOpts = getPreprocessorOpts(); | 
|  |  | 
|  | // Create a PTH manager if we are using some form of a token cache. | 
|  | PTHManager *PTHMgr = 0; | 
|  | if (!PPOpts.TokenCache.empty()) | 
|  | PTHMgr = PTHManager::Create(PPOpts.TokenCache, getDiagnostics()); | 
|  |  | 
|  | // Create the Preprocessor. | 
|  | HeaderSearch *HeaderInfo = new HeaderSearch(getFileManager(), | 
|  | getDiagnostics()); | 
|  | PP = new Preprocessor(getDiagnostics(), getLangOpts(), &getTarget(), | 
|  | getSourceManager(), *HeaderInfo, *this, PTHMgr, | 
|  | /*OwnsHeaderSearch=*/true); | 
|  |  | 
|  | // 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) { | 
|  | PTHMgr->setPreprocessor(&*PP); | 
|  | PP->setPTHManager(PTHMgr); | 
|  | } | 
|  |  | 
|  | if (PPOpts.DetailedRecord) | 
|  | PP->createPreprocessingRecord( | 
|  | PPOpts.DetailedRecordIncludesNestedMacroExpansions); | 
|  |  | 
|  | InitializePreprocessor(*PP, PPOpts, getHeaderSearchOpts(), getFrontendOpts()); | 
|  |  | 
|  | // Set up the module path, including the hash for the | 
|  | // module-creation options. | 
|  | llvm::SmallString<256> SpecificModuleCache( | 
|  | getHeaderSearchOpts().ModuleCachePath); | 
|  | if (!getHeaderSearchOpts().DisableModuleHash) | 
|  | llvm::sys::path::append(SpecificModuleCache, | 
|  | getInvocation().getModuleHash()); | 
|  | PP->getHeaderSearchInfo().configureModules(SpecificModuleCache, | 
|  | getLangOpts().CurrentModule); | 
|  |  | 
|  | // Handle generating dependencies, if requested. | 
|  | const DependencyOutputOptions &DepOpts = getDependencyOutputOpts(); | 
|  | if (!DepOpts.OutputFile.empty()) | 
|  | AttachDependencyFileGen(*PP, DepOpts); | 
|  |  | 
|  | // Handle generating header include information, if requested. | 
|  | if (DepOpts.ShowHeaderIncludes) | 
|  | AttachHeaderIncludeGen(*PP); | 
|  | if (!DepOpts.HeaderIncludeOutputFile.empty()) { | 
|  | StringRef OutputPath = DepOpts.HeaderIncludeOutputFile; | 
|  | if (OutputPath == "-") | 
|  | OutputPath = ""; | 
|  | AttachHeaderIncludeGen(*PP, /*ShowAllHeaders=*/true, OutputPath, | 
|  | /*ShowDepth=*/false); | 
|  | } | 
|  | } | 
|  |  | 
|  | // ASTContext | 
|  |  | 
|  | void CompilerInstance::createASTContext() { | 
|  | Preprocessor &PP = getPreprocessor(); | 
|  | Context = new ASTContext(getLangOpts(), PP.getSourceManager(), | 
|  | &getTarget(), PP.getIdentifierTable(), | 
|  | PP.getSelectorTable(), PP.getBuiltinInfo(), | 
|  | /*size_reserve=*/ 0); | 
|  | } | 
|  |  | 
|  | // ExternalASTSource | 
|  |  | 
|  | void CompilerInstance::createPCHExternalASTSource(StringRef Path, | 
|  | bool DisablePCHValidation, | 
|  | bool DisableStatCache, | 
|  | void *DeserializationListener){ | 
|  | llvm::OwningPtr<ExternalASTSource> Source; | 
|  | bool Preamble = getPreprocessorOpts().PrecompiledPreambleBytes.first != 0; | 
|  | Source.reset(createPCHExternalASTSource(Path, getHeaderSearchOpts().Sysroot, | 
|  | DisablePCHValidation, | 
|  | DisableStatCache, | 
|  | getPreprocessor(), getASTContext(), | 
|  | DeserializationListener, | 
|  | Preamble)); | 
|  | ModuleManager = static_cast<ASTReader*>(Source.get()); | 
|  | getASTContext().setExternalSource(Source); | 
|  | } | 
|  |  | 
|  | ExternalASTSource * | 
|  | CompilerInstance::createPCHExternalASTSource(StringRef Path, | 
|  | const std::string &Sysroot, | 
|  | bool DisablePCHValidation, | 
|  | bool DisableStatCache, | 
|  | Preprocessor &PP, | 
|  | ASTContext &Context, | 
|  | void *DeserializationListener, | 
|  | bool Preamble) { | 
|  | llvm::OwningPtr<ASTReader> Reader; | 
|  | Reader.reset(new ASTReader(PP, Context, | 
|  | Sysroot.empty() ? "" : Sysroot.c_str(), | 
|  | DisablePCHValidation, DisableStatCache)); | 
|  |  | 
|  | Reader->setDeserializationListener( | 
|  | static_cast<ASTDeserializationListener *>(DeserializationListener)); | 
|  | switch (Reader->ReadAST(Path, | 
|  | Preamble ? serialization::MK_Preamble | 
|  | : serialization::MK_PCH)) { | 
|  | case ASTReader::Success: | 
|  | // Set the predefines buffer as suggested by the PCH reader. Typically, the | 
|  | // predefines buffer will be empty. | 
|  | PP.setPredefines(Reader->getSuggestedPredefines()); | 
|  | return Reader.take(); | 
|  |  | 
|  | case ASTReader::Failure: | 
|  | // Unrecoverable failure: don't even try to process the input file. | 
|  | break; | 
|  |  | 
|  | case ASTReader::IgnorePCH: | 
|  | // No suitable PCH file could be found. Return an error. | 
|  | break; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | // Code Completion | 
|  |  | 
|  | static bool EnableCodeCompletion(Preprocessor &PP, | 
|  | const std::string &Filename, | 
|  | unsigned Line, | 
|  | unsigned Column) { | 
|  | // Tell the source manager to chop off the given file at a specific | 
|  | // line and column. | 
|  | const FileEntry *Entry = PP.getFileManager().getFile(Filename); | 
|  | 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; | 
|  | } | 
|  |  | 
|  | void CompilerInstance::createCodeCompletionConsumer() { | 
|  | const ParsedSourceLocation &Loc = getFrontendOpts().CodeCompletionAt; | 
|  | if (!CompletionConsumer) { | 
|  | CompletionConsumer.reset( | 
|  | createCodeCompletionConsumer(getPreprocessor(), | 
|  | Loc.FileName, Loc.Line, Loc.Column, | 
|  | getFrontendOpts().ShowMacrosInCodeCompletion, | 
|  | getFrontendOpts().ShowCodePatternsInCodeCompletion, | 
|  | getFrontendOpts().ShowGlobalSymbolsInCodeCompletion, | 
|  | llvm::outs())); | 
|  | if (!CompletionConsumer) | 
|  | return; | 
|  | } else if (EnableCodeCompletion(getPreprocessor(), Loc.FileName, | 
|  | Loc.Line, Loc.Column)) { | 
|  | CompletionConsumer.reset(); | 
|  | return; | 
|  | } | 
|  |  | 
|  | if (CompletionConsumer->isOutputBinary() && | 
|  | llvm::sys::Program::ChangeStdoutToBinary()) { | 
|  | getPreprocessor().getDiagnostics().Report(diag::err_fe_stdout_binary); | 
|  | CompletionConsumer.reset(); | 
|  | } | 
|  | } | 
|  |  | 
|  | void CompilerInstance::createFrontendTimer() { | 
|  | FrontendTimer.reset(new llvm::Timer("Clang front-end timer")); | 
|  | } | 
|  |  | 
|  | CodeCompleteConsumer * | 
|  | CompilerInstance::createCodeCompletionConsumer(Preprocessor &PP, | 
|  | const std::string &Filename, | 
|  | unsigned Line, | 
|  | unsigned Column, | 
|  | bool ShowMacros, | 
|  | bool ShowCodePatterns, | 
|  | bool ShowGlobals, | 
|  | raw_ostream &OS) { | 
|  | if (EnableCodeCompletion(PP, Filename, Line, Column)) | 
|  | return 0; | 
|  |  | 
|  | // Set up the creation routine for code-completion. | 
|  | return new PrintingCodeCompleteConsumer(ShowMacros, ShowCodePatterns, | 
|  | ShowGlobals, OS); | 
|  | } | 
|  |  | 
|  | void CompilerInstance::createSema(TranslationUnitKind TUKind, | 
|  | CodeCompleteConsumer *CompletionConsumer) { | 
|  | TheSema.reset(new Sema(getPreprocessor(), getASTContext(), getASTConsumer(), | 
|  | TUKind, CompletionConsumer)); | 
|  | } | 
|  |  | 
|  | // Output Files | 
|  |  | 
|  | void CompilerInstance::addOutputFile(const OutputFile &OutFile) { | 
|  | assert(OutFile.OS && "Attempt to add empty stream to output list!"); | 
|  | OutputFiles.push_back(OutFile); | 
|  | } | 
|  |  | 
|  | void CompilerInstance::clearOutputFiles(bool EraseFiles) { | 
|  | for (std::list<OutputFile>::iterator | 
|  | it = OutputFiles.begin(), ie = OutputFiles.end(); it != ie; ++it) { | 
|  | delete it->OS; | 
|  | if (!it->TempFilename.empty()) { | 
|  | if (EraseFiles) { | 
|  | bool existed; | 
|  | llvm::sys::fs::remove(it->TempFilename, existed); | 
|  | } else { | 
|  | llvm::SmallString<128> NewOutFile(it->Filename); | 
|  |  | 
|  | // If '-working-directory' was passed, the output filename should be | 
|  | // relative to that. | 
|  | FileMgr->FixupRelativePath(NewOutFile); | 
|  | if (llvm::error_code ec = llvm::sys::fs::rename(it->TempFilename, | 
|  | NewOutFile.str())) { | 
|  | getDiagnostics().Report(diag::err_fe_unable_to_rename_temp) | 
|  | << it->TempFilename << it->Filename << ec.message(); | 
|  |  | 
|  | bool existed; | 
|  | llvm::sys::fs::remove(it->TempFilename, existed); | 
|  | } | 
|  | } | 
|  | } else if (!it->Filename.empty() && EraseFiles) | 
|  | llvm::sys::Path(it->Filename).eraseFromDisk(); | 
|  |  | 
|  | } | 
|  | OutputFiles.clear(); | 
|  | } | 
|  |  | 
|  | llvm::raw_fd_ostream * | 
|  | CompilerInstance::createDefaultOutputFile(bool Binary, | 
|  | StringRef InFile, | 
|  | StringRef Extension) { | 
|  | return createOutputFile(getFrontendOpts().OutputFile, Binary, | 
|  | /*RemoveFileOnSignal=*/true, InFile, Extension); | 
|  | } | 
|  |  | 
|  | llvm::raw_fd_ostream * | 
|  | CompilerInstance::createOutputFile(StringRef OutputPath, | 
|  | bool Binary, bool RemoveFileOnSignal, | 
|  | StringRef InFile, | 
|  | StringRef Extension, | 
|  | bool UseTemporary) { | 
|  | std::string Error, OutputPathName, TempPathName; | 
|  | llvm::raw_fd_ostream *OS = createOutputFile(OutputPath, Error, Binary, | 
|  | RemoveFileOnSignal, | 
|  | InFile, Extension, | 
|  | UseTemporary, | 
|  | &OutputPathName, | 
|  | &TempPathName); | 
|  | if (!OS) { | 
|  | getDiagnostics().Report(diag::err_fe_unable_to_open_output) | 
|  | << OutputPath << Error; | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | // Add the output file -- but don't try to remove "-", since this means we are | 
|  | // using stdin. | 
|  | addOutputFile(OutputFile((OutputPathName != "-") ? OutputPathName : "", | 
|  | TempPathName, OS)); | 
|  |  | 
|  | return OS; | 
|  | } | 
|  |  | 
|  | llvm::raw_fd_ostream * | 
|  | CompilerInstance::createOutputFile(StringRef OutputPath, | 
|  | std::string &Error, | 
|  | bool Binary, | 
|  | bool RemoveFileOnSignal, | 
|  | StringRef InFile, | 
|  | StringRef Extension, | 
|  | bool UseTemporary, | 
|  | std::string *ResultPathName, | 
|  | std::string *TempPathName) { | 
|  | std::string OutFile, TempFile; | 
|  | if (!OutputPath.empty()) { | 
|  | OutFile = OutputPath; | 
|  | } else if (InFile == "-") { | 
|  | OutFile = "-"; | 
|  | } else if (!Extension.empty()) { | 
|  | llvm::sys::Path Path(InFile); | 
|  | Path.eraseSuffix(); | 
|  | Path.appendSuffix(Extension); | 
|  | OutFile = Path.str(); | 
|  | } else { | 
|  | OutFile = "-"; | 
|  | } | 
|  |  | 
|  | llvm::OwningPtr<llvm::raw_fd_ostream> OS; | 
|  | std::string OSFile; | 
|  |  | 
|  | if (UseTemporary && OutFile != "-") { | 
|  | llvm::sys::Path OutPath(OutFile); | 
|  | // Only create the temporary if we can actually write to OutPath, otherwise | 
|  | // we want to fail early. | 
|  | bool Exists; | 
|  | if ((llvm::sys::fs::exists(OutPath.str(), Exists) || !Exists) || | 
|  | (OutPath.isRegularFile() && OutPath.canWrite())) { | 
|  | // Create a temporary file. | 
|  | llvm::SmallString<128> TempPath; | 
|  | TempPath = OutFile; | 
|  | TempPath += "-%%%%%%%%"; | 
|  | int fd; | 
|  | if (llvm::sys::fs::unique_file(TempPath.str(), fd, TempPath, | 
|  | /*makeAbsolute=*/false) == llvm::errc::success) { | 
|  | OS.reset(new llvm::raw_fd_ostream(fd, /*shouldClose=*/true)); | 
|  | OSFile = TempFile = TempPath.str(); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | if (!OS) { | 
|  | OSFile = OutFile; | 
|  | OS.reset( | 
|  | new llvm::raw_fd_ostream(OSFile.c_str(), Error, | 
|  | (Binary ? llvm::raw_fd_ostream::F_Binary : 0))); | 
|  | if (!Error.empty()) | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | // Make sure the out stream file gets removed if we crash. | 
|  | if (RemoveFileOnSignal) | 
|  | llvm::sys::RemoveFileOnSignal(llvm::sys::Path(OSFile)); | 
|  |  | 
|  | if (ResultPathName) | 
|  | *ResultPathName = OutFile; | 
|  | if (TempPathName) | 
|  | *TempPathName = TempFile; | 
|  |  | 
|  | return OS.take(); | 
|  | } | 
|  |  | 
|  | // Initialization Utilities | 
|  |  | 
|  | bool CompilerInstance::InitializeSourceManager(StringRef InputFile) { | 
|  | return InitializeSourceManager(InputFile, getDiagnostics(), getFileManager(), | 
|  | getSourceManager(), getFrontendOpts()); | 
|  | } | 
|  |  | 
|  | bool CompilerInstance::InitializeSourceManager(StringRef InputFile, | 
|  | DiagnosticsEngine &Diags, | 
|  | FileManager &FileMgr, | 
|  | SourceManager &SourceMgr, | 
|  | const FrontendOptions &Opts) { | 
|  | // Figure out where to get and map in the main file. | 
|  | if (InputFile != "-") { | 
|  | const FileEntry *File = FileMgr.getFile(InputFile); | 
|  | if (!File) { | 
|  | Diags.Report(diag::err_fe_error_reading) << InputFile; | 
|  | return false; | 
|  | } | 
|  | SourceMgr.createMainFileID(File); | 
|  | } else { | 
|  | llvm::OwningPtr<llvm::MemoryBuffer> SB; | 
|  | if (llvm::MemoryBuffer::getSTDIN(SB)) { | 
|  | // FIXME: Give ec.message() in this diag. | 
|  | Diags.Report(diag::err_fe_error_reading_stdin); | 
|  | return false; | 
|  | } | 
|  | const FileEntry *File = FileMgr.getVirtualFile(SB->getBufferIdentifier(), | 
|  | SB->getBufferSize(), 0); | 
|  | SourceMgr.createMainFileID(File); | 
|  | SourceMgr.overrideFileContents(File, SB.take()); | 
|  | } | 
|  |  | 
|  | assert(!SourceMgr.getMainFileID().isInvalid() && | 
|  | "Couldn't establish MainFileID!"); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | // 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. | 
|  | raw_ostream &OS = llvm::errs(); | 
|  |  | 
|  | // Create the target instance. | 
|  | setTarget(TargetInfo::CreateTargetInfo(getDiagnostics(), getTargetOpts())); | 
|  | 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. | 
|  | getTarget().setForcedLangOptions(getLangOpts()); | 
|  |  | 
|  | // Validate/process some options. | 
|  | if (getHeaderSearchOpts().Verbose) | 
|  | OS << "clang -cc1 version " CLANG_VERSION_STRING | 
|  | << " based upon " << PACKAGE_STRING | 
|  | << " default target " << llvm::sys::getDefaultTargetTriple() << "\n"; | 
|  |  | 
|  | if (getFrontendOpts().ShowTimers) | 
|  | createFrontendTimer(); | 
|  |  | 
|  | if (getFrontendOpts().ShowStats) | 
|  | llvm::EnableStatistics(); | 
|  |  | 
|  | for (unsigned i = 0, e = getFrontendOpts().Inputs.size(); i != e; ++i) { | 
|  | const std::string &InFile = getFrontendOpts().Inputs[i].second; | 
|  |  | 
|  | // Reset the ID tables if we are reusing the SourceManager. | 
|  | if (hasSourceManager()) | 
|  | getSourceManager().clearIDTables(); | 
|  |  | 
|  | if (Act.BeginSourceFile(*this, InFile, getFrontendOpts().Inputs[i].first)) { | 
|  | Act.Execute(); | 
|  | Act.EndSourceFile(); | 
|  | } | 
|  | } | 
|  |  | 
|  | if (getDiagnosticOpts().ShowCarets) { | 
|  | // 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(); | 
|  |  | 
|  | 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"; | 
|  | } | 
|  |  | 
|  | if (getFrontendOpts().ShowStats && hasFileManager()) { | 
|  | getFileManager().PrintStats(); | 
|  | OS << "\n"; | 
|  | } | 
|  |  | 
|  | return !getDiagnostics().getClient()->getNumErrors(); | 
|  | } | 
|  |  | 
|  | /// \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; | 
|  | } | 
|  |  | 
|  | namespace { | 
|  | struct CompileModuleData { | 
|  | CompilerInstance &Instance; | 
|  | GeneratePCHAction &CreateModuleAction; | 
|  | }; | 
|  | } | 
|  |  | 
|  | /// \brief Helper function that executes the module-generating action under | 
|  | /// a crash recovery context. | 
|  | static void doCompileModule(void *UserData) { | 
|  | CompileModuleData &Data = *reinterpret_cast<CompileModuleData *>(UserData); | 
|  | Data.Instance.ExecuteAction(Data.CreateModuleAction); | 
|  | } | 
|  |  | 
|  | namespace { | 
|  | /// \brief Class that manages the creation of a lock file to aid | 
|  | /// implicit coordination between different processes. | 
|  | /// | 
|  | /// The implicit coordination works by creating a ".lock" file alongside | 
|  | /// the file that we're coordinating for, using the atomicity of the file | 
|  | /// system to ensure that only a single process can create that ".lock" file. | 
|  | /// When the lock file is removed, the owning process has finished the | 
|  | /// operation. | 
|  | class LockFileManager { | 
|  | public: | 
|  | /// \brief Describes the state of a lock file. | 
|  | enum LockFileState { | 
|  | /// \brief The lock file has been created and is owned by this instance | 
|  | /// of the object. | 
|  | LFS_Owned, | 
|  | /// \brief The lock file already exists and is owned by some other | 
|  | /// instance. | 
|  | LFS_Shared, | 
|  | /// \brief An error occurred while trying to create or find the lock | 
|  | /// file. | 
|  | LFS_Error | 
|  | }; | 
|  |  | 
|  | private: | 
|  | llvm::SmallString<128> LockFileName; | 
|  | llvm::SmallString<128> UniqueLockFileName; | 
|  |  | 
|  | llvm::Optional<std::pair<std::string, int> > Owner; | 
|  | llvm::Optional<llvm::error_code> Error; | 
|  |  | 
|  | LockFileManager(const LockFileManager &); | 
|  | LockFileManager &operator=(const LockFileManager &); | 
|  |  | 
|  | static llvm::Optional<std::pair<std::string, int> > | 
|  | readLockFile(StringRef LockFileName); | 
|  |  | 
|  | static bool processStillExecuting(StringRef Hostname, int PID); | 
|  |  | 
|  | public: | 
|  |  | 
|  | LockFileManager(StringRef FileName); | 
|  | ~LockFileManager(); | 
|  |  | 
|  | /// \brief Determine the state of the lock file. | 
|  | LockFileState getState() const; | 
|  |  | 
|  | operator LockFileState() const { return getState(); } | 
|  |  | 
|  | /// \brief For a shared lock, wait until the owner releases the lock. | 
|  | void waitForUnlock(); | 
|  | }; | 
|  | } | 
|  |  | 
|  | /// \brief Attempt to read the lock file with the given name, if it exists. | 
|  | /// | 
|  | /// \param LockFileName The name of the lock file to read. | 
|  | /// | 
|  | /// \returns The process ID of the process that owns this lock file | 
|  | llvm::Optional<std::pair<std::string, int> > | 
|  | LockFileManager::readLockFile(StringRef LockFileName) { | 
|  | // Check whether the lock file exists. If not, clearly there's nothing | 
|  | // to read, so we just return. | 
|  | bool Exists = false; | 
|  | if (llvm::sys::fs::exists(LockFileName, Exists) || !Exists) | 
|  | return llvm::Optional<std::pair<std::string, int> >(); | 
|  |  | 
|  | // Read the owning host and PID out of the lock file. If it appears that the | 
|  | // owning process is dead, the lock file is invalid. | 
|  | int PID = 0; | 
|  | std::string Hostname; | 
|  | std::ifstream Input(LockFileName.str().c_str()); | 
|  | if (Input >> Hostname >> PID && PID > 0 && | 
|  | processStillExecuting(Hostname, PID)) | 
|  | return std::make_pair(Hostname, PID); | 
|  |  | 
|  | // Delete the lock file. It's invalid anyway. | 
|  | bool Existed; | 
|  | llvm::sys::fs::remove(LockFileName, Existed); | 
|  | return llvm::Optional<std::pair<std::string, int> >(); | 
|  | } | 
|  |  | 
|  | bool LockFileManager::processStillExecuting(StringRef Hostname, int PID) { | 
|  | #if LLVM_ON_UNIX | 
|  | char MyHostname[256]; | 
|  | MyHostname[255] = 0; | 
|  | MyHostname[0] = 0; | 
|  | gethostname(MyHostname, 255); | 
|  | // Check whether the process is dead. If so, we're done. | 
|  | if (MyHostname == Hostname && getsid(PID) == -1 && errno == ESRCH) | 
|  | return false; | 
|  | #endif | 
|  |  | 
|  | return true; | 
|  | } | 
|  |  | 
|  | LockFileManager::LockFileManager(StringRef FileName) | 
|  | { | 
|  | LockFileName = FileName; | 
|  | LockFileName += ".lock"; | 
|  |  | 
|  | // If the lock file already exists, don't bother to try to create our own | 
|  | // lock file; it won't work anyway. Just figure out who owns this lock file. | 
|  | if ((Owner = readLockFile(LockFileName))) | 
|  | return; | 
|  |  | 
|  | // Create a lock file that is unique to this instance. | 
|  | UniqueLockFileName = LockFileName; | 
|  | UniqueLockFileName += "-%%%%%%%%"; | 
|  | int UniqueLockFileID; | 
|  | if (llvm::error_code EC | 
|  | = llvm::sys::fs::unique_file(UniqueLockFileName.str(), | 
|  | UniqueLockFileID, | 
|  | UniqueLockFileName, | 
|  | /*makeAbsolute=*/false)) { | 
|  | Error = EC; | 
|  | return; | 
|  | } | 
|  |  | 
|  | // Write our process ID to our unique lock file. | 
|  | { | 
|  | llvm::raw_fd_ostream Out(UniqueLockFileID, /*shouldClose=*/true); | 
|  |  | 
|  | #if LLVM_ON_UNIX | 
|  | // FIXME: move getpid() call into LLVM | 
|  | char hostname[256]; | 
|  | hostname[255] = 0; | 
|  | hostname[0] = 0; | 
|  | gethostname(hostname, 255); | 
|  | Out << hostname << ' ' << getpid(); | 
|  | #else | 
|  | Out << "localhost 1"; | 
|  | #endif | 
|  | Out.close(); | 
|  |  | 
|  | if (Out.has_error()) { | 
|  | // We failed to write out PID, so make up an excuse, remove the | 
|  | // unique lock file, and fail. | 
|  | Error = llvm::make_error_code(llvm::errc::no_space_on_device); | 
|  | bool Existed; | 
|  | llvm::sys::fs::remove(UniqueLockFileName.c_str(), Existed); | 
|  | return; | 
|  | } | 
|  | } | 
|  |  | 
|  | // Create a hard link from the lock file name. If this succeeds, we're done. | 
|  | llvm::error_code EC | 
|  | = llvm::sys::fs::create_hard_link(UniqueLockFileName.str(), | 
|  | LockFileName.str()); | 
|  | if (EC == llvm::errc::success) | 
|  | return; | 
|  |  | 
|  | // Creating the hard link failed. | 
|  |  | 
|  | #ifdef LLVM_ON_UNIX | 
|  | // The creation of the hard link may appear to fail, but if stat'ing the | 
|  | // unique file returns a link count of 2, then we can still declare success. | 
|  | struct stat StatBuf; | 
|  | if (stat(UniqueLockFileName.c_str(), &StatBuf) == 0 && | 
|  | StatBuf.st_nlink == 2) | 
|  | return; | 
|  | #endif | 
|  |  | 
|  | // Someone else managed to create the lock file first. Wipe out our unique | 
|  | // lock file (it's useless now) and read the process ID from the lock file. | 
|  | bool Existed; | 
|  | llvm::sys::fs::remove(UniqueLockFileName.str(), Existed); | 
|  | if ((Owner = readLockFile(LockFileName))) | 
|  | return; | 
|  |  | 
|  | // There is a lock file that nobody owns; try to clean it up and report | 
|  | // an error. | 
|  | llvm::sys::fs::remove(LockFileName.str(), Existed); | 
|  | Error = EC; | 
|  | } | 
|  |  | 
|  | LockFileManager::LockFileState LockFileManager::getState() const { | 
|  | if (Owner) | 
|  | return LFS_Shared; | 
|  |  | 
|  | if (Error) | 
|  | return LFS_Error; | 
|  |  | 
|  | return LFS_Owned; | 
|  | } | 
|  |  | 
|  | LockFileManager::~LockFileManager() { | 
|  | if (getState() != LFS_Owned) | 
|  | return; | 
|  |  | 
|  | // Since we own the lock, remove the lock file and our own unique lock file. | 
|  | bool Existed; | 
|  | llvm::sys::fs::remove(LockFileName.str(), Existed); | 
|  | llvm::sys::fs::remove(UniqueLockFileName.str(), Existed); | 
|  | } | 
|  |  | 
|  | void LockFileManager::waitForUnlock() { | 
|  | if (getState() != LFS_Shared) | 
|  | return; | 
|  |  | 
|  | #if LLVM_ON_WIN32 | 
|  | unsigned long Interval = 1; | 
|  | #else | 
|  | struct timespec Interval; | 
|  | Interval.tv_sec = 0; | 
|  | Interval.tv_nsec = 1000000; | 
|  | #endif | 
|  | // Don't wait more than an hour for the file to appear. | 
|  | const unsigned MaxSeconds = 3600; | 
|  | do { | 
|  | // Sleep for the designated interval, to allow the owning process time to | 
|  | // finish up and | 
|  | // FIXME: Should we hook in to system APIs to get a notification when the | 
|  | // lock file is deleted? | 
|  | #if LLVM_ON_WIN32 | 
|  | Sleep(Interval); | 
|  | #else | 
|  | nanosleep(&Interval, NULL); | 
|  | #endif | 
|  | // If the file no longer exists, we're done. | 
|  | bool Exists = false; | 
|  | if (!llvm::sys::fs::exists(LockFileName.str(), Exists) && !Exists) | 
|  | return; | 
|  |  | 
|  | if (!processStillExecuting((*Owner).first, (*Owner).second)) | 
|  | return; | 
|  |  | 
|  | // Exponentially increase the time we wait for the lock to be removed. | 
|  | #if LLVM_ON_WIN32 | 
|  | Interval *= 2; | 
|  | #else | 
|  | Interval.tv_sec *= 2; | 
|  | Interval.tv_nsec *= 2; | 
|  | if (Interval.tv_nsec >= 1000000000) { | 
|  | ++Interval.tv_sec; | 
|  | Interval.tv_nsec -= 1000000000; | 
|  | } | 
|  | #endif | 
|  | } while ( | 
|  | #if LLVM_ON_WIN32 | 
|  | Interval < MaxSeconds * 1000 | 
|  | #else | 
|  | Interval.tv_sec < (time_t)MaxSeconds | 
|  | #endif | 
|  | ); | 
|  |  | 
|  | // Give up. | 
|  | } | 
|  |  | 
|  | /// \brief Compile a module file for the given module name with the given | 
|  | /// umbrella header, using the options provided by the importing compiler | 
|  | /// instance. | 
|  | static void compileModule(CompilerInstance &ImportingInstance, | 
|  | StringRef ModuleName, | 
|  | StringRef ModuleFileName, | 
|  | StringRef UmbrellaHeader) { | 
|  | LockFileManager Locked(ModuleFileName); | 
|  | switch (Locked) { | 
|  | case LockFileManager::LFS_Error: | 
|  | return; | 
|  |  | 
|  | case LockFileManager::LFS_Owned: | 
|  | // We're responsible for building the module ourselves. Do so below. | 
|  | break; | 
|  |  | 
|  | case LockFileManager::LFS_Shared: | 
|  | // Someone else is responsible for building the module. Wait for them to | 
|  | // finish. | 
|  | Locked.waitForUnlock(); | 
|  | break; | 
|  | } | 
|  |  | 
|  | // Construct a compiler invocation for creating this module. | 
|  | llvm::IntrusiveRefCntPtr<CompilerInvocation> Invocation | 
|  | (new CompilerInvocation(ImportingInstance.getInvocation())); | 
|  |  | 
|  | // For any options that aren't intended to affect how a module is built, | 
|  | // reset them to their default values. | 
|  | Invocation->getLangOpts()->resetNonModularOptions(); | 
|  | Invocation->getPreprocessorOpts().resetNonModularOptions(); | 
|  |  | 
|  | // Note the name of the module we're building. | 
|  | Invocation->getLangOpts()->CurrentModule = ModuleName; | 
|  |  | 
|  | // Note that this module is part of the module build path, so that we | 
|  | // can detect cycles in the module graph. | 
|  | Invocation->getPreprocessorOpts().ModuleBuildPath.push_back(ModuleName); | 
|  |  | 
|  | // Set up the inputs/outputs so that we build the module from its umbrella | 
|  | // header. | 
|  | FrontendOptions &FrontendOpts = Invocation->getFrontendOpts(); | 
|  | FrontendOpts.OutputFile = ModuleFileName.str(); | 
|  | FrontendOpts.DisableFree = false; | 
|  | FrontendOpts.Inputs.clear(); | 
|  | FrontendOpts.Inputs.push_back( | 
|  | std::make_pair(getSourceInputKindFromOptions(*Invocation->getLangOpts()), | 
|  | UmbrellaHeader)); | 
|  |  | 
|  | Invocation->getDiagnosticOpts().VerifyDiagnostics = 0; | 
|  |  | 
|  |  | 
|  | assert(ImportingInstance.getInvocation().getModuleHash() == | 
|  | Invocation->getModuleHash() && "Module hash mismatch!"); | 
|  |  | 
|  | // Construct a compiler instance that will be used to actually create the | 
|  | // module. | 
|  | CompilerInstance Instance; | 
|  | Instance.setInvocation(&*Invocation); | 
|  | Instance.createDiagnostics(/*argc=*/0, /*argv=*/0, | 
|  | &ImportingInstance.getDiagnosticClient(), | 
|  | /*ShouldOwnClient=*/true, | 
|  | /*ShouldCloneClient=*/true); | 
|  |  | 
|  | // Construct a module-generating action. | 
|  | GeneratePCHAction CreateModuleAction(true); | 
|  |  | 
|  | // 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; | 
|  | CompileModuleData Data = { Instance, CreateModuleAction }; | 
|  | CRC.RunSafelyOnThread(&doCompileModule, &Data, ThreadStackSize); | 
|  | } | 
|  |  | 
|  | ModuleKey CompilerInstance::loadModule(SourceLocation ImportLoc, | 
|  | IdentifierInfo &ModuleName, | 
|  | SourceLocation ModuleNameLoc) { | 
|  | // Determine what file we're searching from. | 
|  | SourceManager &SourceMgr = getSourceManager(); | 
|  | SourceLocation ExpandedImportLoc = SourceMgr.getExpansionLoc(ImportLoc); | 
|  | const FileEntry *CurFile | 
|  | = SourceMgr.getFileEntryForID(SourceMgr.getFileID(ExpandedImportLoc)); | 
|  | if (!CurFile) | 
|  | CurFile = SourceMgr.getFileEntryForID(SourceMgr.getMainFileID()); | 
|  |  | 
|  | // Search for a module with the given name. | 
|  | ModuleMap::Module *Module = 0; | 
|  | std::string ModuleFileName; | 
|  | const FileEntry *ModuleFile | 
|  | = PP->getHeaderSearchInfo().lookupModule(ModuleName.getName(), Module, | 
|  | &ModuleFileName); | 
|  |  | 
|  | bool BuildingModule = false; | 
|  | if (!ModuleFile && Module && Module->UmbrellaHeader) { | 
|  | // We didn't find the module, but there is an umbrella header that | 
|  | // can be used to create the module file. Create a separate compilation | 
|  | // module to do so. | 
|  |  | 
|  | // Check whether there is a cycle in the module graph. | 
|  | SmallVectorImpl<std::string> &ModuleBuildPath | 
|  | = getPreprocessorOpts().ModuleBuildPath; | 
|  | SmallVectorImpl<std::string>::iterator Pos | 
|  | = std::find(ModuleBuildPath.begin(), ModuleBuildPath.end(), | 
|  | ModuleName.getName()); | 
|  | if (Pos != ModuleBuildPath.end()) { | 
|  | llvm::SmallString<256> CyclePath; | 
|  | for (; Pos != ModuleBuildPath.end(); ++Pos) { | 
|  | CyclePath += *Pos; | 
|  | CyclePath += " -> "; | 
|  | } | 
|  | CyclePath += ModuleName.getName(); | 
|  |  | 
|  | getDiagnostics().Report(ModuleNameLoc, diag::err_module_cycle) | 
|  | << ModuleName.getName() << CyclePath; | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | getDiagnostics().Report(ModuleNameLoc, diag::warn_module_build) | 
|  | << ModuleName.getName(); | 
|  | BuildingModule = true; | 
|  | compileModule(*this, ModuleName.getName(), ModuleFileName, | 
|  | Module->UmbrellaHeader->getName()); | 
|  | ModuleFile = FileMgr->getFile(ModuleFileName); | 
|  | } | 
|  |  | 
|  | if (!ModuleFile) { | 
|  | getDiagnostics().Report(ModuleNameLoc, | 
|  | BuildingModule? diag::err_module_not_built | 
|  | : diag::err_module_not_found) | 
|  | << ModuleName.getName() | 
|  | << SourceRange(ImportLoc, ModuleNameLoc); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | // If we don't already have an ASTReader, create one now. | 
|  | if (!ModuleManager) { | 
|  | if (!hasASTContext()) | 
|  | createASTContext(); | 
|  |  | 
|  | std::string Sysroot = getHeaderSearchOpts().Sysroot; | 
|  | const PreprocessorOptions &PPOpts = getPreprocessorOpts(); | 
|  | ModuleManager = new ASTReader(getPreprocessor(), *Context, | 
|  | Sysroot.empty() ? "" : Sysroot.c_str(), | 
|  | PPOpts.DisablePCHValidation, | 
|  | PPOpts.DisableStatCache); | 
|  | if (hasASTConsumer()) { | 
|  | ModuleManager->setDeserializationListener( | 
|  | getASTConsumer().GetASTDeserializationListener()); | 
|  | getASTContext().setASTMutationListener( | 
|  | getASTConsumer().GetASTMutationListener()); | 
|  | } | 
|  | llvm::OwningPtr<ExternalASTSource> Source; | 
|  | Source.reset(ModuleManager); | 
|  | getASTContext().setExternalSource(Source); | 
|  | if (hasSema()) | 
|  | ModuleManager->InitializeSema(getSema()); | 
|  | if (hasASTConsumer()) | 
|  | ModuleManager->StartTranslationUnit(&getASTConsumer()); | 
|  | } | 
|  |  | 
|  | // Try to load the module we found. | 
|  | switch (ModuleManager->ReadAST(ModuleFile->getName(), | 
|  | serialization::MK_Module)) { | 
|  | case ASTReader::Success: | 
|  | break; | 
|  |  | 
|  | case ASTReader::IgnorePCH: | 
|  | // FIXME: The ASTReader will already have complained, but can we showhorn | 
|  | // that diagnostic information into a more useful form? | 
|  | return 0; | 
|  |  | 
|  | case ASTReader::Failure: | 
|  | // Already complained. | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | // FIXME: The module file's FileEntry makes a poor key indeed! | 
|  | return (ModuleKey)ModuleFile; | 
|  | } | 
|  |  |