1*67e74705SXin Li //===--- ASTUnit.cpp - ASTUnit utility --------------------------*- C++ -*-===//
2*67e74705SXin Li //
3*67e74705SXin Li // The LLVM Compiler Infrastructure
4*67e74705SXin Li //
5*67e74705SXin Li // This file is distributed under the University of Illinois Open Source
6*67e74705SXin Li // License. See LICENSE.TXT for details.
7*67e74705SXin Li //
8*67e74705SXin Li //===----------------------------------------------------------------------===//
9*67e74705SXin Li //
10*67e74705SXin Li // ASTUnit Implementation.
11*67e74705SXin Li //
12*67e74705SXin Li //===----------------------------------------------------------------------===//
13*67e74705SXin Li
14*67e74705SXin Li #include "clang/Frontend/ASTUnit.h"
15*67e74705SXin Li #include "clang/AST/ASTConsumer.h"
16*67e74705SXin Li #include "clang/AST/ASTContext.h"
17*67e74705SXin Li #include "clang/AST/DeclVisitor.h"
18*67e74705SXin Li #include "clang/AST/StmtVisitor.h"
19*67e74705SXin Li #include "clang/AST/TypeOrdering.h"
20*67e74705SXin Li #include "clang/Basic/Diagnostic.h"
21*67e74705SXin Li #include "clang/Basic/TargetInfo.h"
22*67e74705SXin Li #include "clang/Basic/TargetOptions.h"
23*67e74705SXin Li #include "clang/Basic/VirtualFileSystem.h"
24*67e74705SXin Li #include "clang/Frontend/CompilerInstance.h"
25*67e74705SXin Li #include "clang/Frontend/FrontendActions.h"
26*67e74705SXin Li #include "clang/Frontend/FrontendDiagnostic.h"
27*67e74705SXin Li #include "clang/Frontend/FrontendOptions.h"
28*67e74705SXin Li #include "clang/Frontend/MultiplexConsumer.h"
29*67e74705SXin Li #include "clang/Frontend/Utils.h"
30*67e74705SXin Li #include "clang/Lex/HeaderSearch.h"
31*67e74705SXin Li #include "clang/Lex/Preprocessor.h"
32*67e74705SXin Li #include "clang/Lex/PreprocessorOptions.h"
33*67e74705SXin Li #include "clang/Sema/Sema.h"
34*67e74705SXin Li #include "clang/Serialization/ASTReader.h"
35*67e74705SXin Li #include "clang/Serialization/ASTWriter.h"
36*67e74705SXin Li #include "llvm/ADT/ArrayRef.h"
37*67e74705SXin Li #include "llvm/ADT/StringExtras.h"
38*67e74705SXin Li #include "llvm/ADT/StringSet.h"
39*67e74705SXin Li #include "llvm/Support/CrashRecoveryContext.h"
40*67e74705SXin Li #include "llvm/Support/Host.h"
41*67e74705SXin Li #include "llvm/Support/MemoryBuffer.h"
42*67e74705SXin Li #include "llvm/Support/Mutex.h"
43*67e74705SXin Li #include "llvm/Support/MutexGuard.h"
44*67e74705SXin Li #include "llvm/Support/Path.h"
45*67e74705SXin Li #include "llvm/Support/Timer.h"
46*67e74705SXin Li #include "llvm/Support/raw_ostream.h"
47*67e74705SXin Li #include <atomic>
48*67e74705SXin Li #include <cstdio>
49*67e74705SXin Li #include <cstdlib>
50*67e74705SXin Li
51*67e74705SXin Li using namespace clang;
52*67e74705SXin Li
53*67e74705SXin Li using llvm::TimeRecord;
54*67e74705SXin Li
55*67e74705SXin Li namespace {
56*67e74705SXin Li class SimpleTimer {
57*67e74705SXin Li bool WantTiming;
58*67e74705SXin Li TimeRecord Start;
59*67e74705SXin Li std::string Output;
60*67e74705SXin Li
61*67e74705SXin Li public:
SimpleTimer(bool WantTiming)62*67e74705SXin Li explicit SimpleTimer(bool WantTiming) : WantTiming(WantTiming) {
63*67e74705SXin Li if (WantTiming)
64*67e74705SXin Li Start = TimeRecord::getCurrentTime();
65*67e74705SXin Li }
66*67e74705SXin Li
setOutput(const Twine & Output)67*67e74705SXin Li void setOutput(const Twine &Output) {
68*67e74705SXin Li if (WantTiming)
69*67e74705SXin Li this->Output = Output.str();
70*67e74705SXin Li }
71*67e74705SXin Li
~SimpleTimer()72*67e74705SXin Li ~SimpleTimer() {
73*67e74705SXin Li if (WantTiming) {
74*67e74705SXin Li TimeRecord Elapsed = TimeRecord::getCurrentTime();
75*67e74705SXin Li Elapsed -= Start;
76*67e74705SXin Li llvm::errs() << Output << ':';
77*67e74705SXin Li Elapsed.print(Elapsed, llvm::errs());
78*67e74705SXin Li llvm::errs() << '\n';
79*67e74705SXin Li }
80*67e74705SXin Li }
81*67e74705SXin Li };
82*67e74705SXin Li
83*67e74705SXin Li struct OnDiskData {
84*67e74705SXin Li /// \brief The file in which the precompiled preamble is stored.
85*67e74705SXin Li std::string PreambleFile;
86*67e74705SXin Li
87*67e74705SXin Li /// \brief Temporary files that should be removed when the ASTUnit is
88*67e74705SXin Li /// destroyed.
89*67e74705SXin Li SmallVector<std::string, 4> TemporaryFiles;
90*67e74705SXin Li
91*67e74705SXin Li /// \brief Erase temporary files.
92*67e74705SXin Li void CleanTemporaryFiles();
93*67e74705SXin Li
94*67e74705SXin Li /// \brief Erase the preamble file.
95*67e74705SXin Li void CleanPreambleFile();
96*67e74705SXin Li
97*67e74705SXin Li /// \brief Erase temporary files and the preamble file.
98*67e74705SXin Li void Cleanup();
99*67e74705SXin Li };
100*67e74705SXin Li }
101*67e74705SXin Li
getOnDiskMutex()102*67e74705SXin Li static llvm::sys::SmartMutex<false> &getOnDiskMutex() {
103*67e74705SXin Li static llvm::sys::SmartMutex<false> M(/* recursive = */ true);
104*67e74705SXin Li return M;
105*67e74705SXin Li }
106*67e74705SXin Li
107*67e74705SXin Li static void cleanupOnDiskMapAtExit();
108*67e74705SXin Li
109*67e74705SXin Li typedef llvm::DenseMap<const ASTUnit *,
110*67e74705SXin Li std::unique_ptr<OnDiskData>> OnDiskDataMap;
getOnDiskDataMap()111*67e74705SXin Li static OnDiskDataMap &getOnDiskDataMap() {
112*67e74705SXin Li static OnDiskDataMap M;
113*67e74705SXin Li static bool hasRegisteredAtExit = false;
114*67e74705SXin Li if (!hasRegisteredAtExit) {
115*67e74705SXin Li hasRegisteredAtExit = true;
116*67e74705SXin Li atexit(cleanupOnDiskMapAtExit);
117*67e74705SXin Li }
118*67e74705SXin Li return M;
119*67e74705SXin Li }
120*67e74705SXin Li
cleanupOnDiskMapAtExit()121*67e74705SXin Li static void cleanupOnDiskMapAtExit() {
122*67e74705SXin Li // Use the mutex because there can be an alive thread destroying an ASTUnit.
123*67e74705SXin Li llvm::MutexGuard Guard(getOnDiskMutex());
124*67e74705SXin Li for (const auto &I : getOnDiskDataMap()) {
125*67e74705SXin Li // We don't worry about freeing the memory associated with OnDiskDataMap.
126*67e74705SXin Li // All we care about is erasing stale files.
127*67e74705SXin Li I.second->Cleanup();
128*67e74705SXin Li }
129*67e74705SXin Li }
130*67e74705SXin Li
getOnDiskData(const ASTUnit * AU)131*67e74705SXin Li static OnDiskData &getOnDiskData(const ASTUnit *AU) {
132*67e74705SXin Li // We require the mutex since we are modifying the structure of the
133*67e74705SXin Li // DenseMap.
134*67e74705SXin Li llvm::MutexGuard Guard(getOnDiskMutex());
135*67e74705SXin Li OnDiskDataMap &M = getOnDiskDataMap();
136*67e74705SXin Li auto &D = M[AU];
137*67e74705SXin Li if (!D)
138*67e74705SXin Li D = llvm::make_unique<OnDiskData>();
139*67e74705SXin Li return *D;
140*67e74705SXin Li }
141*67e74705SXin Li
erasePreambleFile(const ASTUnit * AU)142*67e74705SXin Li static void erasePreambleFile(const ASTUnit *AU) {
143*67e74705SXin Li getOnDiskData(AU).CleanPreambleFile();
144*67e74705SXin Li }
145*67e74705SXin Li
removeOnDiskEntry(const ASTUnit * AU)146*67e74705SXin Li static void removeOnDiskEntry(const ASTUnit *AU) {
147*67e74705SXin Li // We require the mutex since we are modifying the structure of the
148*67e74705SXin Li // DenseMap.
149*67e74705SXin Li llvm::MutexGuard Guard(getOnDiskMutex());
150*67e74705SXin Li OnDiskDataMap &M = getOnDiskDataMap();
151*67e74705SXin Li OnDiskDataMap::iterator I = M.find(AU);
152*67e74705SXin Li if (I != M.end()) {
153*67e74705SXin Li I->second->Cleanup();
154*67e74705SXin Li M.erase(I);
155*67e74705SXin Li }
156*67e74705SXin Li }
157*67e74705SXin Li
setPreambleFile(const ASTUnit * AU,StringRef preambleFile)158*67e74705SXin Li static void setPreambleFile(const ASTUnit *AU, StringRef preambleFile) {
159*67e74705SXin Li getOnDiskData(AU).PreambleFile = preambleFile;
160*67e74705SXin Li }
161*67e74705SXin Li
getPreambleFile(const ASTUnit * AU)162*67e74705SXin Li static const std::string &getPreambleFile(const ASTUnit *AU) {
163*67e74705SXin Li return getOnDiskData(AU).PreambleFile;
164*67e74705SXin Li }
165*67e74705SXin Li
CleanTemporaryFiles()166*67e74705SXin Li void OnDiskData::CleanTemporaryFiles() {
167*67e74705SXin Li for (StringRef File : TemporaryFiles)
168*67e74705SXin Li llvm::sys::fs::remove(File);
169*67e74705SXin Li TemporaryFiles.clear();
170*67e74705SXin Li }
171*67e74705SXin Li
CleanPreambleFile()172*67e74705SXin Li void OnDiskData::CleanPreambleFile() {
173*67e74705SXin Li if (!PreambleFile.empty()) {
174*67e74705SXin Li llvm::sys::fs::remove(PreambleFile);
175*67e74705SXin Li PreambleFile.clear();
176*67e74705SXin Li }
177*67e74705SXin Li }
178*67e74705SXin Li
Cleanup()179*67e74705SXin Li void OnDiskData::Cleanup() {
180*67e74705SXin Li CleanTemporaryFiles();
181*67e74705SXin Li CleanPreambleFile();
182*67e74705SXin Li }
183*67e74705SXin Li
184*67e74705SXin Li struct ASTUnit::ASTWriterData {
185*67e74705SXin Li SmallString<128> Buffer;
186*67e74705SXin Li llvm::BitstreamWriter Stream;
187*67e74705SXin Li ASTWriter Writer;
188*67e74705SXin Li
ASTWriterDataASTUnit::ASTWriterData189*67e74705SXin Li ASTWriterData() : Stream(Buffer), Writer(Stream, { }) { }
190*67e74705SXin Li };
191*67e74705SXin Li
clearFileLevelDecls()192*67e74705SXin Li void ASTUnit::clearFileLevelDecls() {
193*67e74705SXin Li llvm::DeleteContainerSeconds(FileDecls);
194*67e74705SXin Li }
195*67e74705SXin Li
CleanTemporaryFiles()196*67e74705SXin Li void ASTUnit::CleanTemporaryFiles() {
197*67e74705SXin Li getOnDiskData(this).CleanTemporaryFiles();
198*67e74705SXin Li }
199*67e74705SXin Li
addTemporaryFile(StringRef TempFile)200*67e74705SXin Li void ASTUnit::addTemporaryFile(StringRef TempFile) {
201*67e74705SXin Li getOnDiskData(this).TemporaryFiles.push_back(TempFile);
202*67e74705SXin Li }
203*67e74705SXin Li
204*67e74705SXin Li /// \brief After failing to build a precompiled preamble (due to
205*67e74705SXin Li /// errors in the source that occurs in the preamble), the number of
206*67e74705SXin Li /// reparses during which we'll skip even trying to precompile the
207*67e74705SXin Li /// preamble.
208*67e74705SXin Li const unsigned DefaultPreambleRebuildInterval = 5;
209*67e74705SXin Li
210*67e74705SXin Li /// \brief Tracks the number of ASTUnit objects that are currently active.
211*67e74705SXin Li ///
212*67e74705SXin Li /// Used for debugging purposes only.
213*67e74705SXin Li static std::atomic<unsigned> ActiveASTUnitObjects;
214*67e74705SXin Li
ASTUnit(bool _MainFileIsAST)215*67e74705SXin Li ASTUnit::ASTUnit(bool _MainFileIsAST)
216*67e74705SXin Li : Reader(nullptr), HadModuleLoaderFatalFailure(false),
217*67e74705SXin Li OnlyLocalDecls(false), CaptureDiagnostics(false),
218*67e74705SXin Li MainFileIsAST(_MainFileIsAST),
219*67e74705SXin Li TUKind(TU_Complete), WantTiming(getenv("LIBCLANG_TIMING")),
220*67e74705SXin Li OwnsRemappedFileBuffers(true),
221*67e74705SXin Li NumStoredDiagnosticsFromDriver(0),
222*67e74705SXin Li PreambleRebuildCounter(0),
223*67e74705SXin Li NumWarningsInPreamble(0),
224*67e74705SXin Li ShouldCacheCodeCompletionResults(false),
225*67e74705SXin Li IncludeBriefCommentsInCodeCompletion(false), UserFilesAreVolatile(false),
226*67e74705SXin Li CompletionCacheTopLevelHashValue(0),
227*67e74705SXin Li PreambleTopLevelHashValue(0),
228*67e74705SXin Li CurrentTopLevelHashValue(0),
229*67e74705SXin Li UnsafeToFree(false) {
230*67e74705SXin Li if (getenv("LIBCLANG_OBJTRACKING"))
231*67e74705SXin Li fprintf(stderr, "+++ %u translation units\n", ++ActiveASTUnitObjects);
232*67e74705SXin Li }
233*67e74705SXin Li
~ASTUnit()234*67e74705SXin Li ASTUnit::~ASTUnit() {
235*67e74705SXin Li // If we loaded from an AST file, balance out the BeginSourceFile call.
236*67e74705SXin Li if (MainFileIsAST && getDiagnostics().getClient()) {
237*67e74705SXin Li getDiagnostics().getClient()->EndSourceFile();
238*67e74705SXin Li }
239*67e74705SXin Li
240*67e74705SXin Li clearFileLevelDecls();
241*67e74705SXin Li
242*67e74705SXin Li // Clean up the temporary files and the preamble file.
243*67e74705SXin Li removeOnDiskEntry(this);
244*67e74705SXin Li
245*67e74705SXin Li // Free the buffers associated with remapped files. We are required to
246*67e74705SXin Li // perform this operation here because we explicitly request that the
247*67e74705SXin Li // compiler instance *not* free these buffers for each invocation of the
248*67e74705SXin Li // parser.
249*67e74705SXin Li if (Invocation.get() && OwnsRemappedFileBuffers) {
250*67e74705SXin Li PreprocessorOptions &PPOpts = Invocation->getPreprocessorOpts();
251*67e74705SXin Li for (const auto &RB : PPOpts.RemappedFileBuffers)
252*67e74705SXin Li delete RB.second;
253*67e74705SXin Li }
254*67e74705SXin Li
255*67e74705SXin Li ClearCachedCompletionResults();
256*67e74705SXin Li
257*67e74705SXin Li if (getenv("LIBCLANG_OBJTRACKING"))
258*67e74705SXin Li fprintf(stderr, "--- %u translation units\n", --ActiveASTUnitObjects);
259*67e74705SXin Li }
260*67e74705SXin Li
setPreprocessor(Preprocessor * pp)261*67e74705SXin Li void ASTUnit::setPreprocessor(Preprocessor *pp) { PP = pp; }
262*67e74705SXin Li
263*67e74705SXin Li /// \brief Determine the set of code-completion contexts in which this
264*67e74705SXin Li /// declaration should be shown.
getDeclShowContexts(const NamedDecl * ND,const LangOptions & LangOpts,bool & IsNestedNameSpecifier)265*67e74705SXin Li static unsigned getDeclShowContexts(const NamedDecl *ND,
266*67e74705SXin Li const LangOptions &LangOpts,
267*67e74705SXin Li bool &IsNestedNameSpecifier) {
268*67e74705SXin Li IsNestedNameSpecifier = false;
269*67e74705SXin Li
270*67e74705SXin Li if (isa<UsingShadowDecl>(ND))
271*67e74705SXin Li ND = dyn_cast<NamedDecl>(ND->getUnderlyingDecl());
272*67e74705SXin Li if (!ND)
273*67e74705SXin Li return 0;
274*67e74705SXin Li
275*67e74705SXin Li uint64_t Contexts = 0;
276*67e74705SXin Li if (isa<TypeDecl>(ND) || isa<ObjCInterfaceDecl>(ND) ||
277*67e74705SXin Li isa<ClassTemplateDecl>(ND) || isa<TemplateTemplateParmDecl>(ND)) {
278*67e74705SXin Li // Types can appear in these contexts.
279*67e74705SXin Li if (LangOpts.CPlusPlus || !isa<TagDecl>(ND))
280*67e74705SXin Li Contexts |= (1LL << CodeCompletionContext::CCC_TopLevel)
281*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_ObjCIvarList)
282*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_ClassStructUnion)
283*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_Statement)
284*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_Type)
285*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_ParenthesizedExpression);
286*67e74705SXin Li
287*67e74705SXin Li // In C++, types can appear in expressions contexts (for functional casts).
288*67e74705SXin Li if (LangOpts.CPlusPlus)
289*67e74705SXin Li Contexts |= (1LL << CodeCompletionContext::CCC_Expression);
290*67e74705SXin Li
291*67e74705SXin Li // In Objective-C, message sends can send interfaces. In Objective-C++,
292*67e74705SXin Li // all types are available due to functional casts.
293*67e74705SXin Li if (LangOpts.CPlusPlus || isa<ObjCInterfaceDecl>(ND))
294*67e74705SXin Li Contexts |= (1LL << CodeCompletionContext::CCC_ObjCMessageReceiver);
295*67e74705SXin Li
296*67e74705SXin Li // In Objective-C, you can only be a subclass of another Objective-C class
297*67e74705SXin Li if (isa<ObjCInterfaceDecl>(ND))
298*67e74705SXin Li Contexts |= (1LL << CodeCompletionContext::CCC_ObjCInterfaceName);
299*67e74705SXin Li
300*67e74705SXin Li // Deal with tag names.
301*67e74705SXin Li if (isa<EnumDecl>(ND)) {
302*67e74705SXin Li Contexts |= (1LL << CodeCompletionContext::CCC_EnumTag);
303*67e74705SXin Li
304*67e74705SXin Li // Part of the nested-name-specifier in C++0x.
305*67e74705SXin Li if (LangOpts.CPlusPlus11)
306*67e74705SXin Li IsNestedNameSpecifier = true;
307*67e74705SXin Li } else if (const RecordDecl *Record = dyn_cast<RecordDecl>(ND)) {
308*67e74705SXin Li if (Record->isUnion())
309*67e74705SXin Li Contexts |= (1LL << CodeCompletionContext::CCC_UnionTag);
310*67e74705SXin Li else
311*67e74705SXin Li Contexts |= (1LL << CodeCompletionContext::CCC_ClassOrStructTag);
312*67e74705SXin Li
313*67e74705SXin Li if (LangOpts.CPlusPlus)
314*67e74705SXin Li IsNestedNameSpecifier = true;
315*67e74705SXin Li } else if (isa<ClassTemplateDecl>(ND))
316*67e74705SXin Li IsNestedNameSpecifier = true;
317*67e74705SXin Li } else if (isa<ValueDecl>(ND) || isa<FunctionTemplateDecl>(ND)) {
318*67e74705SXin Li // Values can appear in these contexts.
319*67e74705SXin Li Contexts = (1LL << CodeCompletionContext::CCC_Statement)
320*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_Expression)
321*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_ParenthesizedExpression)
322*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_ObjCMessageReceiver);
323*67e74705SXin Li } else if (isa<ObjCProtocolDecl>(ND)) {
324*67e74705SXin Li Contexts = (1LL << CodeCompletionContext::CCC_ObjCProtocolName);
325*67e74705SXin Li } else if (isa<ObjCCategoryDecl>(ND)) {
326*67e74705SXin Li Contexts = (1LL << CodeCompletionContext::CCC_ObjCCategoryName);
327*67e74705SXin Li } else if (isa<NamespaceDecl>(ND) || isa<NamespaceAliasDecl>(ND)) {
328*67e74705SXin Li Contexts = (1LL << CodeCompletionContext::CCC_Namespace);
329*67e74705SXin Li
330*67e74705SXin Li // Part of the nested-name-specifier.
331*67e74705SXin Li IsNestedNameSpecifier = true;
332*67e74705SXin Li }
333*67e74705SXin Li
334*67e74705SXin Li return Contexts;
335*67e74705SXin Li }
336*67e74705SXin Li
CacheCodeCompletionResults()337*67e74705SXin Li void ASTUnit::CacheCodeCompletionResults() {
338*67e74705SXin Li if (!TheSema)
339*67e74705SXin Li return;
340*67e74705SXin Li
341*67e74705SXin Li SimpleTimer Timer(WantTiming);
342*67e74705SXin Li Timer.setOutput("Cache global code completions for " + getMainFileName());
343*67e74705SXin Li
344*67e74705SXin Li // Clear out the previous results.
345*67e74705SXin Li ClearCachedCompletionResults();
346*67e74705SXin Li
347*67e74705SXin Li // Gather the set of global code completions.
348*67e74705SXin Li typedef CodeCompletionResult Result;
349*67e74705SXin Li SmallVector<Result, 8> Results;
350*67e74705SXin Li CachedCompletionAllocator = new GlobalCodeCompletionAllocator;
351*67e74705SXin Li CodeCompletionTUInfo CCTUInfo(CachedCompletionAllocator);
352*67e74705SXin Li TheSema->GatherGlobalCodeCompletions(*CachedCompletionAllocator,
353*67e74705SXin Li CCTUInfo, Results);
354*67e74705SXin Li
355*67e74705SXin Li // Translate global code completions into cached completions.
356*67e74705SXin Li llvm::DenseMap<CanQualType, unsigned> CompletionTypes;
357*67e74705SXin Li CodeCompletionContext CCContext(CodeCompletionContext::CCC_TopLevel);
358*67e74705SXin Li
359*67e74705SXin Li for (Result &R : Results) {
360*67e74705SXin Li switch (R.Kind) {
361*67e74705SXin Li case Result::RK_Declaration: {
362*67e74705SXin Li bool IsNestedNameSpecifier = false;
363*67e74705SXin Li CachedCodeCompletionResult CachedResult;
364*67e74705SXin Li CachedResult.Completion = R.CreateCodeCompletionString(
365*67e74705SXin Li *TheSema, CCContext, *CachedCompletionAllocator, CCTUInfo,
366*67e74705SXin Li IncludeBriefCommentsInCodeCompletion);
367*67e74705SXin Li CachedResult.ShowInContexts = getDeclShowContexts(
368*67e74705SXin Li R.Declaration, Ctx->getLangOpts(), IsNestedNameSpecifier);
369*67e74705SXin Li CachedResult.Priority = R.Priority;
370*67e74705SXin Li CachedResult.Kind = R.CursorKind;
371*67e74705SXin Li CachedResult.Availability = R.Availability;
372*67e74705SXin Li
373*67e74705SXin Li // Keep track of the type of this completion in an ASTContext-agnostic
374*67e74705SXin Li // way.
375*67e74705SXin Li QualType UsageType = getDeclUsageType(*Ctx, R.Declaration);
376*67e74705SXin Li if (UsageType.isNull()) {
377*67e74705SXin Li CachedResult.TypeClass = STC_Void;
378*67e74705SXin Li CachedResult.Type = 0;
379*67e74705SXin Li } else {
380*67e74705SXin Li CanQualType CanUsageType
381*67e74705SXin Li = Ctx->getCanonicalType(UsageType.getUnqualifiedType());
382*67e74705SXin Li CachedResult.TypeClass = getSimplifiedTypeClass(CanUsageType);
383*67e74705SXin Li
384*67e74705SXin Li // Determine whether we have already seen this type. If so, we save
385*67e74705SXin Li // ourselves the work of formatting the type string by using the
386*67e74705SXin Li // temporary, CanQualType-based hash table to find the associated value.
387*67e74705SXin Li unsigned &TypeValue = CompletionTypes[CanUsageType];
388*67e74705SXin Li if (TypeValue == 0) {
389*67e74705SXin Li TypeValue = CompletionTypes.size();
390*67e74705SXin Li CachedCompletionTypes[QualType(CanUsageType).getAsString()]
391*67e74705SXin Li = TypeValue;
392*67e74705SXin Li }
393*67e74705SXin Li
394*67e74705SXin Li CachedResult.Type = TypeValue;
395*67e74705SXin Li }
396*67e74705SXin Li
397*67e74705SXin Li CachedCompletionResults.push_back(CachedResult);
398*67e74705SXin Li
399*67e74705SXin Li /// Handle nested-name-specifiers in C++.
400*67e74705SXin Li if (TheSema->Context.getLangOpts().CPlusPlus && IsNestedNameSpecifier &&
401*67e74705SXin Li !R.StartsNestedNameSpecifier) {
402*67e74705SXin Li // The contexts in which a nested-name-specifier can appear in C++.
403*67e74705SXin Li uint64_t NNSContexts
404*67e74705SXin Li = (1LL << CodeCompletionContext::CCC_TopLevel)
405*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_ObjCIvarList)
406*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_ClassStructUnion)
407*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_Statement)
408*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_Expression)
409*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_ObjCMessageReceiver)
410*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_EnumTag)
411*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_UnionTag)
412*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_ClassOrStructTag)
413*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_Type)
414*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_PotentiallyQualifiedName)
415*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_ParenthesizedExpression);
416*67e74705SXin Li
417*67e74705SXin Li if (isa<NamespaceDecl>(R.Declaration) ||
418*67e74705SXin Li isa<NamespaceAliasDecl>(R.Declaration))
419*67e74705SXin Li NNSContexts |= (1LL << CodeCompletionContext::CCC_Namespace);
420*67e74705SXin Li
421*67e74705SXin Li if (unsigned RemainingContexts
422*67e74705SXin Li = NNSContexts & ~CachedResult.ShowInContexts) {
423*67e74705SXin Li // If there any contexts where this completion can be a
424*67e74705SXin Li // nested-name-specifier but isn't already an option, create a
425*67e74705SXin Li // nested-name-specifier completion.
426*67e74705SXin Li R.StartsNestedNameSpecifier = true;
427*67e74705SXin Li CachedResult.Completion = R.CreateCodeCompletionString(
428*67e74705SXin Li *TheSema, CCContext, *CachedCompletionAllocator, CCTUInfo,
429*67e74705SXin Li IncludeBriefCommentsInCodeCompletion);
430*67e74705SXin Li CachedResult.ShowInContexts = RemainingContexts;
431*67e74705SXin Li CachedResult.Priority = CCP_NestedNameSpecifier;
432*67e74705SXin Li CachedResult.TypeClass = STC_Void;
433*67e74705SXin Li CachedResult.Type = 0;
434*67e74705SXin Li CachedCompletionResults.push_back(CachedResult);
435*67e74705SXin Li }
436*67e74705SXin Li }
437*67e74705SXin Li break;
438*67e74705SXin Li }
439*67e74705SXin Li
440*67e74705SXin Li case Result::RK_Keyword:
441*67e74705SXin Li case Result::RK_Pattern:
442*67e74705SXin Li // Ignore keywords and patterns; we don't care, since they are so
443*67e74705SXin Li // easily regenerated.
444*67e74705SXin Li break;
445*67e74705SXin Li
446*67e74705SXin Li case Result::RK_Macro: {
447*67e74705SXin Li CachedCodeCompletionResult CachedResult;
448*67e74705SXin Li CachedResult.Completion = R.CreateCodeCompletionString(
449*67e74705SXin Li *TheSema, CCContext, *CachedCompletionAllocator, CCTUInfo,
450*67e74705SXin Li IncludeBriefCommentsInCodeCompletion);
451*67e74705SXin Li CachedResult.ShowInContexts
452*67e74705SXin Li = (1LL << CodeCompletionContext::CCC_TopLevel)
453*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_ObjCInterface)
454*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_ObjCImplementation)
455*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_ObjCIvarList)
456*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_ClassStructUnion)
457*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_Statement)
458*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_Expression)
459*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_ObjCMessageReceiver)
460*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_MacroNameUse)
461*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_PreprocessorExpression)
462*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_ParenthesizedExpression)
463*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_OtherWithMacros);
464*67e74705SXin Li
465*67e74705SXin Li CachedResult.Priority = R.Priority;
466*67e74705SXin Li CachedResult.Kind = R.CursorKind;
467*67e74705SXin Li CachedResult.Availability = R.Availability;
468*67e74705SXin Li CachedResult.TypeClass = STC_Void;
469*67e74705SXin Li CachedResult.Type = 0;
470*67e74705SXin Li CachedCompletionResults.push_back(CachedResult);
471*67e74705SXin Li break;
472*67e74705SXin Li }
473*67e74705SXin Li }
474*67e74705SXin Li }
475*67e74705SXin Li
476*67e74705SXin Li // Save the current top-level hash value.
477*67e74705SXin Li CompletionCacheTopLevelHashValue = CurrentTopLevelHashValue;
478*67e74705SXin Li }
479*67e74705SXin Li
ClearCachedCompletionResults()480*67e74705SXin Li void ASTUnit::ClearCachedCompletionResults() {
481*67e74705SXin Li CachedCompletionResults.clear();
482*67e74705SXin Li CachedCompletionTypes.clear();
483*67e74705SXin Li CachedCompletionAllocator = nullptr;
484*67e74705SXin Li }
485*67e74705SXin Li
486*67e74705SXin Li namespace {
487*67e74705SXin Li
488*67e74705SXin Li /// \brief Gathers information from ASTReader that will be used to initialize
489*67e74705SXin Li /// a Preprocessor.
490*67e74705SXin Li class ASTInfoCollector : public ASTReaderListener {
491*67e74705SXin Li Preprocessor &PP;
492*67e74705SXin Li ASTContext &Context;
493*67e74705SXin Li LangOptions &LangOpt;
494*67e74705SXin Li std::shared_ptr<TargetOptions> &TargetOpts;
495*67e74705SXin Li IntrusiveRefCntPtr<TargetInfo> &Target;
496*67e74705SXin Li unsigned &Counter;
497*67e74705SXin Li
498*67e74705SXin Li bool InitializedLanguage;
499*67e74705SXin Li public:
ASTInfoCollector(Preprocessor & PP,ASTContext & Context,LangOptions & LangOpt,std::shared_ptr<TargetOptions> & TargetOpts,IntrusiveRefCntPtr<TargetInfo> & Target,unsigned & Counter)500*67e74705SXin Li ASTInfoCollector(Preprocessor &PP, ASTContext &Context, LangOptions &LangOpt,
501*67e74705SXin Li std::shared_ptr<TargetOptions> &TargetOpts,
502*67e74705SXin Li IntrusiveRefCntPtr<TargetInfo> &Target, unsigned &Counter)
503*67e74705SXin Li : PP(PP), Context(Context), LangOpt(LangOpt), TargetOpts(TargetOpts),
504*67e74705SXin Li Target(Target), Counter(Counter), InitializedLanguage(false) {}
505*67e74705SXin Li
ReadLanguageOptions(const LangOptions & LangOpts,bool Complain,bool AllowCompatibleDifferences)506*67e74705SXin Li bool ReadLanguageOptions(const LangOptions &LangOpts, bool Complain,
507*67e74705SXin Li bool AllowCompatibleDifferences) override {
508*67e74705SXin Li if (InitializedLanguage)
509*67e74705SXin Li return false;
510*67e74705SXin Li
511*67e74705SXin Li LangOpt = LangOpts;
512*67e74705SXin Li InitializedLanguage = true;
513*67e74705SXin Li
514*67e74705SXin Li updated();
515*67e74705SXin Li return false;
516*67e74705SXin Li }
517*67e74705SXin Li
ReadTargetOptions(const TargetOptions & TargetOpts,bool Complain,bool AllowCompatibleDifferences)518*67e74705SXin Li bool ReadTargetOptions(const TargetOptions &TargetOpts, bool Complain,
519*67e74705SXin Li bool AllowCompatibleDifferences) override {
520*67e74705SXin Li // If we've already initialized the target, don't do it again.
521*67e74705SXin Li if (Target)
522*67e74705SXin Li return false;
523*67e74705SXin Li
524*67e74705SXin Li this->TargetOpts = std::make_shared<TargetOptions>(TargetOpts);
525*67e74705SXin Li Target =
526*67e74705SXin Li TargetInfo::CreateTargetInfo(PP.getDiagnostics(), this->TargetOpts);
527*67e74705SXin Li
528*67e74705SXin Li updated();
529*67e74705SXin Li return false;
530*67e74705SXin Li }
531*67e74705SXin Li
ReadCounter(const serialization::ModuleFile & M,unsigned Value)532*67e74705SXin Li void ReadCounter(const serialization::ModuleFile &M,
533*67e74705SXin Li unsigned Value) override {
534*67e74705SXin Li Counter = Value;
535*67e74705SXin Li }
536*67e74705SXin Li
537*67e74705SXin Li private:
updated()538*67e74705SXin Li void updated() {
539*67e74705SXin Li if (!Target || !InitializedLanguage)
540*67e74705SXin Li return;
541*67e74705SXin Li
542*67e74705SXin Li // Inform the target of the language options.
543*67e74705SXin Li //
544*67e74705SXin Li // FIXME: We shouldn't need to do this, the target should be immutable once
545*67e74705SXin Li // created. This complexity should be lifted elsewhere.
546*67e74705SXin Li Target->adjust(LangOpt);
547*67e74705SXin Li
548*67e74705SXin Li // Initialize the preprocessor.
549*67e74705SXin Li PP.Initialize(*Target);
550*67e74705SXin Li
551*67e74705SXin Li // Initialize the ASTContext
552*67e74705SXin Li Context.InitBuiltinTypes(*Target);
553*67e74705SXin Li
554*67e74705SXin Li // We didn't have access to the comment options when the ASTContext was
555*67e74705SXin Li // constructed, so register them now.
556*67e74705SXin Li Context.getCommentCommandTraits().registerCommentOptions(
557*67e74705SXin Li LangOpt.CommentOpts);
558*67e74705SXin Li }
559*67e74705SXin Li };
560*67e74705SXin Li
561*67e74705SXin Li /// \brief Diagnostic consumer that saves each diagnostic it is given.
562*67e74705SXin Li class StoredDiagnosticConsumer : public DiagnosticConsumer {
563*67e74705SXin Li SmallVectorImpl<StoredDiagnostic> &StoredDiags;
564*67e74705SXin Li SourceManager *SourceMgr;
565*67e74705SXin Li
566*67e74705SXin Li public:
StoredDiagnosticConsumer(SmallVectorImpl<StoredDiagnostic> & StoredDiags)567*67e74705SXin Li explicit StoredDiagnosticConsumer(
568*67e74705SXin Li SmallVectorImpl<StoredDiagnostic> &StoredDiags)
569*67e74705SXin Li : StoredDiags(StoredDiags), SourceMgr(nullptr) {}
570*67e74705SXin Li
BeginSourceFile(const LangOptions & LangOpts,const Preprocessor * PP=nullptr)571*67e74705SXin Li void BeginSourceFile(const LangOptions &LangOpts,
572*67e74705SXin Li const Preprocessor *PP = nullptr) override {
573*67e74705SXin Li if (PP)
574*67e74705SXin Li SourceMgr = &PP->getSourceManager();
575*67e74705SXin Li }
576*67e74705SXin Li
577*67e74705SXin Li void HandleDiagnostic(DiagnosticsEngine::Level Level,
578*67e74705SXin Li const Diagnostic &Info) override;
579*67e74705SXin Li };
580*67e74705SXin Li
581*67e74705SXin Li /// \brief RAII object that optionally captures diagnostics, if
582*67e74705SXin Li /// there is no diagnostic client to capture them already.
583*67e74705SXin Li class CaptureDroppedDiagnostics {
584*67e74705SXin Li DiagnosticsEngine &Diags;
585*67e74705SXin Li StoredDiagnosticConsumer Client;
586*67e74705SXin Li DiagnosticConsumer *PreviousClient;
587*67e74705SXin Li std::unique_ptr<DiagnosticConsumer> OwningPreviousClient;
588*67e74705SXin Li
589*67e74705SXin Li public:
CaptureDroppedDiagnostics(bool RequestCapture,DiagnosticsEngine & Diags,SmallVectorImpl<StoredDiagnostic> & StoredDiags)590*67e74705SXin Li CaptureDroppedDiagnostics(bool RequestCapture, DiagnosticsEngine &Diags,
591*67e74705SXin Li SmallVectorImpl<StoredDiagnostic> &StoredDiags)
592*67e74705SXin Li : Diags(Diags), Client(StoredDiags), PreviousClient(nullptr)
593*67e74705SXin Li {
594*67e74705SXin Li if (RequestCapture || Diags.getClient() == nullptr) {
595*67e74705SXin Li OwningPreviousClient = Diags.takeClient();
596*67e74705SXin Li PreviousClient = Diags.getClient();
597*67e74705SXin Li Diags.setClient(&Client, false);
598*67e74705SXin Li }
599*67e74705SXin Li }
600*67e74705SXin Li
~CaptureDroppedDiagnostics()601*67e74705SXin Li ~CaptureDroppedDiagnostics() {
602*67e74705SXin Li if (Diags.getClient() == &Client)
603*67e74705SXin Li Diags.setClient(PreviousClient, !!OwningPreviousClient.release());
604*67e74705SXin Li }
605*67e74705SXin Li };
606*67e74705SXin Li
607*67e74705SXin Li } // anonymous namespace
608*67e74705SXin Li
HandleDiagnostic(DiagnosticsEngine::Level Level,const Diagnostic & Info)609*67e74705SXin Li void StoredDiagnosticConsumer::HandleDiagnostic(DiagnosticsEngine::Level Level,
610*67e74705SXin Li const Diagnostic &Info) {
611*67e74705SXin Li // Default implementation (Warnings/errors count).
612*67e74705SXin Li DiagnosticConsumer::HandleDiagnostic(Level, Info);
613*67e74705SXin Li
614*67e74705SXin Li // Only record the diagnostic if it's part of the source manager we know
615*67e74705SXin Li // about. This effectively drops diagnostics from modules we're building.
616*67e74705SXin Li // FIXME: In the long run, ee don't want to drop source managers from modules.
617*67e74705SXin Li if (!Info.hasSourceManager() || &Info.getSourceManager() == SourceMgr)
618*67e74705SXin Li StoredDiags.emplace_back(Level, Info);
619*67e74705SXin Li }
620*67e74705SXin Li
getASTMutationListener()621*67e74705SXin Li ASTMutationListener *ASTUnit::getASTMutationListener() {
622*67e74705SXin Li if (WriterData)
623*67e74705SXin Li return &WriterData->Writer;
624*67e74705SXin Li return nullptr;
625*67e74705SXin Li }
626*67e74705SXin Li
getDeserializationListener()627*67e74705SXin Li ASTDeserializationListener *ASTUnit::getDeserializationListener() {
628*67e74705SXin Li if (WriterData)
629*67e74705SXin Li return &WriterData->Writer;
630*67e74705SXin Li return nullptr;
631*67e74705SXin Li }
632*67e74705SXin Li
633*67e74705SXin Li std::unique_ptr<llvm::MemoryBuffer>
getBufferForFile(StringRef Filename,std::string * ErrorStr)634*67e74705SXin Li ASTUnit::getBufferForFile(StringRef Filename, std::string *ErrorStr) {
635*67e74705SXin Li assert(FileMgr);
636*67e74705SXin Li auto Buffer = FileMgr->getBufferForFile(Filename);
637*67e74705SXin Li if (Buffer)
638*67e74705SXin Li return std::move(*Buffer);
639*67e74705SXin Li if (ErrorStr)
640*67e74705SXin Li *ErrorStr = Buffer.getError().message();
641*67e74705SXin Li return nullptr;
642*67e74705SXin Li }
643*67e74705SXin Li
644*67e74705SXin Li /// \brief Configure the diagnostics object for use with ASTUnit.
ConfigureDiags(IntrusiveRefCntPtr<DiagnosticsEngine> Diags,ASTUnit & AST,bool CaptureDiagnostics)645*67e74705SXin Li void ASTUnit::ConfigureDiags(IntrusiveRefCntPtr<DiagnosticsEngine> Diags,
646*67e74705SXin Li ASTUnit &AST, bool CaptureDiagnostics) {
647*67e74705SXin Li assert(Diags.get() && "no DiagnosticsEngine was provided");
648*67e74705SXin Li if (CaptureDiagnostics)
649*67e74705SXin Li Diags->setClient(new StoredDiagnosticConsumer(AST.StoredDiagnostics));
650*67e74705SXin Li }
651*67e74705SXin Li
LoadFromASTFile(const std::string & Filename,const PCHContainerReader & PCHContainerRdr,IntrusiveRefCntPtr<DiagnosticsEngine> Diags,const FileSystemOptions & FileSystemOpts,bool UseDebugInfo,bool OnlyLocalDecls,ArrayRef<RemappedFile> RemappedFiles,bool CaptureDiagnostics,bool AllowPCHWithCompilerErrors,bool UserFilesAreVolatile)652*67e74705SXin Li std::unique_ptr<ASTUnit> ASTUnit::LoadFromASTFile(
653*67e74705SXin Li const std::string &Filename, const PCHContainerReader &PCHContainerRdr,
654*67e74705SXin Li IntrusiveRefCntPtr<DiagnosticsEngine> Diags,
655*67e74705SXin Li const FileSystemOptions &FileSystemOpts, bool UseDebugInfo,
656*67e74705SXin Li bool OnlyLocalDecls, ArrayRef<RemappedFile> RemappedFiles,
657*67e74705SXin Li bool CaptureDiagnostics, bool AllowPCHWithCompilerErrors,
658*67e74705SXin Li bool UserFilesAreVolatile) {
659*67e74705SXin Li std::unique_ptr<ASTUnit> AST(new ASTUnit(true));
660*67e74705SXin Li
661*67e74705SXin Li // Recover resources if we crash before exiting this method.
662*67e74705SXin Li llvm::CrashRecoveryContextCleanupRegistrar<ASTUnit>
663*67e74705SXin Li ASTUnitCleanup(AST.get());
664*67e74705SXin Li llvm::CrashRecoveryContextCleanupRegistrar<DiagnosticsEngine,
665*67e74705SXin Li llvm::CrashRecoveryContextReleaseRefCleanup<DiagnosticsEngine> >
666*67e74705SXin Li DiagCleanup(Diags.get());
667*67e74705SXin Li
668*67e74705SXin Li ConfigureDiags(Diags, *AST, CaptureDiagnostics);
669*67e74705SXin Li
670*67e74705SXin Li AST->OnlyLocalDecls = OnlyLocalDecls;
671*67e74705SXin Li AST->CaptureDiagnostics = CaptureDiagnostics;
672*67e74705SXin Li AST->Diagnostics = Diags;
673*67e74705SXin Li IntrusiveRefCntPtr<vfs::FileSystem> VFS = vfs::getRealFileSystem();
674*67e74705SXin Li AST->FileMgr = new FileManager(FileSystemOpts, VFS);
675*67e74705SXin Li AST->UserFilesAreVolatile = UserFilesAreVolatile;
676*67e74705SXin Li AST->SourceMgr = new SourceManager(AST->getDiagnostics(),
677*67e74705SXin Li AST->getFileManager(),
678*67e74705SXin Li UserFilesAreVolatile);
679*67e74705SXin Li AST->HSOpts = new HeaderSearchOptions();
680*67e74705SXin Li AST->HSOpts->ModuleFormat = PCHContainerRdr.getFormat();
681*67e74705SXin Li AST->HeaderInfo.reset(new HeaderSearch(AST->HSOpts,
682*67e74705SXin Li AST->getSourceManager(),
683*67e74705SXin Li AST->getDiagnostics(),
684*67e74705SXin Li AST->ASTFileLangOpts,
685*67e74705SXin Li /*Target=*/nullptr));
686*67e74705SXin Li
687*67e74705SXin Li PreprocessorOptions *PPOpts = new PreprocessorOptions();
688*67e74705SXin Li
689*67e74705SXin Li for (const auto &RemappedFile : RemappedFiles)
690*67e74705SXin Li PPOpts->addRemappedFile(RemappedFile.first, RemappedFile.second);
691*67e74705SXin Li
692*67e74705SXin Li // Gather Info for preprocessor construction later on.
693*67e74705SXin Li
694*67e74705SXin Li HeaderSearch &HeaderInfo = *AST->HeaderInfo;
695*67e74705SXin Li unsigned Counter;
696*67e74705SXin Li
697*67e74705SXin Li AST->PP =
698*67e74705SXin Li new Preprocessor(PPOpts, AST->getDiagnostics(), AST->ASTFileLangOpts,
699*67e74705SXin Li AST->getSourceManager(), HeaderInfo, *AST,
700*67e74705SXin Li /*IILookup=*/nullptr,
701*67e74705SXin Li /*OwnsHeaderSearch=*/false);
702*67e74705SXin Li Preprocessor &PP = *AST->PP;
703*67e74705SXin Li
704*67e74705SXin Li AST->Ctx = new ASTContext(AST->ASTFileLangOpts, AST->getSourceManager(),
705*67e74705SXin Li PP.getIdentifierTable(), PP.getSelectorTable(),
706*67e74705SXin Li PP.getBuiltinInfo());
707*67e74705SXin Li ASTContext &Context = *AST->Ctx;
708*67e74705SXin Li
709*67e74705SXin Li bool disableValid = false;
710*67e74705SXin Li if (::getenv("LIBCLANG_DISABLE_PCH_VALIDATION"))
711*67e74705SXin Li disableValid = true;
712*67e74705SXin Li AST->Reader = new ASTReader(PP, Context, PCHContainerRdr, { },
713*67e74705SXin Li /*isysroot=*/"",
714*67e74705SXin Li /*DisableValidation=*/disableValid,
715*67e74705SXin Li AllowPCHWithCompilerErrors);
716*67e74705SXin Li
717*67e74705SXin Li AST->Reader->setListener(llvm::make_unique<ASTInfoCollector>(
718*67e74705SXin Li *AST->PP, Context, AST->ASTFileLangOpts, AST->TargetOpts, AST->Target,
719*67e74705SXin Li Counter));
720*67e74705SXin Li
721*67e74705SXin Li // Attach the AST reader to the AST context as an external AST
722*67e74705SXin Li // source, so that declarations will be deserialized from the
723*67e74705SXin Li // AST file as needed.
724*67e74705SXin Li // We need the external source to be set up before we read the AST, because
725*67e74705SXin Li // eagerly-deserialized declarations may use it.
726*67e74705SXin Li Context.setExternalSource(AST->Reader);
727*67e74705SXin Li
728*67e74705SXin Li switch (AST->Reader->ReadAST(Filename, serialization::MK_MainFile,
729*67e74705SXin Li SourceLocation(), ASTReader::ARR_None)) {
730*67e74705SXin Li case ASTReader::Success:
731*67e74705SXin Li break;
732*67e74705SXin Li
733*67e74705SXin Li case ASTReader::Failure:
734*67e74705SXin Li case ASTReader::Missing:
735*67e74705SXin Li case ASTReader::OutOfDate:
736*67e74705SXin Li case ASTReader::VersionMismatch:
737*67e74705SXin Li case ASTReader::ConfigurationMismatch:
738*67e74705SXin Li case ASTReader::HadErrors:
739*67e74705SXin Li AST->getDiagnostics().Report(diag::err_fe_unable_to_load_pch);
740*67e74705SXin Li return nullptr;
741*67e74705SXin Li }
742*67e74705SXin Li
743*67e74705SXin Li AST->OriginalSourceFile = AST->Reader->getOriginalSourceFile();
744*67e74705SXin Li
745*67e74705SXin Li PP.setCounterValue(Counter);
746*67e74705SXin Li
747*67e74705SXin Li // Create an AST consumer, even though it isn't used.
748*67e74705SXin Li AST->Consumer.reset(new ASTConsumer);
749*67e74705SXin Li
750*67e74705SXin Li // Create a semantic analysis object and tell the AST reader about it.
751*67e74705SXin Li AST->TheSema.reset(new Sema(PP, Context, *AST->Consumer));
752*67e74705SXin Li AST->TheSema->Initialize();
753*67e74705SXin Li AST->Reader->InitializeSema(*AST->TheSema);
754*67e74705SXin Li
755*67e74705SXin Li // Tell the diagnostic client that we have started a source file.
756*67e74705SXin Li AST->getDiagnostics().getClient()->BeginSourceFile(Context.getLangOpts(),&PP);
757*67e74705SXin Li
758*67e74705SXin Li return AST;
759*67e74705SXin Li }
760*67e74705SXin Li
761*67e74705SXin Li namespace {
762*67e74705SXin Li
763*67e74705SXin Li /// \brief Preprocessor callback class that updates a hash value with the names
764*67e74705SXin Li /// of all macros that have been defined by the translation unit.
765*67e74705SXin Li class MacroDefinitionTrackerPPCallbacks : public PPCallbacks {
766*67e74705SXin Li unsigned &Hash;
767*67e74705SXin Li
768*67e74705SXin Li public:
MacroDefinitionTrackerPPCallbacks(unsigned & Hash)769*67e74705SXin Li explicit MacroDefinitionTrackerPPCallbacks(unsigned &Hash) : Hash(Hash) { }
770*67e74705SXin Li
MacroDefined(const Token & MacroNameTok,const MacroDirective * MD)771*67e74705SXin Li void MacroDefined(const Token &MacroNameTok,
772*67e74705SXin Li const MacroDirective *MD) override {
773*67e74705SXin Li Hash = llvm::HashString(MacroNameTok.getIdentifierInfo()->getName(), Hash);
774*67e74705SXin Li }
775*67e74705SXin Li };
776*67e74705SXin Li
777*67e74705SXin Li /// \brief Add the given declaration to the hash of all top-level entities.
AddTopLevelDeclarationToHash(Decl * D,unsigned & Hash)778*67e74705SXin Li void AddTopLevelDeclarationToHash(Decl *D, unsigned &Hash) {
779*67e74705SXin Li if (!D)
780*67e74705SXin Li return;
781*67e74705SXin Li
782*67e74705SXin Li DeclContext *DC = D->getDeclContext();
783*67e74705SXin Li if (!DC)
784*67e74705SXin Li return;
785*67e74705SXin Li
786*67e74705SXin Li if (!(DC->isTranslationUnit() || DC->getLookupParent()->isTranslationUnit()))
787*67e74705SXin Li return;
788*67e74705SXin Li
789*67e74705SXin Li if (NamedDecl *ND = dyn_cast<NamedDecl>(D)) {
790*67e74705SXin Li if (EnumDecl *EnumD = dyn_cast<EnumDecl>(D)) {
791*67e74705SXin Li // For an unscoped enum include the enumerators in the hash since they
792*67e74705SXin Li // enter the top-level namespace.
793*67e74705SXin Li if (!EnumD->isScoped()) {
794*67e74705SXin Li for (const auto *EI : EnumD->enumerators()) {
795*67e74705SXin Li if (EI->getIdentifier())
796*67e74705SXin Li Hash = llvm::HashString(EI->getIdentifier()->getName(), Hash);
797*67e74705SXin Li }
798*67e74705SXin Li }
799*67e74705SXin Li }
800*67e74705SXin Li
801*67e74705SXin Li if (ND->getIdentifier())
802*67e74705SXin Li Hash = llvm::HashString(ND->getIdentifier()->getName(), Hash);
803*67e74705SXin Li else if (DeclarationName Name = ND->getDeclName()) {
804*67e74705SXin Li std::string NameStr = Name.getAsString();
805*67e74705SXin Li Hash = llvm::HashString(NameStr, Hash);
806*67e74705SXin Li }
807*67e74705SXin Li return;
808*67e74705SXin Li }
809*67e74705SXin Li
810*67e74705SXin Li if (ImportDecl *ImportD = dyn_cast<ImportDecl>(D)) {
811*67e74705SXin Li if (Module *Mod = ImportD->getImportedModule()) {
812*67e74705SXin Li std::string ModName = Mod->getFullModuleName();
813*67e74705SXin Li Hash = llvm::HashString(ModName, Hash);
814*67e74705SXin Li }
815*67e74705SXin Li return;
816*67e74705SXin Li }
817*67e74705SXin Li }
818*67e74705SXin Li
819*67e74705SXin Li class TopLevelDeclTrackerConsumer : public ASTConsumer {
820*67e74705SXin Li ASTUnit &Unit;
821*67e74705SXin Li unsigned &Hash;
822*67e74705SXin Li
823*67e74705SXin Li public:
TopLevelDeclTrackerConsumer(ASTUnit & _Unit,unsigned & Hash)824*67e74705SXin Li TopLevelDeclTrackerConsumer(ASTUnit &_Unit, unsigned &Hash)
825*67e74705SXin Li : Unit(_Unit), Hash(Hash) {
826*67e74705SXin Li Hash = 0;
827*67e74705SXin Li }
828*67e74705SXin Li
handleTopLevelDecl(Decl * D)829*67e74705SXin Li void handleTopLevelDecl(Decl *D) {
830*67e74705SXin Li if (!D)
831*67e74705SXin Li return;
832*67e74705SXin Li
833*67e74705SXin Li // FIXME: Currently ObjC method declarations are incorrectly being
834*67e74705SXin Li // reported as top-level declarations, even though their DeclContext
835*67e74705SXin Li // is the containing ObjC @interface/@implementation. This is a
836*67e74705SXin Li // fundamental problem in the parser right now.
837*67e74705SXin Li if (isa<ObjCMethodDecl>(D))
838*67e74705SXin Li return;
839*67e74705SXin Li
840*67e74705SXin Li AddTopLevelDeclarationToHash(D, Hash);
841*67e74705SXin Li Unit.addTopLevelDecl(D);
842*67e74705SXin Li
843*67e74705SXin Li handleFileLevelDecl(D);
844*67e74705SXin Li }
845*67e74705SXin Li
handleFileLevelDecl(Decl * D)846*67e74705SXin Li void handleFileLevelDecl(Decl *D) {
847*67e74705SXin Li Unit.addFileLevelDecl(D);
848*67e74705SXin Li if (NamespaceDecl *NSD = dyn_cast<NamespaceDecl>(D)) {
849*67e74705SXin Li for (auto *I : NSD->decls())
850*67e74705SXin Li handleFileLevelDecl(I);
851*67e74705SXin Li }
852*67e74705SXin Li }
853*67e74705SXin Li
HandleTopLevelDecl(DeclGroupRef D)854*67e74705SXin Li bool HandleTopLevelDecl(DeclGroupRef D) override {
855*67e74705SXin Li for (Decl *TopLevelDecl : D)
856*67e74705SXin Li handleTopLevelDecl(TopLevelDecl);
857*67e74705SXin Li return true;
858*67e74705SXin Li }
859*67e74705SXin Li
860*67e74705SXin Li // We're not interested in "interesting" decls.
HandleInterestingDecl(DeclGroupRef)861*67e74705SXin Li void HandleInterestingDecl(DeclGroupRef) override {}
862*67e74705SXin Li
HandleTopLevelDeclInObjCContainer(DeclGroupRef D)863*67e74705SXin Li void HandleTopLevelDeclInObjCContainer(DeclGroupRef D) override {
864*67e74705SXin Li for (Decl *TopLevelDecl : D)
865*67e74705SXin Li handleTopLevelDecl(TopLevelDecl);
866*67e74705SXin Li }
867*67e74705SXin Li
GetASTMutationListener()868*67e74705SXin Li ASTMutationListener *GetASTMutationListener() override {
869*67e74705SXin Li return Unit.getASTMutationListener();
870*67e74705SXin Li }
871*67e74705SXin Li
GetASTDeserializationListener()872*67e74705SXin Li ASTDeserializationListener *GetASTDeserializationListener() override {
873*67e74705SXin Li return Unit.getDeserializationListener();
874*67e74705SXin Li }
875*67e74705SXin Li };
876*67e74705SXin Li
877*67e74705SXin Li class TopLevelDeclTrackerAction : public ASTFrontendAction {
878*67e74705SXin Li public:
879*67e74705SXin Li ASTUnit &Unit;
880*67e74705SXin Li
CreateASTConsumer(CompilerInstance & CI,StringRef InFile)881*67e74705SXin Li std::unique_ptr<ASTConsumer> CreateASTConsumer(CompilerInstance &CI,
882*67e74705SXin Li StringRef InFile) override {
883*67e74705SXin Li CI.getPreprocessor().addPPCallbacks(
884*67e74705SXin Li llvm::make_unique<MacroDefinitionTrackerPPCallbacks>(
885*67e74705SXin Li Unit.getCurrentTopLevelHashValue()));
886*67e74705SXin Li return llvm::make_unique<TopLevelDeclTrackerConsumer>(
887*67e74705SXin Li Unit, Unit.getCurrentTopLevelHashValue());
888*67e74705SXin Li }
889*67e74705SXin Li
890*67e74705SXin Li public:
TopLevelDeclTrackerAction(ASTUnit & _Unit)891*67e74705SXin Li TopLevelDeclTrackerAction(ASTUnit &_Unit) : Unit(_Unit) {}
892*67e74705SXin Li
hasCodeCompletionSupport() const893*67e74705SXin Li bool hasCodeCompletionSupport() const override { return false; }
getTranslationUnitKind()894*67e74705SXin Li TranslationUnitKind getTranslationUnitKind() override {
895*67e74705SXin Li return Unit.getTranslationUnitKind();
896*67e74705SXin Li }
897*67e74705SXin Li };
898*67e74705SXin Li
899*67e74705SXin Li class PrecompilePreambleAction : public ASTFrontendAction {
900*67e74705SXin Li ASTUnit &Unit;
901*67e74705SXin Li bool HasEmittedPreamblePCH;
902*67e74705SXin Li
903*67e74705SXin Li public:
PrecompilePreambleAction(ASTUnit & Unit)904*67e74705SXin Li explicit PrecompilePreambleAction(ASTUnit &Unit)
905*67e74705SXin Li : Unit(Unit), HasEmittedPreamblePCH(false) {}
906*67e74705SXin Li
907*67e74705SXin Li std::unique_ptr<ASTConsumer> CreateASTConsumer(CompilerInstance &CI,
908*67e74705SXin Li StringRef InFile) override;
hasEmittedPreamblePCH() const909*67e74705SXin Li bool hasEmittedPreamblePCH() const { return HasEmittedPreamblePCH; }
setHasEmittedPreamblePCH()910*67e74705SXin Li void setHasEmittedPreamblePCH() { HasEmittedPreamblePCH = true; }
shouldEraseOutputFiles()911*67e74705SXin Li bool shouldEraseOutputFiles() override { return !hasEmittedPreamblePCH(); }
912*67e74705SXin Li
hasCodeCompletionSupport() const913*67e74705SXin Li bool hasCodeCompletionSupport() const override { return false; }
hasASTFileSupport() const914*67e74705SXin Li bool hasASTFileSupport() const override { return false; }
getTranslationUnitKind()915*67e74705SXin Li TranslationUnitKind getTranslationUnitKind() override { return TU_Prefix; }
916*67e74705SXin Li };
917*67e74705SXin Li
918*67e74705SXin Li class PrecompilePreambleConsumer : public PCHGenerator {
919*67e74705SXin Li ASTUnit &Unit;
920*67e74705SXin Li unsigned &Hash;
921*67e74705SXin Li std::vector<Decl *> TopLevelDecls;
922*67e74705SXin Li PrecompilePreambleAction *Action;
923*67e74705SXin Li raw_ostream *Out;
924*67e74705SXin Li
925*67e74705SXin Li public:
PrecompilePreambleConsumer(ASTUnit & Unit,PrecompilePreambleAction * Action,const Preprocessor & PP,StringRef isysroot,raw_ostream * Out)926*67e74705SXin Li PrecompilePreambleConsumer(ASTUnit &Unit, PrecompilePreambleAction *Action,
927*67e74705SXin Li const Preprocessor &PP, StringRef isysroot,
928*67e74705SXin Li raw_ostream *Out)
929*67e74705SXin Li : PCHGenerator(PP, "", nullptr, isysroot, std::make_shared<PCHBuffer>(),
930*67e74705SXin Li ArrayRef<llvm::IntrusiveRefCntPtr<ModuleFileExtension>>(),
931*67e74705SXin Li /*AllowASTWithErrors=*/true),
932*67e74705SXin Li Unit(Unit), Hash(Unit.getCurrentTopLevelHashValue()), Action(Action),
933*67e74705SXin Li Out(Out) {
934*67e74705SXin Li Hash = 0;
935*67e74705SXin Li }
936*67e74705SXin Li
HandleTopLevelDecl(DeclGroupRef DG)937*67e74705SXin Li bool HandleTopLevelDecl(DeclGroupRef DG) override {
938*67e74705SXin Li for (Decl *D : DG) {
939*67e74705SXin Li // FIXME: Currently ObjC method declarations are incorrectly being
940*67e74705SXin Li // reported as top-level declarations, even though their DeclContext
941*67e74705SXin Li // is the containing ObjC @interface/@implementation. This is a
942*67e74705SXin Li // fundamental problem in the parser right now.
943*67e74705SXin Li if (isa<ObjCMethodDecl>(D))
944*67e74705SXin Li continue;
945*67e74705SXin Li AddTopLevelDeclarationToHash(D, Hash);
946*67e74705SXin Li TopLevelDecls.push_back(D);
947*67e74705SXin Li }
948*67e74705SXin Li return true;
949*67e74705SXin Li }
950*67e74705SXin Li
HandleTranslationUnit(ASTContext & Ctx)951*67e74705SXin Li void HandleTranslationUnit(ASTContext &Ctx) override {
952*67e74705SXin Li PCHGenerator::HandleTranslationUnit(Ctx);
953*67e74705SXin Li if (hasEmittedPCH()) {
954*67e74705SXin Li // Write the generated bitstream to "Out".
955*67e74705SXin Li *Out << getPCH();
956*67e74705SXin Li // Make sure it hits disk now.
957*67e74705SXin Li Out->flush();
958*67e74705SXin Li // Free the buffer.
959*67e74705SXin Li llvm::SmallVector<char, 0> Empty;
960*67e74705SXin Li getPCH() = std::move(Empty);
961*67e74705SXin Li
962*67e74705SXin Li // Translate the top-level declarations we captured during
963*67e74705SXin Li // parsing into declaration IDs in the precompiled
964*67e74705SXin Li // preamble. This will allow us to deserialize those top-level
965*67e74705SXin Li // declarations when requested.
966*67e74705SXin Li for (Decl *D : TopLevelDecls) {
967*67e74705SXin Li // Invalid top-level decls may not have been serialized.
968*67e74705SXin Li if (D->isInvalidDecl())
969*67e74705SXin Li continue;
970*67e74705SXin Li Unit.addTopLevelDeclFromPreamble(getWriter().getDeclID(D));
971*67e74705SXin Li }
972*67e74705SXin Li
973*67e74705SXin Li Action->setHasEmittedPreamblePCH();
974*67e74705SXin Li }
975*67e74705SXin Li }
976*67e74705SXin Li };
977*67e74705SXin Li
978*67e74705SXin Li } // anonymous namespace
979*67e74705SXin Li
980*67e74705SXin Li std::unique_ptr<ASTConsumer>
CreateASTConsumer(CompilerInstance & CI,StringRef InFile)981*67e74705SXin Li PrecompilePreambleAction::CreateASTConsumer(CompilerInstance &CI,
982*67e74705SXin Li StringRef InFile) {
983*67e74705SXin Li std::string Sysroot;
984*67e74705SXin Li std::string OutputFile;
985*67e74705SXin Li raw_ostream *OS = GeneratePCHAction::ComputeASTConsumerArguments(
986*67e74705SXin Li CI, InFile, Sysroot, OutputFile);
987*67e74705SXin Li if (!OS)
988*67e74705SXin Li return nullptr;
989*67e74705SXin Li
990*67e74705SXin Li if (!CI.getFrontendOpts().RelocatablePCH)
991*67e74705SXin Li Sysroot.clear();
992*67e74705SXin Li
993*67e74705SXin Li CI.getPreprocessor().addPPCallbacks(
994*67e74705SXin Li llvm::make_unique<MacroDefinitionTrackerPPCallbacks>(
995*67e74705SXin Li Unit.getCurrentTopLevelHashValue()));
996*67e74705SXin Li return llvm::make_unique<PrecompilePreambleConsumer>(
997*67e74705SXin Li Unit, this, CI.getPreprocessor(), Sysroot, OS);
998*67e74705SXin Li }
999*67e74705SXin Li
isNonDriverDiag(const StoredDiagnostic & StoredDiag)1000*67e74705SXin Li static bool isNonDriverDiag(const StoredDiagnostic &StoredDiag) {
1001*67e74705SXin Li return StoredDiag.getLocation().isValid();
1002*67e74705SXin Li }
1003*67e74705SXin Li
1004*67e74705SXin Li static void
checkAndRemoveNonDriverDiags(SmallVectorImpl<StoredDiagnostic> & StoredDiags)1005*67e74705SXin Li checkAndRemoveNonDriverDiags(SmallVectorImpl<StoredDiagnostic> &StoredDiags) {
1006*67e74705SXin Li // Get rid of stored diagnostics except the ones from the driver which do not
1007*67e74705SXin Li // have a source location.
1008*67e74705SXin Li StoredDiags.erase(
1009*67e74705SXin Li std::remove_if(StoredDiags.begin(), StoredDiags.end(), isNonDriverDiag),
1010*67e74705SXin Li StoredDiags.end());
1011*67e74705SXin Li }
1012*67e74705SXin Li
checkAndSanitizeDiags(SmallVectorImpl<StoredDiagnostic> & StoredDiagnostics,SourceManager & SM)1013*67e74705SXin Li static void checkAndSanitizeDiags(SmallVectorImpl<StoredDiagnostic> &
1014*67e74705SXin Li StoredDiagnostics,
1015*67e74705SXin Li SourceManager &SM) {
1016*67e74705SXin Li // The stored diagnostic has the old source manager in it; update
1017*67e74705SXin Li // the locations to refer into the new source manager. Since we've
1018*67e74705SXin Li // been careful to make sure that the source manager's state
1019*67e74705SXin Li // before and after are identical, so that we can reuse the source
1020*67e74705SXin Li // location itself.
1021*67e74705SXin Li for (StoredDiagnostic &SD : StoredDiagnostics) {
1022*67e74705SXin Li if (SD.getLocation().isValid()) {
1023*67e74705SXin Li FullSourceLoc Loc(SD.getLocation(), SM);
1024*67e74705SXin Li SD.setLocation(Loc);
1025*67e74705SXin Li }
1026*67e74705SXin Li }
1027*67e74705SXin Li }
1028*67e74705SXin Li
1029*67e74705SXin Li /// Parse the source file into a translation unit using the given compiler
1030*67e74705SXin Li /// invocation, replacing the current translation unit.
1031*67e74705SXin Li ///
1032*67e74705SXin Li /// \returns True if a failure occurred that causes the ASTUnit not to
1033*67e74705SXin Li /// contain any translation-unit information, false otherwise.
Parse(std::shared_ptr<PCHContainerOperations> PCHContainerOps,std::unique_ptr<llvm::MemoryBuffer> OverrideMainBuffer)1034*67e74705SXin Li bool ASTUnit::Parse(std::shared_ptr<PCHContainerOperations> PCHContainerOps,
1035*67e74705SXin Li std::unique_ptr<llvm::MemoryBuffer> OverrideMainBuffer) {
1036*67e74705SXin Li SavedMainFileBuffer.reset();
1037*67e74705SXin Li
1038*67e74705SXin Li if (!Invocation)
1039*67e74705SXin Li return true;
1040*67e74705SXin Li
1041*67e74705SXin Li // Create the compiler instance to use for building the AST.
1042*67e74705SXin Li std::unique_ptr<CompilerInstance> Clang(
1043*67e74705SXin Li new CompilerInstance(std::move(PCHContainerOps)));
1044*67e74705SXin Li
1045*67e74705SXin Li // Recover resources if we crash before exiting this method.
1046*67e74705SXin Li llvm::CrashRecoveryContextCleanupRegistrar<CompilerInstance>
1047*67e74705SXin Li CICleanup(Clang.get());
1048*67e74705SXin Li
1049*67e74705SXin Li IntrusiveRefCntPtr<CompilerInvocation>
1050*67e74705SXin Li CCInvocation(new CompilerInvocation(*Invocation));
1051*67e74705SXin Li
1052*67e74705SXin Li Clang->setInvocation(CCInvocation.get());
1053*67e74705SXin Li OriginalSourceFile = Clang->getFrontendOpts().Inputs[0].getFile();
1054*67e74705SXin Li
1055*67e74705SXin Li // Set up diagnostics, capturing any diagnostics that would
1056*67e74705SXin Li // otherwise be dropped.
1057*67e74705SXin Li Clang->setDiagnostics(&getDiagnostics());
1058*67e74705SXin Li
1059*67e74705SXin Li // Create the target instance.
1060*67e74705SXin Li Clang->setTarget(TargetInfo::CreateTargetInfo(
1061*67e74705SXin Li Clang->getDiagnostics(), Clang->getInvocation().TargetOpts));
1062*67e74705SXin Li if (!Clang->hasTarget())
1063*67e74705SXin Li return true;
1064*67e74705SXin Li
1065*67e74705SXin Li // Inform the target of the language options.
1066*67e74705SXin Li //
1067*67e74705SXin Li // FIXME: We shouldn't need to do this, the target should be immutable once
1068*67e74705SXin Li // created. This complexity should be lifted elsewhere.
1069*67e74705SXin Li Clang->getTarget().adjust(Clang->getLangOpts());
1070*67e74705SXin Li
1071*67e74705SXin Li assert(Clang->getFrontendOpts().Inputs.size() == 1 &&
1072*67e74705SXin Li "Invocation must have exactly one source file!");
1073*67e74705SXin Li assert(Clang->getFrontendOpts().Inputs[0].getKind() != IK_AST &&
1074*67e74705SXin Li "FIXME: AST inputs not yet supported here!");
1075*67e74705SXin Li assert(Clang->getFrontendOpts().Inputs[0].getKind() != IK_LLVM_IR &&
1076*67e74705SXin Li "IR inputs not support here!");
1077*67e74705SXin Li
1078*67e74705SXin Li // Configure the various subsystems.
1079*67e74705SXin Li LangOpts = Clang->getInvocation().LangOpts;
1080*67e74705SXin Li FileSystemOpts = Clang->getFileSystemOpts();
1081*67e74705SXin Li if (!FileMgr) {
1082*67e74705SXin Li Clang->createFileManager();
1083*67e74705SXin Li FileMgr = &Clang->getFileManager();
1084*67e74705SXin Li }
1085*67e74705SXin Li SourceMgr = new SourceManager(getDiagnostics(), *FileMgr,
1086*67e74705SXin Li UserFilesAreVolatile);
1087*67e74705SXin Li TheSema.reset();
1088*67e74705SXin Li Ctx = nullptr;
1089*67e74705SXin Li PP = nullptr;
1090*67e74705SXin Li Reader = nullptr;
1091*67e74705SXin Li
1092*67e74705SXin Li // Clear out old caches and data.
1093*67e74705SXin Li TopLevelDecls.clear();
1094*67e74705SXin Li clearFileLevelDecls();
1095*67e74705SXin Li CleanTemporaryFiles();
1096*67e74705SXin Li
1097*67e74705SXin Li if (!OverrideMainBuffer) {
1098*67e74705SXin Li checkAndRemoveNonDriverDiags(StoredDiagnostics);
1099*67e74705SXin Li TopLevelDeclsInPreamble.clear();
1100*67e74705SXin Li }
1101*67e74705SXin Li
1102*67e74705SXin Li // Create a file manager object to provide access to and cache the filesystem.
1103*67e74705SXin Li Clang->setFileManager(&getFileManager());
1104*67e74705SXin Li
1105*67e74705SXin Li // Create the source manager.
1106*67e74705SXin Li Clang->setSourceManager(&getSourceManager());
1107*67e74705SXin Li
1108*67e74705SXin Li // If the main file has been overridden due to the use of a preamble,
1109*67e74705SXin Li // make that override happen and introduce the preamble.
1110*67e74705SXin Li PreprocessorOptions &PreprocessorOpts = Clang->getPreprocessorOpts();
1111*67e74705SXin Li if (OverrideMainBuffer) {
1112*67e74705SXin Li PreprocessorOpts.addRemappedFile(OriginalSourceFile,
1113*67e74705SXin Li OverrideMainBuffer.get());
1114*67e74705SXin Li PreprocessorOpts.PrecompiledPreambleBytes.first = Preamble.size();
1115*67e74705SXin Li PreprocessorOpts.PrecompiledPreambleBytes.second
1116*67e74705SXin Li = PreambleEndsAtStartOfLine;
1117*67e74705SXin Li PreprocessorOpts.ImplicitPCHInclude = getPreambleFile(this);
1118*67e74705SXin Li PreprocessorOpts.DisablePCHValidation = true;
1119*67e74705SXin Li
1120*67e74705SXin Li // The stored diagnostic has the old source manager in it; update
1121*67e74705SXin Li // the locations to refer into the new source manager. Since we've
1122*67e74705SXin Li // been careful to make sure that the source manager's state
1123*67e74705SXin Li // before and after are identical, so that we can reuse the source
1124*67e74705SXin Li // location itself.
1125*67e74705SXin Li checkAndSanitizeDiags(StoredDiagnostics, getSourceManager());
1126*67e74705SXin Li
1127*67e74705SXin Li // Keep track of the override buffer;
1128*67e74705SXin Li SavedMainFileBuffer = std::move(OverrideMainBuffer);
1129*67e74705SXin Li }
1130*67e74705SXin Li
1131*67e74705SXin Li std::unique_ptr<TopLevelDeclTrackerAction> Act(
1132*67e74705SXin Li new TopLevelDeclTrackerAction(*this));
1133*67e74705SXin Li
1134*67e74705SXin Li // Recover resources if we crash before exiting this method.
1135*67e74705SXin Li llvm::CrashRecoveryContextCleanupRegistrar<TopLevelDeclTrackerAction>
1136*67e74705SXin Li ActCleanup(Act.get());
1137*67e74705SXin Li
1138*67e74705SXin Li if (!Act->BeginSourceFile(*Clang.get(), Clang->getFrontendOpts().Inputs[0]))
1139*67e74705SXin Li goto error;
1140*67e74705SXin Li
1141*67e74705SXin Li if (SavedMainFileBuffer)
1142*67e74705SXin Li TranslateStoredDiagnostics(getFileManager(), getSourceManager(),
1143*67e74705SXin Li PreambleDiagnostics, StoredDiagnostics);
1144*67e74705SXin Li
1145*67e74705SXin Li if (!Act->Execute())
1146*67e74705SXin Li goto error;
1147*67e74705SXin Li
1148*67e74705SXin Li transferASTDataFromCompilerInstance(*Clang);
1149*67e74705SXin Li
1150*67e74705SXin Li Act->EndSourceFile();
1151*67e74705SXin Li
1152*67e74705SXin Li FailedParseDiagnostics.clear();
1153*67e74705SXin Li
1154*67e74705SXin Li return false;
1155*67e74705SXin Li
1156*67e74705SXin Li error:
1157*67e74705SXin Li // Remove the overridden buffer we used for the preamble.
1158*67e74705SXin Li SavedMainFileBuffer = nullptr;
1159*67e74705SXin Li
1160*67e74705SXin Li // Keep the ownership of the data in the ASTUnit because the client may
1161*67e74705SXin Li // want to see the diagnostics.
1162*67e74705SXin Li transferASTDataFromCompilerInstance(*Clang);
1163*67e74705SXin Li FailedParseDiagnostics.swap(StoredDiagnostics);
1164*67e74705SXin Li StoredDiagnostics.clear();
1165*67e74705SXin Li NumStoredDiagnosticsFromDriver = 0;
1166*67e74705SXin Li return true;
1167*67e74705SXin Li }
1168*67e74705SXin Li
1169*67e74705SXin Li /// \brief Simple function to retrieve a path for a preamble precompiled header.
GetPreamblePCHPath()1170*67e74705SXin Li static std::string GetPreamblePCHPath() {
1171*67e74705SXin Li // FIXME: This is a hack so that we can override the preamble file during
1172*67e74705SXin Li // crash-recovery testing, which is the only case where the preamble files
1173*67e74705SXin Li // are not necessarily cleaned up.
1174*67e74705SXin Li const char *TmpFile = ::getenv("CINDEXTEST_PREAMBLE_FILE");
1175*67e74705SXin Li if (TmpFile)
1176*67e74705SXin Li return TmpFile;
1177*67e74705SXin Li
1178*67e74705SXin Li SmallString<128> Path;
1179*67e74705SXin Li llvm::sys::fs::createTemporaryFile("preamble", "pch", Path);
1180*67e74705SXin Li
1181*67e74705SXin Li return Path.str();
1182*67e74705SXin Li }
1183*67e74705SXin Li
1184*67e74705SXin Li /// \brief Compute the preamble for the main file, providing the source buffer
1185*67e74705SXin Li /// that corresponds to the main file along with a pair (bytes, start-of-line)
1186*67e74705SXin Li /// that describes the preamble.
1187*67e74705SXin Li ASTUnit::ComputedPreamble
ComputePreamble(CompilerInvocation & Invocation,unsigned MaxLines)1188*67e74705SXin Li ASTUnit::ComputePreamble(CompilerInvocation &Invocation, unsigned MaxLines) {
1189*67e74705SXin Li FrontendOptions &FrontendOpts = Invocation.getFrontendOpts();
1190*67e74705SXin Li PreprocessorOptions &PreprocessorOpts = Invocation.getPreprocessorOpts();
1191*67e74705SXin Li
1192*67e74705SXin Li // Try to determine if the main file has been remapped, either from the
1193*67e74705SXin Li // command line (to another file) or directly through the compiler invocation
1194*67e74705SXin Li // (to a memory buffer).
1195*67e74705SXin Li llvm::MemoryBuffer *Buffer = nullptr;
1196*67e74705SXin Li std::unique_ptr<llvm::MemoryBuffer> BufferOwner;
1197*67e74705SXin Li std::string MainFilePath(FrontendOpts.Inputs[0].getFile());
1198*67e74705SXin Li llvm::sys::fs::UniqueID MainFileID;
1199*67e74705SXin Li if (!llvm::sys::fs::getUniqueID(MainFilePath, MainFileID)) {
1200*67e74705SXin Li // Check whether there is a file-file remapping of the main file
1201*67e74705SXin Li for (const auto &RF : PreprocessorOpts.RemappedFiles) {
1202*67e74705SXin Li std::string MPath(RF.first);
1203*67e74705SXin Li llvm::sys::fs::UniqueID MID;
1204*67e74705SXin Li if (!llvm::sys::fs::getUniqueID(MPath, MID)) {
1205*67e74705SXin Li if (MainFileID == MID) {
1206*67e74705SXin Li // We found a remapping. Try to load the resulting, remapped source.
1207*67e74705SXin Li BufferOwner = getBufferForFile(RF.second);
1208*67e74705SXin Li if (!BufferOwner)
1209*67e74705SXin Li return ComputedPreamble(nullptr, nullptr, 0, true);
1210*67e74705SXin Li }
1211*67e74705SXin Li }
1212*67e74705SXin Li }
1213*67e74705SXin Li
1214*67e74705SXin Li // Check whether there is a file-buffer remapping. It supercedes the
1215*67e74705SXin Li // file-file remapping.
1216*67e74705SXin Li for (const auto &RB : PreprocessorOpts.RemappedFileBuffers) {
1217*67e74705SXin Li std::string MPath(RB.first);
1218*67e74705SXin Li llvm::sys::fs::UniqueID MID;
1219*67e74705SXin Li if (!llvm::sys::fs::getUniqueID(MPath, MID)) {
1220*67e74705SXin Li if (MainFileID == MID) {
1221*67e74705SXin Li // We found a remapping.
1222*67e74705SXin Li BufferOwner.reset();
1223*67e74705SXin Li Buffer = const_cast<llvm::MemoryBuffer *>(RB.second);
1224*67e74705SXin Li }
1225*67e74705SXin Li }
1226*67e74705SXin Li }
1227*67e74705SXin Li }
1228*67e74705SXin Li
1229*67e74705SXin Li // If the main source file was not remapped, load it now.
1230*67e74705SXin Li if (!Buffer && !BufferOwner) {
1231*67e74705SXin Li BufferOwner = getBufferForFile(FrontendOpts.Inputs[0].getFile());
1232*67e74705SXin Li if (!BufferOwner)
1233*67e74705SXin Li return ComputedPreamble(nullptr, nullptr, 0, true);
1234*67e74705SXin Li }
1235*67e74705SXin Li
1236*67e74705SXin Li if (!Buffer)
1237*67e74705SXin Li Buffer = BufferOwner.get();
1238*67e74705SXin Li auto Pre = Lexer::ComputePreamble(Buffer->getBuffer(),
1239*67e74705SXin Li *Invocation.getLangOpts(), MaxLines);
1240*67e74705SXin Li return ComputedPreamble(Buffer, std::move(BufferOwner), Pre.first,
1241*67e74705SXin Li Pre.second);
1242*67e74705SXin Li }
1243*67e74705SXin Li
1244*67e74705SXin Li ASTUnit::PreambleFileHash
createForFile(off_t Size,time_t ModTime)1245*67e74705SXin Li ASTUnit::PreambleFileHash::createForFile(off_t Size, time_t ModTime) {
1246*67e74705SXin Li PreambleFileHash Result;
1247*67e74705SXin Li Result.Size = Size;
1248*67e74705SXin Li Result.ModTime = ModTime;
1249*67e74705SXin Li memset(Result.MD5, 0, sizeof(Result.MD5));
1250*67e74705SXin Li return Result;
1251*67e74705SXin Li }
1252*67e74705SXin Li
createForMemoryBuffer(const llvm::MemoryBuffer * Buffer)1253*67e74705SXin Li ASTUnit::PreambleFileHash ASTUnit::PreambleFileHash::createForMemoryBuffer(
1254*67e74705SXin Li const llvm::MemoryBuffer *Buffer) {
1255*67e74705SXin Li PreambleFileHash Result;
1256*67e74705SXin Li Result.Size = Buffer->getBufferSize();
1257*67e74705SXin Li Result.ModTime = 0;
1258*67e74705SXin Li
1259*67e74705SXin Li llvm::MD5 MD5Ctx;
1260*67e74705SXin Li MD5Ctx.update(Buffer->getBuffer().data());
1261*67e74705SXin Li MD5Ctx.final(Result.MD5);
1262*67e74705SXin Li
1263*67e74705SXin Li return Result;
1264*67e74705SXin Li }
1265*67e74705SXin Li
1266*67e74705SXin Li namespace clang {
operator ==(const ASTUnit::PreambleFileHash & LHS,const ASTUnit::PreambleFileHash & RHS)1267*67e74705SXin Li bool operator==(const ASTUnit::PreambleFileHash &LHS,
1268*67e74705SXin Li const ASTUnit::PreambleFileHash &RHS) {
1269*67e74705SXin Li return LHS.Size == RHS.Size && LHS.ModTime == RHS.ModTime &&
1270*67e74705SXin Li memcmp(LHS.MD5, RHS.MD5, sizeof(LHS.MD5)) == 0;
1271*67e74705SXin Li }
1272*67e74705SXin Li } // namespace clang
1273*67e74705SXin Li
1274*67e74705SXin Li static std::pair<unsigned, unsigned>
makeStandaloneRange(CharSourceRange Range,const SourceManager & SM,const LangOptions & LangOpts)1275*67e74705SXin Li makeStandaloneRange(CharSourceRange Range, const SourceManager &SM,
1276*67e74705SXin Li const LangOptions &LangOpts) {
1277*67e74705SXin Li CharSourceRange FileRange = Lexer::makeFileCharRange(Range, SM, LangOpts);
1278*67e74705SXin Li unsigned Offset = SM.getFileOffset(FileRange.getBegin());
1279*67e74705SXin Li unsigned EndOffset = SM.getFileOffset(FileRange.getEnd());
1280*67e74705SXin Li return std::make_pair(Offset, EndOffset);
1281*67e74705SXin Li }
1282*67e74705SXin Li
makeStandaloneFixIt(const SourceManager & SM,const LangOptions & LangOpts,const FixItHint & InFix)1283*67e74705SXin Li static ASTUnit::StandaloneFixIt makeStandaloneFixIt(const SourceManager &SM,
1284*67e74705SXin Li const LangOptions &LangOpts,
1285*67e74705SXin Li const FixItHint &InFix) {
1286*67e74705SXin Li ASTUnit::StandaloneFixIt OutFix;
1287*67e74705SXin Li OutFix.RemoveRange = makeStandaloneRange(InFix.RemoveRange, SM, LangOpts);
1288*67e74705SXin Li OutFix.InsertFromRange = makeStandaloneRange(InFix.InsertFromRange, SM,
1289*67e74705SXin Li LangOpts);
1290*67e74705SXin Li OutFix.CodeToInsert = InFix.CodeToInsert;
1291*67e74705SXin Li OutFix.BeforePreviousInsertions = InFix.BeforePreviousInsertions;
1292*67e74705SXin Li return OutFix;
1293*67e74705SXin Li }
1294*67e74705SXin Li
1295*67e74705SXin Li static ASTUnit::StandaloneDiagnostic
makeStandaloneDiagnostic(const LangOptions & LangOpts,const StoredDiagnostic & InDiag)1296*67e74705SXin Li makeStandaloneDiagnostic(const LangOptions &LangOpts,
1297*67e74705SXin Li const StoredDiagnostic &InDiag) {
1298*67e74705SXin Li ASTUnit::StandaloneDiagnostic OutDiag;
1299*67e74705SXin Li OutDiag.ID = InDiag.getID();
1300*67e74705SXin Li OutDiag.Level = InDiag.getLevel();
1301*67e74705SXin Li OutDiag.Message = InDiag.getMessage();
1302*67e74705SXin Li OutDiag.LocOffset = 0;
1303*67e74705SXin Li if (InDiag.getLocation().isInvalid())
1304*67e74705SXin Li return OutDiag;
1305*67e74705SXin Li const SourceManager &SM = InDiag.getLocation().getManager();
1306*67e74705SXin Li SourceLocation FileLoc = SM.getFileLoc(InDiag.getLocation());
1307*67e74705SXin Li OutDiag.Filename = SM.getFilename(FileLoc);
1308*67e74705SXin Li if (OutDiag.Filename.empty())
1309*67e74705SXin Li return OutDiag;
1310*67e74705SXin Li OutDiag.LocOffset = SM.getFileOffset(FileLoc);
1311*67e74705SXin Li for (const CharSourceRange &Range : InDiag.getRanges())
1312*67e74705SXin Li OutDiag.Ranges.push_back(makeStandaloneRange(Range, SM, LangOpts));
1313*67e74705SXin Li for (const FixItHint &FixIt : InDiag.getFixIts())
1314*67e74705SXin Li OutDiag.FixIts.push_back(makeStandaloneFixIt(SM, LangOpts, FixIt));
1315*67e74705SXin Li
1316*67e74705SXin Li return OutDiag;
1317*67e74705SXin Li }
1318*67e74705SXin Li
1319*67e74705SXin Li /// \brief Attempt to build or re-use a precompiled preamble when (re-)parsing
1320*67e74705SXin Li /// the source file.
1321*67e74705SXin Li ///
1322*67e74705SXin Li /// This routine will compute the preamble of the main source file. If a
1323*67e74705SXin Li /// non-trivial preamble is found, it will precompile that preamble into a
1324*67e74705SXin Li /// precompiled header so that the precompiled preamble can be used to reduce
1325*67e74705SXin Li /// reparsing time. If a precompiled preamble has already been constructed,
1326*67e74705SXin Li /// this routine will determine if it is still valid and, if so, avoid
1327*67e74705SXin Li /// rebuilding the precompiled preamble.
1328*67e74705SXin Li ///
1329*67e74705SXin Li /// \param AllowRebuild When true (the default), this routine is
1330*67e74705SXin Li /// allowed to rebuild the precompiled preamble if it is found to be
1331*67e74705SXin Li /// out-of-date.
1332*67e74705SXin Li ///
1333*67e74705SXin Li /// \param MaxLines When non-zero, the maximum number of lines that
1334*67e74705SXin Li /// can occur within the preamble.
1335*67e74705SXin Li ///
1336*67e74705SXin Li /// \returns If the precompiled preamble can be used, returns a newly-allocated
1337*67e74705SXin Li /// buffer that should be used in place of the main file when doing so.
1338*67e74705SXin Li /// Otherwise, returns a NULL pointer.
1339*67e74705SXin Li std::unique_ptr<llvm::MemoryBuffer>
getMainBufferWithPrecompiledPreamble(std::shared_ptr<PCHContainerOperations> PCHContainerOps,const CompilerInvocation & PreambleInvocationIn,bool AllowRebuild,unsigned MaxLines)1340*67e74705SXin Li ASTUnit::getMainBufferWithPrecompiledPreamble(
1341*67e74705SXin Li std::shared_ptr<PCHContainerOperations> PCHContainerOps,
1342*67e74705SXin Li const CompilerInvocation &PreambleInvocationIn, bool AllowRebuild,
1343*67e74705SXin Li unsigned MaxLines) {
1344*67e74705SXin Li
1345*67e74705SXin Li IntrusiveRefCntPtr<CompilerInvocation>
1346*67e74705SXin Li PreambleInvocation(new CompilerInvocation(PreambleInvocationIn));
1347*67e74705SXin Li FrontendOptions &FrontendOpts = PreambleInvocation->getFrontendOpts();
1348*67e74705SXin Li PreprocessorOptions &PreprocessorOpts
1349*67e74705SXin Li = PreambleInvocation->getPreprocessorOpts();
1350*67e74705SXin Li
1351*67e74705SXin Li ComputedPreamble NewPreamble = ComputePreamble(*PreambleInvocation, MaxLines);
1352*67e74705SXin Li
1353*67e74705SXin Li if (!NewPreamble.Size) {
1354*67e74705SXin Li // We couldn't find a preamble in the main source. Clear out the current
1355*67e74705SXin Li // preamble, if we have one. It's obviously no good any more.
1356*67e74705SXin Li Preamble.clear();
1357*67e74705SXin Li erasePreambleFile(this);
1358*67e74705SXin Li
1359*67e74705SXin Li // The next time we actually see a preamble, precompile it.
1360*67e74705SXin Li PreambleRebuildCounter = 1;
1361*67e74705SXin Li return nullptr;
1362*67e74705SXin Li }
1363*67e74705SXin Li
1364*67e74705SXin Li if (!Preamble.empty()) {
1365*67e74705SXin Li // We've previously computed a preamble. Check whether we have the same
1366*67e74705SXin Li // preamble now that we did before, and that there's enough space in
1367*67e74705SXin Li // the main-file buffer within the precompiled preamble to fit the
1368*67e74705SXin Li // new main file.
1369*67e74705SXin Li if (Preamble.size() == NewPreamble.Size &&
1370*67e74705SXin Li PreambleEndsAtStartOfLine == NewPreamble.PreambleEndsAtStartOfLine &&
1371*67e74705SXin Li memcmp(Preamble.getBufferStart(), NewPreamble.Buffer->getBufferStart(),
1372*67e74705SXin Li NewPreamble.Size) == 0) {
1373*67e74705SXin Li // The preamble has not changed. We may be able to re-use the precompiled
1374*67e74705SXin Li // preamble.
1375*67e74705SXin Li
1376*67e74705SXin Li // Check that none of the files used by the preamble have changed.
1377*67e74705SXin Li bool AnyFileChanged = false;
1378*67e74705SXin Li
1379*67e74705SXin Li // First, make a record of those files that have been overridden via
1380*67e74705SXin Li // remapping or unsaved_files.
1381*67e74705SXin Li std::map<llvm::sys::fs::UniqueID, PreambleFileHash> OverriddenFiles;
1382*67e74705SXin Li for (const auto &R : PreprocessorOpts.RemappedFiles) {
1383*67e74705SXin Li if (AnyFileChanged)
1384*67e74705SXin Li break;
1385*67e74705SXin Li
1386*67e74705SXin Li vfs::Status Status;
1387*67e74705SXin Li if (FileMgr->getNoncachedStatValue(R.second, Status)) {
1388*67e74705SXin Li // If we can't stat the file we're remapping to, assume that something
1389*67e74705SXin Li // horrible happened.
1390*67e74705SXin Li AnyFileChanged = true;
1391*67e74705SXin Li break;
1392*67e74705SXin Li }
1393*67e74705SXin Li
1394*67e74705SXin Li OverriddenFiles[Status.getUniqueID()] = PreambleFileHash::createForFile(
1395*67e74705SXin Li Status.getSize(), Status.getLastModificationTime().toEpochTime());
1396*67e74705SXin Li }
1397*67e74705SXin Li
1398*67e74705SXin Li for (const auto &RB : PreprocessorOpts.RemappedFileBuffers) {
1399*67e74705SXin Li if (AnyFileChanged)
1400*67e74705SXin Li break;
1401*67e74705SXin Li
1402*67e74705SXin Li vfs::Status Status;
1403*67e74705SXin Li if (FileMgr->getNoncachedStatValue(RB.first, Status)) {
1404*67e74705SXin Li AnyFileChanged = true;
1405*67e74705SXin Li break;
1406*67e74705SXin Li }
1407*67e74705SXin Li
1408*67e74705SXin Li OverriddenFiles[Status.getUniqueID()] =
1409*67e74705SXin Li PreambleFileHash::createForMemoryBuffer(RB.second);
1410*67e74705SXin Li }
1411*67e74705SXin Li
1412*67e74705SXin Li // Check whether anything has changed.
1413*67e74705SXin Li for (llvm::StringMap<PreambleFileHash>::iterator
1414*67e74705SXin Li F = FilesInPreamble.begin(), FEnd = FilesInPreamble.end();
1415*67e74705SXin Li !AnyFileChanged && F != FEnd;
1416*67e74705SXin Li ++F) {
1417*67e74705SXin Li vfs::Status Status;
1418*67e74705SXin Li if (FileMgr->getNoncachedStatValue(F->first(), Status)) {
1419*67e74705SXin Li // If we can't stat the file, assume that something horrible happened.
1420*67e74705SXin Li AnyFileChanged = true;
1421*67e74705SXin Li break;
1422*67e74705SXin Li }
1423*67e74705SXin Li
1424*67e74705SXin Li std::map<llvm::sys::fs::UniqueID, PreambleFileHash>::iterator Overridden
1425*67e74705SXin Li = OverriddenFiles.find(Status.getUniqueID());
1426*67e74705SXin Li if (Overridden != OverriddenFiles.end()) {
1427*67e74705SXin Li // This file was remapped; check whether the newly-mapped file
1428*67e74705SXin Li // matches up with the previous mapping.
1429*67e74705SXin Li if (Overridden->second != F->second)
1430*67e74705SXin Li AnyFileChanged = true;
1431*67e74705SXin Li continue;
1432*67e74705SXin Li }
1433*67e74705SXin Li
1434*67e74705SXin Li // The file was not remapped; check whether it has changed on disk.
1435*67e74705SXin Li if (Status.getSize() != uint64_t(F->second.Size) ||
1436*67e74705SXin Li Status.getLastModificationTime().toEpochTime() !=
1437*67e74705SXin Li uint64_t(F->second.ModTime))
1438*67e74705SXin Li AnyFileChanged = true;
1439*67e74705SXin Li }
1440*67e74705SXin Li
1441*67e74705SXin Li if (!AnyFileChanged) {
1442*67e74705SXin Li // Okay! We can re-use the precompiled preamble.
1443*67e74705SXin Li
1444*67e74705SXin Li // Set the state of the diagnostic object to mimic its state
1445*67e74705SXin Li // after parsing the preamble.
1446*67e74705SXin Li getDiagnostics().Reset();
1447*67e74705SXin Li ProcessWarningOptions(getDiagnostics(),
1448*67e74705SXin Li PreambleInvocation->getDiagnosticOpts());
1449*67e74705SXin Li getDiagnostics().setNumWarnings(NumWarningsInPreamble);
1450*67e74705SXin Li
1451*67e74705SXin Li return llvm::MemoryBuffer::getMemBufferCopy(
1452*67e74705SXin Li NewPreamble.Buffer->getBuffer(), FrontendOpts.Inputs[0].getFile());
1453*67e74705SXin Li }
1454*67e74705SXin Li }
1455*67e74705SXin Li
1456*67e74705SXin Li // If we aren't allowed to rebuild the precompiled preamble, just
1457*67e74705SXin Li // return now.
1458*67e74705SXin Li if (!AllowRebuild)
1459*67e74705SXin Li return nullptr;
1460*67e74705SXin Li
1461*67e74705SXin Li // We can't reuse the previously-computed preamble. Build a new one.
1462*67e74705SXin Li Preamble.clear();
1463*67e74705SXin Li PreambleDiagnostics.clear();
1464*67e74705SXin Li erasePreambleFile(this);
1465*67e74705SXin Li PreambleRebuildCounter = 1;
1466*67e74705SXin Li } else if (!AllowRebuild) {
1467*67e74705SXin Li // We aren't allowed to rebuild the precompiled preamble; just
1468*67e74705SXin Li // return now.
1469*67e74705SXin Li return nullptr;
1470*67e74705SXin Li }
1471*67e74705SXin Li
1472*67e74705SXin Li // If the preamble rebuild counter > 1, it's because we previously
1473*67e74705SXin Li // failed to build a preamble and we're not yet ready to try
1474*67e74705SXin Li // again. Decrement the counter and return a failure.
1475*67e74705SXin Li if (PreambleRebuildCounter > 1) {
1476*67e74705SXin Li --PreambleRebuildCounter;
1477*67e74705SXin Li return nullptr;
1478*67e74705SXin Li }
1479*67e74705SXin Li
1480*67e74705SXin Li // Create a temporary file for the precompiled preamble. In rare
1481*67e74705SXin Li // circumstances, this can fail.
1482*67e74705SXin Li std::string PreamblePCHPath = GetPreamblePCHPath();
1483*67e74705SXin Li if (PreamblePCHPath.empty()) {
1484*67e74705SXin Li // Try again next time.
1485*67e74705SXin Li PreambleRebuildCounter = 1;
1486*67e74705SXin Li return nullptr;
1487*67e74705SXin Li }
1488*67e74705SXin Li
1489*67e74705SXin Li // We did not previously compute a preamble, or it can't be reused anyway.
1490*67e74705SXin Li SimpleTimer PreambleTimer(WantTiming);
1491*67e74705SXin Li PreambleTimer.setOutput("Precompiling preamble");
1492*67e74705SXin Li
1493*67e74705SXin Li // Save the preamble text for later; we'll need to compare against it for
1494*67e74705SXin Li // subsequent reparses.
1495*67e74705SXin Li StringRef MainFilename = FrontendOpts.Inputs[0].getFile();
1496*67e74705SXin Li Preamble.assign(FileMgr->getFile(MainFilename),
1497*67e74705SXin Li NewPreamble.Buffer->getBufferStart(),
1498*67e74705SXin Li NewPreamble.Buffer->getBufferStart() + NewPreamble.Size);
1499*67e74705SXin Li PreambleEndsAtStartOfLine = NewPreamble.PreambleEndsAtStartOfLine;
1500*67e74705SXin Li
1501*67e74705SXin Li PreambleBuffer = llvm::MemoryBuffer::getMemBufferCopy(
1502*67e74705SXin Li NewPreamble.Buffer->getBuffer().slice(0, Preamble.size()), MainFilename);
1503*67e74705SXin Li
1504*67e74705SXin Li // Remap the main source file to the preamble buffer.
1505*67e74705SXin Li StringRef MainFilePath = FrontendOpts.Inputs[0].getFile();
1506*67e74705SXin Li PreprocessorOpts.addRemappedFile(MainFilePath, PreambleBuffer.get());
1507*67e74705SXin Li
1508*67e74705SXin Li // Tell the compiler invocation to generate a temporary precompiled header.
1509*67e74705SXin Li FrontendOpts.ProgramAction = frontend::GeneratePCH;
1510*67e74705SXin Li // FIXME: Generate the precompiled header into memory?
1511*67e74705SXin Li FrontendOpts.OutputFile = PreamblePCHPath;
1512*67e74705SXin Li PreprocessorOpts.PrecompiledPreambleBytes.first = 0;
1513*67e74705SXin Li PreprocessorOpts.PrecompiledPreambleBytes.second = false;
1514*67e74705SXin Li
1515*67e74705SXin Li // Create the compiler instance to use for building the precompiled preamble.
1516*67e74705SXin Li std::unique_ptr<CompilerInstance> Clang(
1517*67e74705SXin Li new CompilerInstance(std::move(PCHContainerOps)));
1518*67e74705SXin Li
1519*67e74705SXin Li // Recover resources if we crash before exiting this method.
1520*67e74705SXin Li llvm::CrashRecoveryContextCleanupRegistrar<CompilerInstance>
1521*67e74705SXin Li CICleanup(Clang.get());
1522*67e74705SXin Li
1523*67e74705SXin Li Clang->setInvocation(&*PreambleInvocation);
1524*67e74705SXin Li OriginalSourceFile = Clang->getFrontendOpts().Inputs[0].getFile();
1525*67e74705SXin Li
1526*67e74705SXin Li // Set up diagnostics, capturing all of the diagnostics produced.
1527*67e74705SXin Li Clang->setDiagnostics(&getDiagnostics());
1528*67e74705SXin Li
1529*67e74705SXin Li // Create the target instance.
1530*67e74705SXin Li Clang->setTarget(TargetInfo::CreateTargetInfo(
1531*67e74705SXin Li Clang->getDiagnostics(), Clang->getInvocation().TargetOpts));
1532*67e74705SXin Li if (!Clang->hasTarget()) {
1533*67e74705SXin Li llvm::sys::fs::remove(FrontendOpts.OutputFile);
1534*67e74705SXin Li Preamble.clear();
1535*67e74705SXin Li PreambleRebuildCounter = DefaultPreambleRebuildInterval;
1536*67e74705SXin Li PreprocessorOpts.RemappedFileBuffers.pop_back();
1537*67e74705SXin Li return nullptr;
1538*67e74705SXin Li }
1539*67e74705SXin Li
1540*67e74705SXin Li // Inform the target of the language options.
1541*67e74705SXin Li //
1542*67e74705SXin Li // FIXME: We shouldn't need to do this, the target should be immutable once
1543*67e74705SXin Li // created. This complexity should be lifted elsewhere.
1544*67e74705SXin Li Clang->getTarget().adjust(Clang->getLangOpts());
1545*67e74705SXin Li
1546*67e74705SXin Li assert(Clang->getFrontendOpts().Inputs.size() == 1 &&
1547*67e74705SXin Li "Invocation must have exactly one source file!");
1548*67e74705SXin Li assert(Clang->getFrontendOpts().Inputs[0].getKind() != IK_AST &&
1549*67e74705SXin Li "FIXME: AST inputs not yet supported here!");
1550*67e74705SXin Li assert(Clang->getFrontendOpts().Inputs[0].getKind() != IK_LLVM_IR &&
1551*67e74705SXin Li "IR inputs not support here!");
1552*67e74705SXin Li
1553*67e74705SXin Li // Clear out old caches and data.
1554*67e74705SXin Li getDiagnostics().Reset();
1555*67e74705SXin Li ProcessWarningOptions(getDiagnostics(), Clang->getDiagnosticOpts());
1556*67e74705SXin Li checkAndRemoveNonDriverDiags(StoredDiagnostics);
1557*67e74705SXin Li TopLevelDecls.clear();
1558*67e74705SXin Li TopLevelDeclsInPreamble.clear();
1559*67e74705SXin Li PreambleDiagnostics.clear();
1560*67e74705SXin Li
1561*67e74705SXin Li IntrusiveRefCntPtr<vfs::FileSystem> VFS =
1562*67e74705SXin Li createVFSFromCompilerInvocation(Clang->getInvocation(), getDiagnostics());
1563*67e74705SXin Li if (!VFS)
1564*67e74705SXin Li return nullptr;
1565*67e74705SXin Li
1566*67e74705SXin Li // Create a file manager object to provide access to and cache the filesystem.
1567*67e74705SXin Li Clang->setFileManager(new FileManager(Clang->getFileSystemOpts(), VFS));
1568*67e74705SXin Li
1569*67e74705SXin Li // Create the source manager.
1570*67e74705SXin Li Clang->setSourceManager(new SourceManager(getDiagnostics(),
1571*67e74705SXin Li Clang->getFileManager()));
1572*67e74705SXin Li
1573*67e74705SXin Li auto PreambleDepCollector = std::make_shared<DependencyCollector>();
1574*67e74705SXin Li Clang->addDependencyCollector(PreambleDepCollector);
1575*67e74705SXin Li
1576*67e74705SXin Li std::unique_ptr<PrecompilePreambleAction> Act;
1577*67e74705SXin Li Act.reset(new PrecompilePreambleAction(*this));
1578*67e74705SXin Li if (!Act->BeginSourceFile(*Clang.get(), Clang->getFrontendOpts().Inputs[0])) {
1579*67e74705SXin Li llvm::sys::fs::remove(FrontendOpts.OutputFile);
1580*67e74705SXin Li Preamble.clear();
1581*67e74705SXin Li PreambleRebuildCounter = DefaultPreambleRebuildInterval;
1582*67e74705SXin Li PreprocessorOpts.RemappedFileBuffers.pop_back();
1583*67e74705SXin Li return nullptr;
1584*67e74705SXin Li }
1585*67e74705SXin Li
1586*67e74705SXin Li Act->Execute();
1587*67e74705SXin Li
1588*67e74705SXin Li // Transfer any diagnostics generated when parsing the preamble into the set
1589*67e74705SXin Li // of preamble diagnostics.
1590*67e74705SXin Li for (stored_diag_iterator I = stored_diag_afterDriver_begin(),
1591*67e74705SXin Li E = stored_diag_end();
1592*67e74705SXin Li I != E; ++I)
1593*67e74705SXin Li PreambleDiagnostics.push_back(
1594*67e74705SXin Li makeStandaloneDiagnostic(Clang->getLangOpts(), *I));
1595*67e74705SXin Li
1596*67e74705SXin Li Act->EndSourceFile();
1597*67e74705SXin Li
1598*67e74705SXin Li checkAndRemoveNonDriverDiags(StoredDiagnostics);
1599*67e74705SXin Li
1600*67e74705SXin Li if (!Act->hasEmittedPreamblePCH()) {
1601*67e74705SXin Li // The preamble PCH failed (e.g. there was a module loading fatal error),
1602*67e74705SXin Li // so no precompiled header was generated. Forget that we even tried.
1603*67e74705SXin Li // FIXME: Should we leave a note for ourselves to try again?
1604*67e74705SXin Li llvm::sys::fs::remove(FrontendOpts.OutputFile);
1605*67e74705SXin Li Preamble.clear();
1606*67e74705SXin Li TopLevelDeclsInPreamble.clear();
1607*67e74705SXin Li PreambleRebuildCounter = DefaultPreambleRebuildInterval;
1608*67e74705SXin Li PreprocessorOpts.RemappedFileBuffers.pop_back();
1609*67e74705SXin Li return nullptr;
1610*67e74705SXin Li }
1611*67e74705SXin Li
1612*67e74705SXin Li // Keep track of the preamble we precompiled.
1613*67e74705SXin Li setPreambleFile(this, FrontendOpts.OutputFile);
1614*67e74705SXin Li NumWarningsInPreamble = getDiagnostics().getNumWarnings();
1615*67e74705SXin Li
1616*67e74705SXin Li // Keep track of all of the files that the source manager knows about,
1617*67e74705SXin Li // so we can verify whether they have changed or not.
1618*67e74705SXin Li FilesInPreamble.clear();
1619*67e74705SXin Li SourceManager &SourceMgr = Clang->getSourceManager();
1620*67e74705SXin Li for (auto &Filename : PreambleDepCollector->getDependencies()) {
1621*67e74705SXin Li const FileEntry *File = Clang->getFileManager().getFile(Filename);
1622*67e74705SXin Li if (!File || File == SourceMgr.getFileEntryForID(SourceMgr.getMainFileID()))
1623*67e74705SXin Li continue;
1624*67e74705SXin Li if (time_t ModTime = File->getModificationTime()) {
1625*67e74705SXin Li FilesInPreamble[File->getName()] = PreambleFileHash::createForFile(
1626*67e74705SXin Li File->getSize(), ModTime);
1627*67e74705SXin Li } else {
1628*67e74705SXin Li llvm::MemoryBuffer *Buffer = SourceMgr.getMemoryBufferForFile(File);
1629*67e74705SXin Li FilesInPreamble[File->getName()] =
1630*67e74705SXin Li PreambleFileHash::createForMemoryBuffer(Buffer);
1631*67e74705SXin Li }
1632*67e74705SXin Li }
1633*67e74705SXin Li
1634*67e74705SXin Li PreambleRebuildCounter = 1;
1635*67e74705SXin Li PreprocessorOpts.RemappedFileBuffers.pop_back();
1636*67e74705SXin Li
1637*67e74705SXin Li // If the hash of top-level entities differs from the hash of the top-level
1638*67e74705SXin Li // entities the last time we rebuilt the preamble, clear out the completion
1639*67e74705SXin Li // cache.
1640*67e74705SXin Li if (CurrentTopLevelHashValue != PreambleTopLevelHashValue) {
1641*67e74705SXin Li CompletionCacheTopLevelHashValue = 0;
1642*67e74705SXin Li PreambleTopLevelHashValue = CurrentTopLevelHashValue;
1643*67e74705SXin Li }
1644*67e74705SXin Li
1645*67e74705SXin Li return llvm::MemoryBuffer::getMemBufferCopy(NewPreamble.Buffer->getBuffer(),
1646*67e74705SXin Li MainFilename);
1647*67e74705SXin Li }
1648*67e74705SXin Li
RealizeTopLevelDeclsFromPreamble()1649*67e74705SXin Li void ASTUnit::RealizeTopLevelDeclsFromPreamble() {
1650*67e74705SXin Li std::vector<Decl *> Resolved;
1651*67e74705SXin Li Resolved.reserve(TopLevelDeclsInPreamble.size());
1652*67e74705SXin Li ExternalASTSource &Source = *getASTContext().getExternalSource();
1653*67e74705SXin Li for (serialization::DeclID TopLevelDecl : TopLevelDeclsInPreamble) {
1654*67e74705SXin Li // Resolve the declaration ID to an actual declaration, possibly
1655*67e74705SXin Li // deserializing the declaration in the process.
1656*67e74705SXin Li if (Decl *D = Source.GetExternalDecl(TopLevelDecl))
1657*67e74705SXin Li Resolved.push_back(D);
1658*67e74705SXin Li }
1659*67e74705SXin Li TopLevelDeclsInPreamble.clear();
1660*67e74705SXin Li TopLevelDecls.insert(TopLevelDecls.begin(), Resolved.begin(), Resolved.end());
1661*67e74705SXin Li }
1662*67e74705SXin Li
transferASTDataFromCompilerInstance(CompilerInstance & CI)1663*67e74705SXin Li void ASTUnit::transferASTDataFromCompilerInstance(CompilerInstance &CI) {
1664*67e74705SXin Li // Steal the created target, context, and preprocessor if they have been
1665*67e74705SXin Li // created.
1666*67e74705SXin Li assert(CI.hasInvocation() && "missing invocation");
1667*67e74705SXin Li LangOpts = CI.getInvocation().LangOpts;
1668*67e74705SXin Li TheSema = CI.takeSema();
1669*67e74705SXin Li Consumer = CI.takeASTConsumer();
1670*67e74705SXin Li if (CI.hasASTContext())
1671*67e74705SXin Li Ctx = &CI.getASTContext();
1672*67e74705SXin Li if (CI.hasPreprocessor())
1673*67e74705SXin Li PP = &CI.getPreprocessor();
1674*67e74705SXin Li CI.setSourceManager(nullptr);
1675*67e74705SXin Li CI.setFileManager(nullptr);
1676*67e74705SXin Li if (CI.hasTarget())
1677*67e74705SXin Li Target = &CI.getTarget();
1678*67e74705SXin Li Reader = CI.getModuleManager();
1679*67e74705SXin Li HadModuleLoaderFatalFailure = CI.hadModuleLoaderFatalFailure();
1680*67e74705SXin Li }
1681*67e74705SXin Li
getMainFileName() const1682*67e74705SXin Li StringRef ASTUnit::getMainFileName() const {
1683*67e74705SXin Li if (Invocation && !Invocation->getFrontendOpts().Inputs.empty()) {
1684*67e74705SXin Li const FrontendInputFile &Input = Invocation->getFrontendOpts().Inputs[0];
1685*67e74705SXin Li if (Input.isFile())
1686*67e74705SXin Li return Input.getFile();
1687*67e74705SXin Li else
1688*67e74705SXin Li return Input.getBuffer()->getBufferIdentifier();
1689*67e74705SXin Li }
1690*67e74705SXin Li
1691*67e74705SXin Li if (SourceMgr) {
1692*67e74705SXin Li if (const FileEntry *
1693*67e74705SXin Li FE = SourceMgr->getFileEntryForID(SourceMgr->getMainFileID()))
1694*67e74705SXin Li return FE->getName();
1695*67e74705SXin Li }
1696*67e74705SXin Li
1697*67e74705SXin Li return StringRef();
1698*67e74705SXin Li }
1699*67e74705SXin Li
getASTFileName() const1700*67e74705SXin Li StringRef ASTUnit::getASTFileName() const {
1701*67e74705SXin Li if (!isMainFileAST())
1702*67e74705SXin Li return StringRef();
1703*67e74705SXin Li
1704*67e74705SXin Li serialization::ModuleFile &
1705*67e74705SXin Li Mod = Reader->getModuleManager().getPrimaryModule();
1706*67e74705SXin Li return Mod.FileName;
1707*67e74705SXin Li }
1708*67e74705SXin Li
create(CompilerInvocation * CI,IntrusiveRefCntPtr<DiagnosticsEngine> Diags,bool CaptureDiagnostics,bool UserFilesAreVolatile)1709*67e74705SXin Li ASTUnit *ASTUnit::create(CompilerInvocation *CI,
1710*67e74705SXin Li IntrusiveRefCntPtr<DiagnosticsEngine> Diags,
1711*67e74705SXin Li bool CaptureDiagnostics,
1712*67e74705SXin Li bool UserFilesAreVolatile) {
1713*67e74705SXin Li std::unique_ptr<ASTUnit> AST;
1714*67e74705SXin Li AST.reset(new ASTUnit(false));
1715*67e74705SXin Li ConfigureDiags(Diags, *AST, CaptureDiagnostics);
1716*67e74705SXin Li AST->Diagnostics = Diags;
1717*67e74705SXin Li AST->Invocation = CI;
1718*67e74705SXin Li AST->FileSystemOpts = CI->getFileSystemOpts();
1719*67e74705SXin Li IntrusiveRefCntPtr<vfs::FileSystem> VFS =
1720*67e74705SXin Li createVFSFromCompilerInvocation(*CI, *Diags);
1721*67e74705SXin Li if (!VFS)
1722*67e74705SXin Li return nullptr;
1723*67e74705SXin Li AST->FileMgr = new FileManager(AST->FileSystemOpts, VFS);
1724*67e74705SXin Li AST->UserFilesAreVolatile = UserFilesAreVolatile;
1725*67e74705SXin Li AST->SourceMgr = new SourceManager(AST->getDiagnostics(), *AST->FileMgr,
1726*67e74705SXin Li UserFilesAreVolatile);
1727*67e74705SXin Li
1728*67e74705SXin Li return AST.release();
1729*67e74705SXin Li }
1730*67e74705SXin Li
LoadFromCompilerInvocationAction(CompilerInvocation * CI,std::shared_ptr<PCHContainerOperations> PCHContainerOps,IntrusiveRefCntPtr<DiagnosticsEngine> Diags,FrontendAction * Action,ASTUnit * Unit,bool Persistent,StringRef ResourceFilesPath,bool OnlyLocalDecls,bool CaptureDiagnostics,unsigned PrecompilePreambleAfterNParses,bool CacheCodeCompletionResults,bool IncludeBriefCommentsInCodeCompletion,bool UserFilesAreVolatile,std::unique_ptr<ASTUnit> * ErrAST)1731*67e74705SXin Li ASTUnit *ASTUnit::LoadFromCompilerInvocationAction(
1732*67e74705SXin Li CompilerInvocation *CI,
1733*67e74705SXin Li std::shared_ptr<PCHContainerOperations> PCHContainerOps,
1734*67e74705SXin Li IntrusiveRefCntPtr<DiagnosticsEngine> Diags, FrontendAction *Action,
1735*67e74705SXin Li ASTUnit *Unit, bool Persistent, StringRef ResourceFilesPath,
1736*67e74705SXin Li bool OnlyLocalDecls, bool CaptureDiagnostics,
1737*67e74705SXin Li unsigned PrecompilePreambleAfterNParses, bool CacheCodeCompletionResults,
1738*67e74705SXin Li bool IncludeBriefCommentsInCodeCompletion, bool UserFilesAreVolatile,
1739*67e74705SXin Li std::unique_ptr<ASTUnit> *ErrAST) {
1740*67e74705SXin Li assert(CI && "A CompilerInvocation is required");
1741*67e74705SXin Li
1742*67e74705SXin Li std::unique_ptr<ASTUnit> OwnAST;
1743*67e74705SXin Li ASTUnit *AST = Unit;
1744*67e74705SXin Li if (!AST) {
1745*67e74705SXin Li // Create the AST unit.
1746*67e74705SXin Li OwnAST.reset(create(CI, Diags, CaptureDiagnostics, UserFilesAreVolatile));
1747*67e74705SXin Li AST = OwnAST.get();
1748*67e74705SXin Li if (!AST)
1749*67e74705SXin Li return nullptr;
1750*67e74705SXin Li }
1751*67e74705SXin Li
1752*67e74705SXin Li if (!ResourceFilesPath.empty()) {
1753*67e74705SXin Li // Override the resources path.
1754*67e74705SXin Li CI->getHeaderSearchOpts().ResourceDir = ResourceFilesPath;
1755*67e74705SXin Li }
1756*67e74705SXin Li AST->OnlyLocalDecls = OnlyLocalDecls;
1757*67e74705SXin Li AST->CaptureDiagnostics = CaptureDiagnostics;
1758*67e74705SXin Li if (PrecompilePreambleAfterNParses > 0)
1759*67e74705SXin Li AST->PreambleRebuildCounter = PrecompilePreambleAfterNParses;
1760*67e74705SXin Li AST->TUKind = Action ? Action->getTranslationUnitKind() : TU_Complete;
1761*67e74705SXin Li AST->ShouldCacheCodeCompletionResults = CacheCodeCompletionResults;
1762*67e74705SXin Li AST->IncludeBriefCommentsInCodeCompletion
1763*67e74705SXin Li = IncludeBriefCommentsInCodeCompletion;
1764*67e74705SXin Li
1765*67e74705SXin Li // Recover resources if we crash before exiting this method.
1766*67e74705SXin Li llvm::CrashRecoveryContextCleanupRegistrar<ASTUnit>
1767*67e74705SXin Li ASTUnitCleanup(OwnAST.get());
1768*67e74705SXin Li llvm::CrashRecoveryContextCleanupRegistrar<DiagnosticsEngine,
1769*67e74705SXin Li llvm::CrashRecoveryContextReleaseRefCleanup<DiagnosticsEngine> >
1770*67e74705SXin Li DiagCleanup(Diags.get());
1771*67e74705SXin Li
1772*67e74705SXin Li // We'll manage file buffers ourselves.
1773*67e74705SXin Li CI->getPreprocessorOpts().RetainRemappedFileBuffers = true;
1774*67e74705SXin Li CI->getFrontendOpts().DisableFree = false;
1775*67e74705SXin Li ProcessWarningOptions(AST->getDiagnostics(), CI->getDiagnosticOpts());
1776*67e74705SXin Li
1777*67e74705SXin Li // Create the compiler instance to use for building the AST.
1778*67e74705SXin Li std::unique_ptr<CompilerInstance> Clang(
1779*67e74705SXin Li new CompilerInstance(std::move(PCHContainerOps)));
1780*67e74705SXin Li
1781*67e74705SXin Li // Recover resources if we crash before exiting this method.
1782*67e74705SXin Li llvm::CrashRecoveryContextCleanupRegistrar<CompilerInstance>
1783*67e74705SXin Li CICleanup(Clang.get());
1784*67e74705SXin Li
1785*67e74705SXin Li Clang->setInvocation(CI);
1786*67e74705SXin Li AST->OriginalSourceFile = Clang->getFrontendOpts().Inputs[0].getFile();
1787*67e74705SXin Li
1788*67e74705SXin Li // Set up diagnostics, capturing any diagnostics that would
1789*67e74705SXin Li // otherwise be dropped.
1790*67e74705SXin Li Clang->setDiagnostics(&AST->getDiagnostics());
1791*67e74705SXin Li
1792*67e74705SXin Li // Create the target instance.
1793*67e74705SXin Li Clang->setTarget(TargetInfo::CreateTargetInfo(
1794*67e74705SXin Li Clang->getDiagnostics(), Clang->getInvocation().TargetOpts));
1795*67e74705SXin Li if (!Clang->hasTarget())
1796*67e74705SXin Li return nullptr;
1797*67e74705SXin Li
1798*67e74705SXin Li // Inform the target of the language options.
1799*67e74705SXin Li //
1800*67e74705SXin Li // FIXME: We shouldn't need to do this, the target should be immutable once
1801*67e74705SXin Li // created. This complexity should be lifted elsewhere.
1802*67e74705SXin Li Clang->getTarget().adjust(Clang->getLangOpts());
1803*67e74705SXin Li
1804*67e74705SXin Li assert(Clang->getFrontendOpts().Inputs.size() == 1 &&
1805*67e74705SXin Li "Invocation must have exactly one source file!");
1806*67e74705SXin Li assert(Clang->getFrontendOpts().Inputs[0].getKind() != IK_AST &&
1807*67e74705SXin Li "FIXME: AST inputs not yet supported here!");
1808*67e74705SXin Li assert(Clang->getFrontendOpts().Inputs[0].getKind() != IK_LLVM_IR &&
1809*67e74705SXin Li "IR inputs not supported here!");
1810*67e74705SXin Li
1811*67e74705SXin Li // Configure the various subsystems.
1812*67e74705SXin Li AST->TheSema.reset();
1813*67e74705SXin Li AST->Ctx = nullptr;
1814*67e74705SXin Li AST->PP = nullptr;
1815*67e74705SXin Li AST->Reader = nullptr;
1816*67e74705SXin Li
1817*67e74705SXin Li // Create a file manager object to provide access to and cache the filesystem.
1818*67e74705SXin Li Clang->setFileManager(&AST->getFileManager());
1819*67e74705SXin Li
1820*67e74705SXin Li // Create the source manager.
1821*67e74705SXin Li Clang->setSourceManager(&AST->getSourceManager());
1822*67e74705SXin Li
1823*67e74705SXin Li FrontendAction *Act = Action;
1824*67e74705SXin Li
1825*67e74705SXin Li std::unique_ptr<TopLevelDeclTrackerAction> TrackerAct;
1826*67e74705SXin Li if (!Act) {
1827*67e74705SXin Li TrackerAct.reset(new TopLevelDeclTrackerAction(*AST));
1828*67e74705SXin Li Act = TrackerAct.get();
1829*67e74705SXin Li }
1830*67e74705SXin Li
1831*67e74705SXin Li // Recover resources if we crash before exiting this method.
1832*67e74705SXin Li llvm::CrashRecoveryContextCleanupRegistrar<TopLevelDeclTrackerAction>
1833*67e74705SXin Li ActCleanup(TrackerAct.get());
1834*67e74705SXin Li
1835*67e74705SXin Li if (!Act->BeginSourceFile(*Clang.get(), Clang->getFrontendOpts().Inputs[0])) {
1836*67e74705SXin Li AST->transferASTDataFromCompilerInstance(*Clang);
1837*67e74705SXin Li if (OwnAST && ErrAST)
1838*67e74705SXin Li ErrAST->swap(OwnAST);
1839*67e74705SXin Li
1840*67e74705SXin Li return nullptr;
1841*67e74705SXin Li }
1842*67e74705SXin Li
1843*67e74705SXin Li if (Persistent && !TrackerAct) {
1844*67e74705SXin Li Clang->getPreprocessor().addPPCallbacks(
1845*67e74705SXin Li llvm::make_unique<MacroDefinitionTrackerPPCallbacks>(
1846*67e74705SXin Li AST->getCurrentTopLevelHashValue()));
1847*67e74705SXin Li std::vector<std::unique_ptr<ASTConsumer>> Consumers;
1848*67e74705SXin Li if (Clang->hasASTConsumer())
1849*67e74705SXin Li Consumers.push_back(Clang->takeASTConsumer());
1850*67e74705SXin Li Consumers.push_back(llvm::make_unique<TopLevelDeclTrackerConsumer>(
1851*67e74705SXin Li *AST, AST->getCurrentTopLevelHashValue()));
1852*67e74705SXin Li Clang->setASTConsumer(
1853*67e74705SXin Li llvm::make_unique<MultiplexConsumer>(std::move(Consumers)));
1854*67e74705SXin Li }
1855*67e74705SXin Li if (!Act->Execute()) {
1856*67e74705SXin Li AST->transferASTDataFromCompilerInstance(*Clang);
1857*67e74705SXin Li if (OwnAST && ErrAST)
1858*67e74705SXin Li ErrAST->swap(OwnAST);
1859*67e74705SXin Li
1860*67e74705SXin Li return nullptr;
1861*67e74705SXin Li }
1862*67e74705SXin Li
1863*67e74705SXin Li // Steal the created target, context, and preprocessor.
1864*67e74705SXin Li AST->transferASTDataFromCompilerInstance(*Clang);
1865*67e74705SXin Li
1866*67e74705SXin Li Act->EndSourceFile();
1867*67e74705SXin Li
1868*67e74705SXin Li if (OwnAST)
1869*67e74705SXin Li return OwnAST.release();
1870*67e74705SXin Li else
1871*67e74705SXin Li return AST;
1872*67e74705SXin Li }
1873*67e74705SXin Li
LoadFromCompilerInvocation(std::shared_ptr<PCHContainerOperations> PCHContainerOps,unsigned PrecompilePreambleAfterNParses)1874*67e74705SXin Li bool ASTUnit::LoadFromCompilerInvocation(
1875*67e74705SXin Li std::shared_ptr<PCHContainerOperations> PCHContainerOps,
1876*67e74705SXin Li unsigned PrecompilePreambleAfterNParses) {
1877*67e74705SXin Li if (!Invocation)
1878*67e74705SXin Li return true;
1879*67e74705SXin Li
1880*67e74705SXin Li // We'll manage file buffers ourselves.
1881*67e74705SXin Li Invocation->getPreprocessorOpts().RetainRemappedFileBuffers = true;
1882*67e74705SXin Li Invocation->getFrontendOpts().DisableFree = false;
1883*67e74705SXin Li ProcessWarningOptions(getDiagnostics(), Invocation->getDiagnosticOpts());
1884*67e74705SXin Li
1885*67e74705SXin Li std::unique_ptr<llvm::MemoryBuffer> OverrideMainBuffer;
1886*67e74705SXin Li if (PrecompilePreambleAfterNParses > 0) {
1887*67e74705SXin Li PreambleRebuildCounter = PrecompilePreambleAfterNParses;
1888*67e74705SXin Li OverrideMainBuffer =
1889*67e74705SXin Li getMainBufferWithPrecompiledPreamble(PCHContainerOps, *Invocation);
1890*67e74705SXin Li }
1891*67e74705SXin Li
1892*67e74705SXin Li SimpleTimer ParsingTimer(WantTiming);
1893*67e74705SXin Li ParsingTimer.setOutput("Parsing " + getMainFileName());
1894*67e74705SXin Li
1895*67e74705SXin Li // Recover resources if we crash before exiting this method.
1896*67e74705SXin Li llvm::CrashRecoveryContextCleanupRegistrar<llvm::MemoryBuffer>
1897*67e74705SXin Li MemBufferCleanup(OverrideMainBuffer.get());
1898*67e74705SXin Li
1899*67e74705SXin Li return Parse(std::move(PCHContainerOps), std::move(OverrideMainBuffer));
1900*67e74705SXin Li }
1901*67e74705SXin Li
LoadFromCompilerInvocation(CompilerInvocation * CI,std::shared_ptr<PCHContainerOperations> PCHContainerOps,IntrusiveRefCntPtr<DiagnosticsEngine> Diags,FileManager * FileMgr,bool OnlyLocalDecls,bool CaptureDiagnostics,unsigned PrecompilePreambleAfterNParses,TranslationUnitKind TUKind,bool CacheCodeCompletionResults,bool IncludeBriefCommentsInCodeCompletion,bool UserFilesAreVolatile)1902*67e74705SXin Li std::unique_ptr<ASTUnit> ASTUnit::LoadFromCompilerInvocation(
1903*67e74705SXin Li CompilerInvocation *CI,
1904*67e74705SXin Li std::shared_ptr<PCHContainerOperations> PCHContainerOps,
1905*67e74705SXin Li IntrusiveRefCntPtr<DiagnosticsEngine> Diags, FileManager *FileMgr,
1906*67e74705SXin Li bool OnlyLocalDecls, bool CaptureDiagnostics,
1907*67e74705SXin Li unsigned PrecompilePreambleAfterNParses, TranslationUnitKind TUKind,
1908*67e74705SXin Li bool CacheCodeCompletionResults, bool IncludeBriefCommentsInCodeCompletion,
1909*67e74705SXin Li bool UserFilesAreVolatile) {
1910*67e74705SXin Li // Create the AST unit.
1911*67e74705SXin Li std::unique_ptr<ASTUnit> AST(new ASTUnit(false));
1912*67e74705SXin Li ConfigureDiags(Diags, *AST, CaptureDiagnostics);
1913*67e74705SXin Li AST->Diagnostics = Diags;
1914*67e74705SXin Li AST->OnlyLocalDecls = OnlyLocalDecls;
1915*67e74705SXin Li AST->CaptureDiagnostics = CaptureDiagnostics;
1916*67e74705SXin Li AST->TUKind = TUKind;
1917*67e74705SXin Li AST->ShouldCacheCodeCompletionResults = CacheCodeCompletionResults;
1918*67e74705SXin Li AST->IncludeBriefCommentsInCodeCompletion
1919*67e74705SXin Li = IncludeBriefCommentsInCodeCompletion;
1920*67e74705SXin Li AST->Invocation = CI;
1921*67e74705SXin Li AST->FileSystemOpts = FileMgr->getFileSystemOpts();
1922*67e74705SXin Li AST->FileMgr = FileMgr;
1923*67e74705SXin Li AST->UserFilesAreVolatile = UserFilesAreVolatile;
1924*67e74705SXin Li
1925*67e74705SXin Li // Recover resources if we crash before exiting this method.
1926*67e74705SXin Li llvm::CrashRecoveryContextCleanupRegistrar<ASTUnit>
1927*67e74705SXin Li ASTUnitCleanup(AST.get());
1928*67e74705SXin Li llvm::CrashRecoveryContextCleanupRegistrar<DiagnosticsEngine,
1929*67e74705SXin Li llvm::CrashRecoveryContextReleaseRefCleanup<DiagnosticsEngine> >
1930*67e74705SXin Li DiagCleanup(Diags.get());
1931*67e74705SXin Li
1932*67e74705SXin Li if (AST->LoadFromCompilerInvocation(std::move(PCHContainerOps),
1933*67e74705SXin Li PrecompilePreambleAfterNParses))
1934*67e74705SXin Li return nullptr;
1935*67e74705SXin Li return AST;
1936*67e74705SXin Li }
1937*67e74705SXin Li
LoadFromCommandLine(const char ** ArgBegin,const char ** ArgEnd,std::shared_ptr<PCHContainerOperations> PCHContainerOps,IntrusiveRefCntPtr<DiagnosticsEngine> Diags,StringRef ResourceFilesPath,bool OnlyLocalDecls,bool CaptureDiagnostics,ArrayRef<RemappedFile> RemappedFiles,bool RemappedFilesKeepOriginalName,unsigned PrecompilePreambleAfterNParses,TranslationUnitKind TUKind,bool CacheCodeCompletionResults,bool IncludeBriefCommentsInCodeCompletion,bool AllowPCHWithCompilerErrors,bool SkipFunctionBodies,bool UserFilesAreVolatile,bool ForSerialization,llvm::Optional<StringRef> ModuleFormat,std::unique_ptr<ASTUnit> * ErrAST)1938*67e74705SXin Li ASTUnit *ASTUnit::LoadFromCommandLine(
1939*67e74705SXin Li const char **ArgBegin, const char **ArgEnd,
1940*67e74705SXin Li std::shared_ptr<PCHContainerOperations> PCHContainerOps,
1941*67e74705SXin Li IntrusiveRefCntPtr<DiagnosticsEngine> Diags, StringRef ResourceFilesPath,
1942*67e74705SXin Li bool OnlyLocalDecls, bool CaptureDiagnostics,
1943*67e74705SXin Li ArrayRef<RemappedFile> RemappedFiles, bool RemappedFilesKeepOriginalName,
1944*67e74705SXin Li unsigned PrecompilePreambleAfterNParses, TranslationUnitKind TUKind,
1945*67e74705SXin Li bool CacheCodeCompletionResults, bool IncludeBriefCommentsInCodeCompletion,
1946*67e74705SXin Li bool AllowPCHWithCompilerErrors, bool SkipFunctionBodies,
1947*67e74705SXin Li bool UserFilesAreVolatile, bool ForSerialization,
1948*67e74705SXin Li llvm::Optional<StringRef> ModuleFormat, std::unique_ptr<ASTUnit> *ErrAST) {
1949*67e74705SXin Li assert(Diags.get() && "no DiagnosticsEngine was provided");
1950*67e74705SXin Li
1951*67e74705SXin Li SmallVector<StoredDiagnostic, 4> StoredDiagnostics;
1952*67e74705SXin Li
1953*67e74705SXin Li IntrusiveRefCntPtr<CompilerInvocation> CI;
1954*67e74705SXin Li
1955*67e74705SXin Li {
1956*67e74705SXin Li
1957*67e74705SXin Li CaptureDroppedDiagnostics Capture(CaptureDiagnostics, *Diags,
1958*67e74705SXin Li StoredDiagnostics);
1959*67e74705SXin Li
1960*67e74705SXin Li CI = clang::createInvocationFromCommandLine(
1961*67e74705SXin Li llvm::makeArrayRef(ArgBegin, ArgEnd),
1962*67e74705SXin Li Diags);
1963*67e74705SXin Li if (!CI)
1964*67e74705SXin Li return nullptr;
1965*67e74705SXin Li }
1966*67e74705SXin Li
1967*67e74705SXin Li // Override any files that need remapping
1968*67e74705SXin Li for (const auto &RemappedFile : RemappedFiles) {
1969*67e74705SXin Li CI->getPreprocessorOpts().addRemappedFile(RemappedFile.first,
1970*67e74705SXin Li RemappedFile.second);
1971*67e74705SXin Li }
1972*67e74705SXin Li PreprocessorOptions &PPOpts = CI->getPreprocessorOpts();
1973*67e74705SXin Li PPOpts.RemappedFilesKeepOriginalName = RemappedFilesKeepOriginalName;
1974*67e74705SXin Li PPOpts.AllowPCHWithCompilerErrors = AllowPCHWithCompilerErrors;
1975*67e74705SXin Li
1976*67e74705SXin Li // Override the resources path.
1977*67e74705SXin Li CI->getHeaderSearchOpts().ResourceDir = ResourceFilesPath;
1978*67e74705SXin Li
1979*67e74705SXin Li CI->getFrontendOpts().SkipFunctionBodies = SkipFunctionBodies;
1980*67e74705SXin Li
1981*67e74705SXin Li if (ModuleFormat)
1982*67e74705SXin Li CI->getHeaderSearchOpts().ModuleFormat = ModuleFormat.getValue();
1983*67e74705SXin Li
1984*67e74705SXin Li // Create the AST unit.
1985*67e74705SXin Li std::unique_ptr<ASTUnit> AST;
1986*67e74705SXin Li AST.reset(new ASTUnit(false));
1987*67e74705SXin Li ConfigureDiags(Diags, *AST, CaptureDiagnostics);
1988*67e74705SXin Li AST->Diagnostics = Diags;
1989*67e74705SXin Li AST->FileSystemOpts = CI->getFileSystemOpts();
1990*67e74705SXin Li IntrusiveRefCntPtr<vfs::FileSystem> VFS =
1991*67e74705SXin Li createVFSFromCompilerInvocation(*CI, *Diags);
1992*67e74705SXin Li if (!VFS)
1993*67e74705SXin Li return nullptr;
1994*67e74705SXin Li AST->FileMgr = new FileManager(AST->FileSystemOpts, VFS);
1995*67e74705SXin Li AST->OnlyLocalDecls = OnlyLocalDecls;
1996*67e74705SXin Li AST->CaptureDiagnostics = CaptureDiagnostics;
1997*67e74705SXin Li AST->TUKind = TUKind;
1998*67e74705SXin Li AST->ShouldCacheCodeCompletionResults = CacheCodeCompletionResults;
1999*67e74705SXin Li AST->IncludeBriefCommentsInCodeCompletion
2000*67e74705SXin Li = IncludeBriefCommentsInCodeCompletion;
2001*67e74705SXin Li AST->UserFilesAreVolatile = UserFilesAreVolatile;
2002*67e74705SXin Li AST->NumStoredDiagnosticsFromDriver = StoredDiagnostics.size();
2003*67e74705SXin Li AST->StoredDiagnostics.swap(StoredDiagnostics);
2004*67e74705SXin Li AST->Invocation = CI;
2005*67e74705SXin Li if (ForSerialization)
2006*67e74705SXin Li AST->WriterData.reset(new ASTWriterData());
2007*67e74705SXin Li // Zero out now to ease cleanup during crash recovery.
2008*67e74705SXin Li CI = nullptr;
2009*67e74705SXin Li Diags = nullptr;
2010*67e74705SXin Li
2011*67e74705SXin Li // Recover resources if we crash before exiting this method.
2012*67e74705SXin Li llvm::CrashRecoveryContextCleanupRegistrar<ASTUnit>
2013*67e74705SXin Li ASTUnitCleanup(AST.get());
2014*67e74705SXin Li
2015*67e74705SXin Li if (AST->LoadFromCompilerInvocation(std::move(PCHContainerOps),
2016*67e74705SXin Li PrecompilePreambleAfterNParses)) {
2017*67e74705SXin Li // Some error occurred, if caller wants to examine diagnostics, pass it the
2018*67e74705SXin Li // ASTUnit.
2019*67e74705SXin Li if (ErrAST) {
2020*67e74705SXin Li AST->StoredDiagnostics.swap(AST->FailedParseDiagnostics);
2021*67e74705SXin Li ErrAST->swap(AST);
2022*67e74705SXin Li }
2023*67e74705SXin Li return nullptr;
2024*67e74705SXin Li }
2025*67e74705SXin Li
2026*67e74705SXin Li return AST.release();
2027*67e74705SXin Li }
2028*67e74705SXin Li
Reparse(std::shared_ptr<PCHContainerOperations> PCHContainerOps,ArrayRef<RemappedFile> RemappedFiles)2029*67e74705SXin Li bool ASTUnit::Reparse(std::shared_ptr<PCHContainerOperations> PCHContainerOps,
2030*67e74705SXin Li ArrayRef<RemappedFile> RemappedFiles) {
2031*67e74705SXin Li if (!Invocation)
2032*67e74705SXin Li return true;
2033*67e74705SXin Li
2034*67e74705SXin Li clearFileLevelDecls();
2035*67e74705SXin Li
2036*67e74705SXin Li SimpleTimer ParsingTimer(WantTiming);
2037*67e74705SXin Li ParsingTimer.setOutput("Reparsing " + getMainFileName());
2038*67e74705SXin Li
2039*67e74705SXin Li // Remap files.
2040*67e74705SXin Li PreprocessorOptions &PPOpts = Invocation->getPreprocessorOpts();
2041*67e74705SXin Li for (const auto &RB : PPOpts.RemappedFileBuffers)
2042*67e74705SXin Li delete RB.second;
2043*67e74705SXin Li
2044*67e74705SXin Li Invocation->getPreprocessorOpts().clearRemappedFiles();
2045*67e74705SXin Li for (const auto &RemappedFile : RemappedFiles) {
2046*67e74705SXin Li Invocation->getPreprocessorOpts().addRemappedFile(RemappedFile.first,
2047*67e74705SXin Li RemappedFile.second);
2048*67e74705SXin Li }
2049*67e74705SXin Li
2050*67e74705SXin Li // If we have a preamble file lying around, or if we might try to
2051*67e74705SXin Li // build a precompiled preamble, do so now.
2052*67e74705SXin Li std::unique_ptr<llvm::MemoryBuffer> OverrideMainBuffer;
2053*67e74705SXin Li if (!getPreambleFile(this).empty() || PreambleRebuildCounter > 0)
2054*67e74705SXin Li OverrideMainBuffer =
2055*67e74705SXin Li getMainBufferWithPrecompiledPreamble(PCHContainerOps, *Invocation);
2056*67e74705SXin Li
2057*67e74705SXin Li // Clear out the diagnostics state.
2058*67e74705SXin Li FileMgr.reset();
2059*67e74705SXin Li getDiagnostics().Reset();
2060*67e74705SXin Li ProcessWarningOptions(getDiagnostics(), Invocation->getDiagnosticOpts());
2061*67e74705SXin Li if (OverrideMainBuffer)
2062*67e74705SXin Li getDiagnostics().setNumWarnings(NumWarningsInPreamble);
2063*67e74705SXin Li
2064*67e74705SXin Li // Parse the sources
2065*67e74705SXin Li bool Result =
2066*67e74705SXin Li Parse(std::move(PCHContainerOps), std::move(OverrideMainBuffer));
2067*67e74705SXin Li
2068*67e74705SXin Li // If we're caching global code-completion results, and the top-level
2069*67e74705SXin Li // declarations have changed, clear out the code-completion cache.
2070*67e74705SXin Li if (!Result && ShouldCacheCodeCompletionResults &&
2071*67e74705SXin Li CurrentTopLevelHashValue != CompletionCacheTopLevelHashValue)
2072*67e74705SXin Li CacheCodeCompletionResults();
2073*67e74705SXin Li
2074*67e74705SXin Li // We now need to clear out the completion info related to this translation
2075*67e74705SXin Li // unit; it'll be recreated if necessary.
2076*67e74705SXin Li CCTUInfo.reset();
2077*67e74705SXin Li
2078*67e74705SXin Li return Result;
2079*67e74705SXin Li }
2080*67e74705SXin Li
2081*67e74705SXin Li //----------------------------------------------------------------------------//
2082*67e74705SXin Li // Code completion
2083*67e74705SXin Li //----------------------------------------------------------------------------//
2084*67e74705SXin Li
2085*67e74705SXin Li namespace {
2086*67e74705SXin Li /// \brief Code completion consumer that combines the cached code-completion
2087*67e74705SXin Li /// results from an ASTUnit with the code-completion results provided to it,
2088*67e74705SXin Li /// then passes the result on to
2089*67e74705SXin Li class AugmentedCodeCompleteConsumer : public CodeCompleteConsumer {
2090*67e74705SXin Li uint64_t NormalContexts;
2091*67e74705SXin Li ASTUnit &AST;
2092*67e74705SXin Li CodeCompleteConsumer &Next;
2093*67e74705SXin Li
2094*67e74705SXin Li public:
AugmentedCodeCompleteConsumer(ASTUnit & AST,CodeCompleteConsumer & Next,const CodeCompleteOptions & CodeCompleteOpts)2095*67e74705SXin Li AugmentedCodeCompleteConsumer(ASTUnit &AST, CodeCompleteConsumer &Next,
2096*67e74705SXin Li const CodeCompleteOptions &CodeCompleteOpts)
2097*67e74705SXin Li : CodeCompleteConsumer(CodeCompleteOpts, Next.isOutputBinary()),
2098*67e74705SXin Li AST(AST), Next(Next)
2099*67e74705SXin Li {
2100*67e74705SXin Li // Compute the set of contexts in which we will look when we don't have
2101*67e74705SXin Li // any information about the specific context.
2102*67e74705SXin Li NormalContexts
2103*67e74705SXin Li = (1LL << CodeCompletionContext::CCC_TopLevel)
2104*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_ObjCInterface)
2105*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_ObjCImplementation)
2106*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_ObjCIvarList)
2107*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_Statement)
2108*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_Expression)
2109*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_ObjCMessageReceiver)
2110*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_DotMemberAccess)
2111*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_ArrowMemberAccess)
2112*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_ObjCPropertyAccess)
2113*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_ObjCProtocolName)
2114*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_ParenthesizedExpression)
2115*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_Recovery);
2116*67e74705SXin Li
2117*67e74705SXin Li if (AST.getASTContext().getLangOpts().CPlusPlus)
2118*67e74705SXin Li NormalContexts |= (1LL << CodeCompletionContext::CCC_EnumTag)
2119*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_UnionTag)
2120*67e74705SXin Li | (1LL << CodeCompletionContext::CCC_ClassOrStructTag);
2121*67e74705SXin Li }
2122*67e74705SXin Li
2123*67e74705SXin Li void ProcessCodeCompleteResults(Sema &S, CodeCompletionContext Context,
2124*67e74705SXin Li CodeCompletionResult *Results,
2125*67e74705SXin Li unsigned NumResults) override;
2126*67e74705SXin Li
ProcessOverloadCandidates(Sema & S,unsigned CurrentArg,OverloadCandidate * Candidates,unsigned NumCandidates)2127*67e74705SXin Li void ProcessOverloadCandidates(Sema &S, unsigned CurrentArg,
2128*67e74705SXin Li OverloadCandidate *Candidates,
2129*67e74705SXin Li unsigned NumCandidates) override {
2130*67e74705SXin Li Next.ProcessOverloadCandidates(S, CurrentArg, Candidates, NumCandidates);
2131*67e74705SXin Li }
2132*67e74705SXin Li
getAllocator()2133*67e74705SXin Li CodeCompletionAllocator &getAllocator() override {
2134*67e74705SXin Li return Next.getAllocator();
2135*67e74705SXin Li }
2136*67e74705SXin Li
getCodeCompletionTUInfo()2137*67e74705SXin Li CodeCompletionTUInfo &getCodeCompletionTUInfo() override {
2138*67e74705SXin Li return Next.getCodeCompletionTUInfo();
2139*67e74705SXin Li }
2140*67e74705SXin Li };
2141*67e74705SXin Li } // anonymous namespace
2142*67e74705SXin Li
2143*67e74705SXin Li /// \brief Helper function that computes which global names are hidden by the
2144*67e74705SXin Li /// local code-completion results.
CalculateHiddenNames(const CodeCompletionContext & Context,CodeCompletionResult * Results,unsigned NumResults,ASTContext & Ctx,llvm::StringSet<llvm::BumpPtrAllocator> & HiddenNames)2145*67e74705SXin Li static void CalculateHiddenNames(const CodeCompletionContext &Context,
2146*67e74705SXin Li CodeCompletionResult *Results,
2147*67e74705SXin Li unsigned NumResults,
2148*67e74705SXin Li ASTContext &Ctx,
2149*67e74705SXin Li llvm::StringSet<llvm::BumpPtrAllocator> &HiddenNames){
2150*67e74705SXin Li bool OnlyTagNames = false;
2151*67e74705SXin Li switch (Context.getKind()) {
2152*67e74705SXin Li case CodeCompletionContext::CCC_Recovery:
2153*67e74705SXin Li case CodeCompletionContext::CCC_TopLevel:
2154*67e74705SXin Li case CodeCompletionContext::CCC_ObjCInterface:
2155*67e74705SXin Li case CodeCompletionContext::CCC_ObjCImplementation:
2156*67e74705SXin Li case CodeCompletionContext::CCC_ObjCIvarList:
2157*67e74705SXin Li case CodeCompletionContext::CCC_ClassStructUnion:
2158*67e74705SXin Li case CodeCompletionContext::CCC_Statement:
2159*67e74705SXin Li case CodeCompletionContext::CCC_Expression:
2160*67e74705SXin Li case CodeCompletionContext::CCC_ObjCMessageReceiver:
2161*67e74705SXin Li case CodeCompletionContext::CCC_DotMemberAccess:
2162*67e74705SXin Li case CodeCompletionContext::CCC_ArrowMemberAccess:
2163*67e74705SXin Li case CodeCompletionContext::CCC_ObjCPropertyAccess:
2164*67e74705SXin Li case CodeCompletionContext::CCC_Namespace:
2165*67e74705SXin Li case CodeCompletionContext::CCC_Type:
2166*67e74705SXin Li case CodeCompletionContext::CCC_Name:
2167*67e74705SXin Li case CodeCompletionContext::CCC_PotentiallyQualifiedName:
2168*67e74705SXin Li case CodeCompletionContext::CCC_ParenthesizedExpression:
2169*67e74705SXin Li case CodeCompletionContext::CCC_ObjCInterfaceName:
2170*67e74705SXin Li break;
2171*67e74705SXin Li
2172*67e74705SXin Li case CodeCompletionContext::CCC_EnumTag:
2173*67e74705SXin Li case CodeCompletionContext::CCC_UnionTag:
2174*67e74705SXin Li case CodeCompletionContext::CCC_ClassOrStructTag:
2175*67e74705SXin Li OnlyTagNames = true;
2176*67e74705SXin Li break;
2177*67e74705SXin Li
2178*67e74705SXin Li case CodeCompletionContext::CCC_ObjCProtocolName:
2179*67e74705SXin Li case CodeCompletionContext::CCC_MacroName:
2180*67e74705SXin Li case CodeCompletionContext::CCC_MacroNameUse:
2181*67e74705SXin Li case CodeCompletionContext::CCC_PreprocessorExpression:
2182*67e74705SXin Li case CodeCompletionContext::CCC_PreprocessorDirective:
2183*67e74705SXin Li case CodeCompletionContext::CCC_NaturalLanguage:
2184*67e74705SXin Li case CodeCompletionContext::CCC_SelectorName:
2185*67e74705SXin Li case CodeCompletionContext::CCC_TypeQualifiers:
2186*67e74705SXin Li case CodeCompletionContext::CCC_Other:
2187*67e74705SXin Li case CodeCompletionContext::CCC_OtherWithMacros:
2188*67e74705SXin Li case CodeCompletionContext::CCC_ObjCInstanceMessage:
2189*67e74705SXin Li case CodeCompletionContext::CCC_ObjCClassMessage:
2190*67e74705SXin Li case CodeCompletionContext::CCC_ObjCCategoryName:
2191*67e74705SXin Li // We're looking for nothing, or we're looking for names that cannot
2192*67e74705SXin Li // be hidden.
2193*67e74705SXin Li return;
2194*67e74705SXin Li }
2195*67e74705SXin Li
2196*67e74705SXin Li typedef CodeCompletionResult Result;
2197*67e74705SXin Li for (unsigned I = 0; I != NumResults; ++I) {
2198*67e74705SXin Li if (Results[I].Kind != Result::RK_Declaration)
2199*67e74705SXin Li continue;
2200*67e74705SXin Li
2201*67e74705SXin Li unsigned IDNS
2202*67e74705SXin Li = Results[I].Declaration->getUnderlyingDecl()->getIdentifierNamespace();
2203*67e74705SXin Li
2204*67e74705SXin Li bool Hiding = false;
2205*67e74705SXin Li if (OnlyTagNames)
2206*67e74705SXin Li Hiding = (IDNS & Decl::IDNS_Tag);
2207*67e74705SXin Li else {
2208*67e74705SXin Li unsigned HiddenIDNS = (Decl::IDNS_Type | Decl::IDNS_Member |
2209*67e74705SXin Li Decl::IDNS_Namespace | Decl::IDNS_Ordinary |
2210*67e74705SXin Li Decl::IDNS_NonMemberOperator);
2211*67e74705SXin Li if (Ctx.getLangOpts().CPlusPlus)
2212*67e74705SXin Li HiddenIDNS |= Decl::IDNS_Tag;
2213*67e74705SXin Li Hiding = (IDNS & HiddenIDNS);
2214*67e74705SXin Li }
2215*67e74705SXin Li
2216*67e74705SXin Li if (!Hiding)
2217*67e74705SXin Li continue;
2218*67e74705SXin Li
2219*67e74705SXin Li DeclarationName Name = Results[I].Declaration->getDeclName();
2220*67e74705SXin Li if (IdentifierInfo *Identifier = Name.getAsIdentifierInfo())
2221*67e74705SXin Li HiddenNames.insert(Identifier->getName());
2222*67e74705SXin Li else
2223*67e74705SXin Li HiddenNames.insert(Name.getAsString());
2224*67e74705SXin Li }
2225*67e74705SXin Li }
2226*67e74705SXin Li
ProcessCodeCompleteResults(Sema & S,CodeCompletionContext Context,CodeCompletionResult * Results,unsigned NumResults)2227*67e74705SXin Li void AugmentedCodeCompleteConsumer::ProcessCodeCompleteResults(Sema &S,
2228*67e74705SXin Li CodeCompletionContext Context,
2229*67e74705SXin Li CodeCompletionResult *Results,
2230*67e74705SXin Li unsigned NumResults) {
2231*67e74705SXin Li // Merge the results we were given with the results we cached.
2232*67e74705SXin Li bool AddedResult = false;
2233*67e74705SXin Li uint64_t InContexts =
2234*67e74705SXin Li Context.getKind() == CodeCompletionContext::CCC_Recovery
2235*67e74705SXin Li ? NormalContexts : (1LL << Context.getKind());
2236*67e74705SXin Li // Contains the set of names that are hidden by "local" completion results.
2237*67e74705SXin Li llvm::StringSet<llvm::BumpPtrAllocator> HiddenNames;
2238*67e74705SXin Li typedef CodeCompletionResult Result;
2239*67e74705SXin Li SmallVector<Result, 8> AllResults;
2240*67e74705SXin Li for (ASTUnit::cached_completion_iterator
2241*67e74705SXin Li C = AST.cached_completion_begin(),
2242*67e74705SXin Li CEnd = AST.cached_completion_end();
2243*67e74705SXin Li C != CEnd; ++C) {
2244*67e74705SXin Li // If the context we are in matches any of the contexts we are
2245*67e74705SXin Li // interested in, we'll add this result.
2246*67e74705SXin Li if ((C->ShowInContexts & InContexts) == 0)
2247*67e74705SXin Li continue;
2248*67e74705SXin Li
2249*67e74705SXin Li // If we haven't added any results previously, do so now.
2250*67e74705SXin Li if (!AddedResult) {
2251*67e74705SXin Li CalculateHiddenNames(Context, Results, NumResults, S.Context,
2252*67e74705SXin Li HiddenNames);
2253*67e74705SXin Li AllResults.insert(AllResults.end(), Results, Results + NumResults);
2254*67e74705SXin Li AddedResult = true;
2255*67e74705SXin Li }
2256*67e74705SXin Li
2257*67e74705SXin Li // Determine whether this global completion result is hidden by a local
2258*67e74705SXin Li // completion result. If so, skip it.
2259*67e74705SXin Li if (C->Kind != CXCursor_MacroDefinition &&
2260*67e74705SXin Li HiddenNames.count(C->Completion->getTypedText()))
2261*67e74705SXin Li continue;
2262*67e74705SXin Li
2263*67e74705SXin Li // Adjust priority based on similar type classes.
2264*67e74705SXin Li unsigned Priority = C->Priority;
2265*67e74705SXin Li CodeCompletionString *Completion = C->Completion;
2266*67e74705SXin Li if (!Context.getPreferredType().isNull()) {
2267*67e74705SXin Li if (C->Kind == CXCursor_MacroDefinition) {
2268*67e74705SXin Li Priority = getMacroUsagePriority(C->Completion->getTypedText(),
2269*67e74705SXin Li S.getLangOpts(),
2270*67e74705SXin Li Context.getPreferredType()->isAnyPointerType());
2271*67e74705SXin Li } else if (C->Type) {
2272*67e74705SXin Li CanQualType Expected
2273*67e74705SXin Li = S.Context.getCanonicalType(
2274*67e74705SXin Li Context.getPreferredType().getUnqualifiedType());
2275*67e74705SXin Li SimplifiedTypeClass ExpectedSTC = getSimplifiedTypeClass(Expected);
2276*67e74705SXin Li if (ExpectedSTC == C->TypeClass) {
2277*67e74705SXin Li // We know this type is similar; check for an exact match.
2278*67e74705SXin Li llvm::StringMap<unsigned> &CachedCompletionTypes
2279*67e74705SXin Li = AST.getCachedCompletionTypes();
2280*67e74705SXin Li llvm::StringMap<unsigned>::iterator Pos
2281*67e74705SXin Li = CachedCompletionTypes.find(QualType(Expected).getAsString());
2282*67e74705SXin Li if (Pos != CachedCompletionTypes.end() && Pos->second == C->Type)
2283*67e74705SXin Li Priority /= CCF_ExactTypeMatch;
2284*67e74705SXin Li else
2285*67e74705SXin Li Priority /= CCF_SimilarTypeMatch;
2286*67e74705SXin Li }
2287*67e74705SXin Li }
2288*67e74705SXin Li }
2289*67e74705SXin Li
2290*67e74705SXin Li // Adjust the completion string, if required.
2291*67e74705SXin Li if (C->Kind == CXCursor_MacroDefinition &&
2292*67e74705SXin Li Context.getKind() == CodeCompletionContext::CCC_MacroNameUse) {
2293*67e74705SXin Li // Create a new code-completion string that just contains the
2294*67e74705SXin Li // macro name, without its arguments.
2295*67e74705SXin Li CodeCompletionBuilder Builder(getAllocator(), getCodeCompletionTUInfo(),
2296*67e74705SXin Li CCP_CodePattern, C->Availability);
2297*67e74705SXin Li Builder.AddTypedTextChunk(C->Completion->getTypedText());
2298*67e74705SXin Li Priority = CCP_CodePattern;
2299*67e74705SXin Li Completion = Builder.TakeString();
2300*67e74705SXin Li }
2301*67e74705SXin Li
2302*67e74705SXin Li AllResults.push_back(Result(Completion, Priority, C->Kind,
2303*67e74705SXin Li C->Availability));
2304*67e74705SXin Li }
2305*67e74705SXin Li
2306*67e74705SXin Li // If we did not add any cached completion results, just forward the
2307*67e74705SXin Li // results we were given to the next consumer.
2308*67e74705SXin Li if (!AddedResult) {
2309*67e74705SXin Li Next.ProcessCodeCompleteResults(S, Context, Results, NumResults);
2310*67e74705SXin Li return;
2311*67e74705SXin Li }
2312*67e74705SXin Li
2313*67e74705SXin Li Next.ProcessCodeCompleteResults(S, Context, AllResults.data(),
2314*67e74705SXin Li AllResults.size());
2315*67e74705SXin Li }
2316*67e74705SXin Li
CodeComplete(StringRef File,unsigned Line,unsigned Column,ArrayRef<RemappedFile> RemappedFiles,bool IncludeMacros,bool IncludeCodePatterns,bool IncludeBriefComments,CodeCompleteConsumer & Consumer,std::shared_ptr<PCHContainerOperations> PCHContainerOps,DiagnosticsEngine & Diag,LangOptions & LangOpts,SourceManager & SourceMgr,FileManager & FileMgr,SmallVectorImpl<StoredDiagnostic> & StoredDiagnostics,SmallVectorImpl<const llvm::MemoryBuffer * > & OwnedBuffers)2317*67e74705SXin Li void ASTUnit::CodeComplete(
2318*67e74705SXin Li StringRef File, unsigned Line, unsigned Column,
2319*67e74705SXin Li ArrayRef<RemappedFile> RemappedFiles, bool IncludeMacros,
2320*67e74705SXin Li bool IncludeCodePatterns, bool IncludeBriefComments,
2321*67e74705SXin Li CodeCompleteConsumer &Consumer,
2322*67e74705SXin Li std::shared_ptr<PCHContainerOperations> PCHContainerOps,
2323*67e74705SXin Li DiagnosticsEngine &Diag, LangOptions &LangOpts, SourceManager &SourceMgr,
2324*67e74705SXin Li FileManager &FileMgr, SmallVectorImpl<StoredDiagnostic> &StoredDiagnostics,
2325*67e74705SXin Li SmallVectorImpl<const llvm::MemoryBuffer *> &OwnedBuffers) {
2326*67e74705SXin Li if (!Invocation)
2327*67e74705SXin Li return;
2328*67e74705SXin Li
2329*67e74705SXin Li SimpleTimer CompletionTimer(WantTiming);
2330*67e74705SXin Li CompletionTimer.setOutput("Code completion @ " + File + ":" +
2331*67e74705SXin Li Twine(Line) + ":" + Twine(Column));
2332*67e74705SXin Li
2333*67e74705SXin Li IntrusiveRefCntPtr<CompilerInvocation>
2334*67e74705SXin Li CCInvocation(new CompilerInvocation(*Invocation));
2335*67e74705SXin Li
2336*67e74705SXin Li FrontendOptions &FrontendOpts = CCInvocation->getFrontendOpts();
2337*67e74705SXin Li CodeCompleteOptions &CodeCompleteOpts = FrontendOpts.CodeCompleteOpts;
2338*67e74705SXin Li PreprocessorOptions &PreprocessorOpts = CCInvocation->getPreprocessorOpts();
2339*67e74705SXin Li
2340*67e74705SXin Li CodeCompleteOpts.IncludeMacros = IncludeMacros &&
2341*67e74705SXin Li CachedCompletionResults.empty();
2342*67e74705SXin Li CodeCompleteOpts.IncludeCodePatterns = IncludeCodePatterns;
2343*67e74705SXin Li CodeCompleteOpts.IncludeGlobals = CachedCompletionResults.empty();
2344*67e74705SXin Li CodeCompleteOpts.IncludeBriefComments = IncludeBriefComments;
2345*67e74705SXin Li
2346*67e74705SXin Li assert(IncludeBriefComments == this->IncludeBriefCommentsInCodeCompletion);
2347*67e74705SXin Li
2348*67e74705SXin Li FrontendOpts.CodeCompletionAt.FileName = File;
2349*67e74705SXin Li FrontendOpts.CodeCompletionAt.Line = Line;
2350*67e74705SXin Li FrontendOpts.CodeCompletionAt.Column = Column;
2351*67e74705SXin Li
2352*67e74705SXin Li // Set the language options appropriately.
2353*67e74705SXin Li LangOpts = *CCInvocation->getLangOpts();
2354*67e74705SXin Li
2355*67e74705SXin Li // Spell-checking and warnings are wasteful during code-completion.
2356*67e74705SXin Li LangOpts.SpellChecking = false;
2357*67e74705SXin Li CCInvocation->getDiagnosticOpts().IgnoreWarnings = true;
2358*67e74705SXin Li
2359*67e74705SXin Li std::unique_ptr<CompilerInstance> Clang(
2360*67e74705SXin Li new CompilerInstance(PCHContainerOps));
2361*67e74705SXin Li
2362*67e74705SXin Li // Recover resources if we crash before exiting this method.
2363*67e74705SXin Li llvm::CrashRecoveryContextCleanupRegistrar<CompilerInstance>
2364*67e74705SXin Li CICleanup(Clang.get());
2365*67e74705SXin Li
2366*67e74705SXin Li Clang->setInvocation(&*CCInvocation);
2367*67e74705SXin Li OriginalSourceFile = Clang->getFrontendOpts().Inputs[0].getFile();
2368*67e74705SXin Li
2369*67e74705SXin Li // Set up diagnostics, capturing any diagnostics produced.
2370*67e74705SXin Li Clang->setDiagnostics(&Diag);
2371*67e74705SXin Li CaptureDroppedDiagnostics Capture(true,
2372*67e74705SXin Li Clang->getDiagnostics(),
2373*67e74705SXin Li StoredDiagnostics);
2374*67e74705SXin Li ProcessWarningOptions(Diag, CCInvocation->getDiagnosticOpts());
2375*67e74705SXin Li
2376*67e74705SXin Li // Create the target instance.
2377*67e74705SXin Li Clang->setTarget(TargetInfo::CreateTargetInfo(
2378*67e74705SXin Li Clang->getDiagnostics(), Clang->getInvocation().TargetOpts));
2379*67e74705SXin Li if (!Clang->hasTarget()) {
2380*67e74705SXin Li Clang->setInvocation(nullptr);
2381*67e74705SXin Li return;
2382*67e74705SXin Li }
2383*67e74705SXin Li
2384*67e74705SXin Li // Inform the target of the language options.
2385*67e74705SXin Li //
2386*67e74705SXin Li // FIXME: We shouldn't need to do this, the target should be immutable once
2387*67e74705SXin Li // created. This complexity should be lifted elsewhere.
2388*67e74705SXin Li Clang->getTarget().adjust(Clang->getLangOpts());
2389*67e74705SXin Li
2390*67e74705SXin Li assert(Clang->getFrontendOpts().Inputs.size() == 1 &&
2391*67e74705SXin Li "Invocation must have exactly one source file!");
2392*67e74705SXin Li assert(Clang->getFrontendOpts().Inputs[0].getKind() != IK_AST &&
2393*67e74705SXin Li "FIXME: AST inputs not yet supported here!");
2394*67e74705SXin Li assert(Clang->getFrontendOpts().Inputs[0].getKind() != IK_LLVM_IR &&
2395*67e74705SXin Li "IR inputs not support here!");
2396*67e74705SXin Li
2397*67e74705SXin Li
2398*67e74705SXin Li // Use the source and file managers that we were given.
2399*67e74705SXin Li Clang->setFileManager(&FileMgr);
2400*67e74705SXin Li Clang->setSourceManager(&SourceMgr);
2401*67e74705SXin Li
2402*67e74705SXin Li // Remap files.
2403*67e74705SXin Li PreprocessorOpts.clearRemappedFiles();
2404*67e74705SXin Li PreprocessorOpts.RetainRemappedFileBuffers = true;
2405*67e74705SXin Li for (const auto &RemappedFile : RemappedFiles) {
2406*67e74705SXin Li PreprocessorOpts.addRemappedFile(RemappedFile.first, RemappedFile.second);
2407*67e74705SXin Li OwnedBuffers.push_back(RemappedFile.second);
2408*67e74705SXin Li }
2409*67e74705SXin Li
2410*67e74705SXin Li // Use the code completion consumer we were given, but adding any cached
2411*67e74705SXin Li // code-completion results.
2412*67e74705SXin Li AugmentedCodeCompleteConsumer *AugmentedConsumer
2413*67e74705SXin Li = new AugmentedCodeCompleteConsumer(*this, Consumer, CodeCompleteOpts);
2414*67e74705SXin Li Clang->setCodeCompletionConsumer(AugmentedConsumer);
2415*67e74705SXin Li
2416*67e74705SXin Li // If we have a precompiled preamble, try to use it. We only allow
2417*67e74705SXin Li // the use of the precompiled preamble if we're if the completion
2418*67e74705SXin Li // point is within the main file, after the end of the precompiled
2419*67e74705SXin Li // preamble.
2420*67e74705SXin Li std::unique_ptr<llvm::MemoryBuffer> OverrideMainBuffer;
2421*67e74705SXin Li if (!getPreambleFile(this).empty()) {
2422*67e74705SXin Li std::string CompleteFilePath(File);
2423*67e74705SXin Li llvm::sys::fs::UniqueID CompleteFileID;
2424*67e74705SXin Li
2425*67e74705SXin Li if (!llvm::sys::fs::getUniqueID(CompleteFilePath, CompleteFileID)) {
2426*67e74705SXin Li std::string MainPath(OriginalSourceFile);
2427*67e74705SXin Li llvm::sys::fs::UniqueID MainID;
2428*67e74705SXin Li if (!llvm::sys::fs::getUniqueID(MainPath, MainID)) {
2429*67e74705SXin Li if (CompleteFileID == MainID && Line > 1)
2430*67e74705SXin Li OverrideMainBuffer = getMainBufferWithPrecompiledPreamble(
2431*67e74705SXin Li PCHContainerOps, *CCInvocation, false, Line - 1);
2432*67e74705SXin Li }
2433*67e74705SXin Li }
2434*67e74705SXin Li }
2435*67e74705SXin Li
2436*67e74705SXin Li // If the main file has been overridden due to the use of a preamble,
2437*67e74705SXin Li // make that override happen and introduce the preamble.
2438*67e74705SXin Li if (OverrideMainBuffer) {
2439*67e74705SXin Li PreprocessorOpts.addRemappedFile(OriginalSourceFile,
2440*67e74705SXin Li OverrideMainBuffer.get());
2441*67e74705SXin Li PreprocessorOpts.PrecompiledPreambleBytes.first = Preamble.size();
2442*67e74705SXin Li PreprocessorOpts.PrecompiledPreambleBytes.second
2443*67e74705SXin Li = PreambleEndsAtStartOfLine;
2444*67e74705SXin Li PreprocessorOpts.ImplicitPCHInclude = getPreambleFile(this);
2445*67e74705SXin Li PreprocessorOpts.DisablePCHValidation = true;
2446*67e74705SXin Li
2447*67e74705SXin Li OwnedBuffers.push_back(OverrideMainBuffer.release());
2448*67e74705SXin Li } else {
2449*67e74705SXin Li PreprocessorOpts.PrecompiledPreambleBytes.first = 0;
2450*67e74705SXin Li PreprocessorOpts.PrecompiledPreambleBytes.second = false;
2451*67e74705SXin Li }
2452*67e74705SXin Li
2453*67e74705SXin Li // Disable the preprocessing record if modules are not enabled.
2454*67e74705SXin Li if (!Clang->getLangOpts().Modules)
2455*67e74705SXin Li PreprocessorOpts.DetailedRecord = false;
2456*67e74705SXin Li
2457*67e74705SXin Li std::unique_ptr<SyntaxOnlyAction> Act;
2458*67e74705SXin Li Act.reset(new SyntaxOnlyAction);
2459*67e74705SXin Li if (Act->BeginSourceFile(*Clang.get(), Clang->getFrontendOpts().Inputs[0])) {
2460*67e74705SXin Li Act->Execute();
2461*67e74705SXin Li Act->EndSourceFile();
2462*67e74705SXin Li }
2463*67e74705SXin Li }
2464*67e74705SXin Li
Save(StringRef File)2465*67e74705SXin Li bool ASTUnit::Save(StringRef File) {
2466*67e74705SXin Li if (HadModuleLoaderFatalFailure)
2467*67e74705SXin Li return true;
2468*67e74705SXin Li
2469*67e74705SXin Li // Write to a temporary file and later rename it to the actual file, to avoid
2470*67e74705SXin Li // possible race conditions.
2471*67e74705SXin Li SmallString<128> TempPath;
2472*67e74705SXin Li TempPath = File;
2473*67e74705SXin Li TempPath += "-%%%%%%%%";
2474*67e74705SXin Li int fd;
2475*67e74705SXin Li if (llvm::sys::fs::createUniqueFile(TempPath, fd, TempPath))
2476*67e74705SXin Li return true;
2477*67e74705SXin Li
2478*67e74705SXin Li // FIXME: Can we somehow regenerate the stat cache here, or do we need to
2479*67e74705SXin Li // unconditionally create a stat cache when we parse the file?
2480*67e74705SXin Li llvm::raw_fd_ostream Out(fd, /*shouldClose=*/true);
2481*67e74705SXin Li
2482*67e74705SXin Li serialize(Out);
2483*67e74705SXin Li Out.close();
2484*67e74705SXin Li if (Out.has_error()) {
2485*67e74705SXin Li Out.clear_error();
2486*67e74705SXin Li return true;
2487*67e74705SXin Li }
2488*67e74705SXin Li
2489*67e74705SXin Li if (llvm::sys::fs::rename(TempPath, File)) {
2490*67e74705SXin Li llvm::sys::fs::remove(TempPath);
2491*67e74705SXin Li return true;
2492*67e74705SXin Li }
2493*67e74705SXin Li
2494*67e74705SXin Li return false;
2495*67e74705SXin Li }
2496*67e74705SXin Li
serializeUnit(ASTWriter & Writer,SmallVectorImpl<char> & Buffer,Sema & S,bool hasErrors,raw_ostream & OS)2497*67e74705SXin Li static bool serializeUnit(ASTWriter &Writer,
2498*67e74705SXin Li SmallVectorImpl<char> &Buffer,
2499*67e74705SXin Li Sema &S,
2500*67e74705SXin Li bool hasErrors,
2501*67e74705SXin Li raw_ostream &OS) {
2502*67e74705SXin Li Writer.WriteAST(S, std::string(), nullptr, "", hasErrors);
2503*67e74705SXin Li
2504*67e74705SXin Li // Write the generated bitstream to "Out".
2505*67e74705SXin Li if (!Buffer.empty())
2506*67e74705SXin Li OS.write(Buffer.data(), Buffer.size());
2507*67e74705SXin Li
2508*67e74705SXin Li return false;
2509*67e74705SXin Li }
2510*67e74705SXin Li
serialize(raw_ostream & OS)2511*67e74705SXin Li bool ASTUnit::serialize(raw_ostream &OS) {
2512*67e74705SXin Li // For serialization we are lenient if the errors were only warn-as-error kind.
2513*67e74705SXin Li bool hasErrors = getDiagnostics().hasUncompilableErrorOccurred();
2514*67e74705SXin Li
2515*67e74705SXin Li if (WriterData)
2516*67e74705SXin Li return serializeUnit(WriterData->Writer, WriterData->Buffer,
2517*67e74705SXin Li getSema(), hasErrors, OS);
2518*67e74705SXin Li
2519*67e74705SXin Li SmallString<128> Buffer;
2520*67e74705SXin Li llvm::BitstreamWriter Stream(Buffer);
2521*67e74705SXin Li ASTWriter Writer(Stream, { });
2522*67e74705SXin Li return serializeUnit(Writer, Buffer, getSema(), hasErrors, OS);
2523*67e74705SXin Li }
2524*67e74705SXin Li
2525*67e74705SXin Li typedef ContinuousRangeMap<unsigned, int, 2> SLocRemap;
2526*67e74705SXin Li
TranslateStoredDiagnostics(FileManager & FileMgr,SourceManager & SrcMgr,const SmallVectorImpl<StandaloneDiagnostic> & Diags,SmallVectorImpl<StoredDiagnostic> & Out)2527*67e74705SXin Li void ASTUnit::TranslateStoredDiagnostics(
2528*67e74705SXin Li FileManager &FileMgr,
2529*67e74705SXin Li SourceManager &SrcMgr,
2530*67e74705SXin Li const SmallVectorImpl<StandaloneDiagnostic> &Diags,
2531*67e74705SXin Li SmallVectorImpl<StoredDiagnostic> &Out) {
2532*67e74705SXin Li // Map the standalone diagnostic into the new source manager. We also need to
2533*67e74705SXin Li // remap all the locations to the new view. This includes the diag location,
2534*67e74705SXin Li // any associated source ranges, and the source ranges of associated fix-its.
2535*67e74705SXin Li // FIXME: There should be a cleaner way to do this.
2536*67e74705SXin Li
2537*67e74705SXin Li SmallVector<StoredDiagnostic, 4> Result;
2538*67e74705SXin Li Result.reserve(Diags.size());
2539*67e74705SXin Li for (const StandaloneDiagnostic &SD : Diags) {
2540*67e74705SXin Li // Rebuild the StoredDiagnostic.
2541*67e74705SXin Li if (SD.Filename.empty())
2542*67e74705SXin Li continue;
2543*67e74705SXin Li const FileEntry *FE = FileMgr.getFile(SD.Filename);
2544*67e74705SXin Li if (!FE)
2545*67e74705SXin Li continue;
2546*67e74705SXin Li FileID FID = SrcMgr.translateFile(FE);
2547*67e74705SXin Li SourceLocation FileLoc = SrcMgr.getLocForStartOfFile(FID);
2548*67e74705SXin Li if (FileLoc.isInvalid())
2549*67e74705SXin Li continue;
2550*67e74705SXin Li SourceLocation L = FileLoc.getLocWithOffset(SD.LocOffset);
2551*67e74705SXin Li FullSourceLoc Loc(L, SrcMgr);
2552*67e74705SXin Li
2553*67e74705SXin Li SmallVector<CharSourceRange, 4> Ranges;
2554*67e74705SXin Li Ranges.reserve(SD.Ranges.size());
2555*67e74705SXin Li for (const auto &Range : SD.Ranges) {
2556*67e74705SXin Li SourceLocation BL = FileLoc.getLocWithOffset(Range.first);
2557*67e74705SXin Li SourceLocation EL = FileLoc.getLocWithOffset(Range.second);
2558*67e74705SXin Li Ranges.push_back(CharSourceRange::getCharRange(BL, EL));
2559*67e74705SXin Li }
2560*67e74705SXin Li
2561*67e74705SXin Li SmallVector<FixItHint, 2> FixIts;
2562*67e74705SXin Li FixIts.reserve(SD.FixIts.size());
2563*67e74705SXin Li for (const StandaloneFixIt &FixIt : SD.FixIts) {
2564*67e74705SXin Li FixIts.push_back(FixItHint());
2565*67e74705SXin Li FixItHint &FH = FixIts.back();
2566*67e74705SXin Li FH.CodeToInsert = FixIt.CodeToInsert;
2567*67e74705SXin Li SourceLocation BL = FileLoc.getLocWithOffset(FixIt.RemoveRange.first);
2568*67e74705SXin Li SourceLocation EL = FileLoc.getLocWithOffset(FixIt.RemoveRange.second);
2569*67e74705SXin Li FH.RemoveRange = CharSourceRange::getCharRange(BL, EL);
2570*67e74705SXin Li }
2571*67e74705SXin Li
2572*67e74705SXin Li Result.push_back(StoredDiagnostic(SD.Level, SD.ID,
2573*67e74705SXin Li SD.Message, Loc, Ranges, FixIts));
2574*67e74705SXin Li }
2575*67e74705SXin Li Result.swap(Out);
2576*67e74705SXin Li }
2577*67e74705SXin Li
addFileLevelDecl(Decl * D)2578*67e74705SXin Li void ASTUnit::addFileLevelDecl(Decl *D) {
2579*67e74705SXin Li assert(D);
2580*67e74705SXin Li
2581*67e74705SXin Li // We only care about local declarations.
2582*67e74705SXin Li if (D->isFromASTFile())
2583*67e74705SXin Li return;
2584*67e74705SXin Li
2585*67e74705SXin Li SourceManager &SM = *SourceMgr;
2586*67e74705SXin Li SourceLocation Loc = D->getLocation();
2587*67e74705SXin Li if (Loc.isInvalid() || !SM.isLocalSourceLocation(Loc))
2588*67e74705SXin Li return;
2589*67e74705SXin Li
2590*67e74705SXin Li // We only keep track of the file-level declarations of each file.
2591*67e74705SXin Li if (!D->getLexicalDeclContext()->isFileContext())
2592*67e74705SXin Li return;
2593*67e74705SXin Li
2594*67e74705SXin Li SourceLocation FileLoc = SM.getFileLoc(Loc);
2595*67e74705SXin Li assert(SM.isLocalSourceLocation(FileLoc));
2596*67e74705SXin Li FileID FID;
2597*67e74705SXin Li unsigned Offset;
2598*67e74705SXin Li std::tie(FID, Offset) = SM.getDecomposedLoc(FileLoc);
2599*67e74705SXin Li if (FID.isInvalid())
2600*67e74705SXin Li return;
2601*67e74705SXin Li
2602*67e74705SXin Li LocDeclsTy *&Decls = FileDecls[FID];
2603*67e74705SXin Li if (!Decls)
2604*67e74705SXin Li Decls = new LocDeclsTy();
2605*67e74705SXin Li
2606*67e74705SXin Li std::pair<unsigned, Decl *> LocDecl(Offset, D);
2607*67e74705SXin Li
2608*67e74705SXin Li if (Decls->empty() || Decls->back().first <= Offset) {
2609*67e74705SXin Li Decls->push_back(LocDecl);
2610*67e74705SXin Li return;
2611*67e74705SXin Li }
2612*67e74705SXin Li
2613*67e74705SXin Li LocDeclsTy::iterator I = std::upper_bound(Decls->begin(), Decls->end(),
2614*67e74705SXin Li LocDecl, llvm::less_first());
2615*67e74705SXin Li
2616*67e74705SXin Li Decls->insert(I, LocDecl);
2617*67e74705SXin Li }
2618*67e74705SXin Li
findFileRegionDecls(FileID File,unsigned Offset,unsigned Length,SmallVectorImpl<Decl * > & Decls)2619*67e74705SXin Li void ASTUnit::findFileRegionDecls(FileID File, unsigned Offset, unsigned Length,
2620*67e74705SXin Li SmallVectorImpl<Decl *> &Decls) {
2621*67e74705SXin Li if (File.isInvalid())
2622*67e74705SXin Li return;
2623*67e74705SXin Li
2624*67e74705SXin Li if (SourceMgr->isLoadedFileID(File)) {
2625*67e74705SXin Li assert(Ctx->getExternalSource() && "No external source!");
2626*67e74705SXin Li return Ctx->getExternalSource()->FindFileRegionDecls(File, Offset, Length,
2627*67e74705SXin Li Decls);
2628*67e74705SXin Li }
2629*67e74705SXin Li
2630*67e74705SXin Li FileDeclsTy::iterator I = FileDecls.find(File);
2631*67e74705SXin Li if (I == FileDecls.end())
2632*67e74705SXin Li return;
2633*67e74705SXin Li
2634*67e74705SXin Li LocDeclsTy &LocDecls = *I->second;
2635*67e74705SXin Li if (LocDecls.empty())
2636*67e74705SXin Li return;
2637*67e74705SXin Li
2638*67e74705SXin Li LocDeclsTy::iterator BeginIt =
2639*67e74705SXin Li std::lower_bound(LocDecls.begin(), LocDecls.end(),
2640*67e74705SXin Li std::make_pair(Offset, (Decl *)nullptr),
2641*67e74705SXin Li llvm::less_first());
2642*67e74705SXin Li if (BeginIt != LocDecls.begin())
2643*67e74705SXin Li --BeginIt;
2644*67e74705SXin Li
2645*67e74705SXin Li // If we are pointing at a top-level decl inside an objc container, we need
2646*67e74705SXin Li // to backtrack until we find it otherwise we will fail to report that the
2647*67e74705SXin Li // region overlaps with an objc container.
2648*67e74705SXin Li while (BeginIt != LocDecls.begin() &&
2649*67e74705SXin Li BeginIt->second->isTopLevelDeclInObjCContainer())
2650*67e74705SXin Li --BeginIt;
2651*67e74705SXin Li
2652*67e74705SXin Li LocDeclsTy::iterator EndIt = std::upper_bound(
2653*67e74705SXin Li LocDecls.begin(), LocDecls.end(),
2654*67e74705SXin Li std::make_pair(Offset + Length, (Decl *)nullptr), llvm::less_first());
2655*67e74705SXin Li if (EndIt != LocDecls.end())
2656*67e74705SXin Li ++EndIt;
2657*67e74705SXin Li
2658*67e74705SXin Li for (LocDeclsTy::iterator DIt = BeginIt; DIt != EndIt; ++DIt)
2659*67e74705SXin Li Decls.push_back(DIt->second);
2660*67e74705SXin Li }
2661*67e74705SXin Li
getLocation(const FileEntry * File,unsigned Line,unsigned Col) const2662*67e74705SXin Li SourceLocation ASTUnit::getLocation(const FileEntry *File,
2663*67e74705SXin Li unsigned Line, unsigned Col) const {
2664*67e74705SXin Li const SourceManager &SM = getSourceManager();
2665*67e74705SXin Li SourceLocation Loc = SM.translateFileLineCol(File, Line, Col);
2666*67e74705SXin Li return SM.getMacroArgExpandedLocation(Loc);
2667*67e74705SXin Li }
2668*67e74705SXin Li
getLocation(const FileEntry * File,unsigned Offset) const2669*67e74705SXin Li SourceLocation ASTUnit::getLocation(const FileEntry *File,
2670*67e74705SXin Li unsigned Offset) const {
2671*67e74705SXin Li const SourceManager &SM = getSourceManager();
2672*67e74705SXin Li SourceLocation FileLoc = SM.translateFileLineCol(File, 1, 1);
2673*67e74705SXin Li return SM.getMacroArgExpandedLocation(FileLoc.getLocWithOffset(Offset));
2674*67e74705SXin Li }
2675*67e74705SXin Li
2676*67e74705SXin Li /// \brief If \arg Loc is a loaded location from the preamble, returns
2677*67e74705SXin Li /// the corresponding local location of the main file, otherwise it returns
2678*67e74705SXin Li /// \arg Loc.
mapLocationFromPreamble(SourceLocation Loc)2679*67e74705SXin Li SourceLocation ASTUnit::mapLocationFromPreamble(SourceLocation Loc) {
2680*67e74705SXin Li FileID PreambleID;
2681*67e74705SXin Li if (SourceMgr)
2682*67e74705SXin Li PreambleID = SourceMgr->getPreambleFileID();
2683*67e74705SXin Li
2684*67e74705SXin Li if (Loc.isInvalid() || Preamble.empty() || PreambleID.isInvalid())
2685*67e74705SXin Li return Loc;
2686*67e74705SXin Li
2687*67e74705SXin Li unsigned Offs;
2688*67e74705SXin Li if (SourceMgr->isInFileID(Loc, PreambleID, &Offs) && Offs < Preamble.size()) {
2689*67e74705SXin Li SourceLocation FileLoc
2690*67e74705SXin Li = SourceMgr->getLocForStartOfFile(SourceMgr->getMainFileID());
2691*67e74705SXin Li return FileLoc.getLocWithOffset(Offs);
2692*67e74705SXin Li }
2693*67e74705SXin Li
2694*67e74705SXin Li return Loc;
2695*67e74705SXin Li }
2696*67e74705SXin Li
2697*67e74705SXin Li /// \brief If \arg Loc is a local location of the main file but inside the
2698*67e74705SXin Li /// preamble chunk, returns the corresponding loaded location from the
2699*67e74705SXin Li /// preamble, otherwise it returns \arg Loc.
mapLocationToPreamble(SourceLocation Loc)2700*67e74705SXin Li SourceLocation ASTUnit::mapLocationToPreamble(SourceLocation Loc) {
2701*67e74705SXin Li FileID PreambleID;
2702*67e74705SXin Li if (SourceMgr)
2703*67e74705SXin Li PreambleID = SourceMgr->getPreambleFileID();
2704*67e74705SXin Li
2705*67e74705SXin Li if (Loc.isInvalid() || Preamble.empty() || PreambleID.isInvalid())
2706*67e74705SXin Li return Loc;
2707*67e74705SXin Li
2708*67e74705SXin Li unsigned Offs;
2709*67e74705SXin Li if (SourceMgr->isInFileID(Loc, SourceMgr->getMainFileID(), &Offs) &&
2710*67e74705SXin Li Offs < Preamble.size()) {
2711*67e74705SXin Li SourceLocation FileLoc = SourceMgr->getLocForStartOfFile(PreambleID);
2712*67e74705SXin Li return FileLoc.getLocWithOffset(Offs);
2713*67e74705SXin Li }
2714*67e74705SXin Li
2715*67e74705SXin Li return Loc;
2716*67e74705SXin Li }
2717*67e74705SXin Li
isInPreambleFileID(SourceLocation Loc)2718*67e74705SXin Li bool ASTUnit::isInPreambleFileID(SourceLocation Loc) {
2719*67e74705SXin Li FileID FID;
2720*67e74705SXin Li if (SourceMgr)
2721*67e74705SXin Li FID = SourceMgr->getPreambleFileID();
2722*67e74705SXin Li
2723*67e74705SXin Li if (Loc.isInvalid() || FID.isInvalid())
2724*67e74705SXin Li return false;
2725*67e74705SXin Li
2726*67e74705SXin Li return SourceMgr->isInFileID(Loc, FID);
2727*67e74705SXin Li }
2728*67e74705SXin Li
isInMainFileID(SourceLocation Loc)2729*67e74705SXin Li bool ASTUnit::isInMainFileID(SourceLocation Loc) {
2730*67e74705SXin Li FileID FID;
2731*67e74705SXin Li if (SourceMgr)
2732*67e74705SXin Li FID = SourceMgr->getMainFileID();
2733*67e74705SXin Li
2734*67e74705SXin Li if (Loc.isInvalid() || FID.isInvalid())
2735*67e74705SXin Li return false;
2736*67e74705SXin Li
2737*67e74705SXin Li return SourceMgr->isInFileID(Loc, FID);
2738*67e74705SXin Li }
2739*67e74705SXin Li
getEndOfPreambleFileID()2740*67e74705SXin Li SourceLocation ASTUnit::getEndOfPreambleFileID() {
2741*67e74705SXin Li FileID FID;
2742*67e74705SXin Li if (SourceMgr)
2743*67e74705SXin Li FID = SourceMgr->getPreambleFileID();
2744*67e74705SXin Li
2745*67e74705SXin Li if (FID.isInvalid())
2746*67e74705SXin Li return SourceLocation();
2747*67e74705SXin Li
2748*67e74705SXin Li return SourceMgr->getLocForEndOfFile(FID);
2749*67e74705SXin Li }
2750*67e74705SXin Li
getStartOfMainFileID()2751*67e74705SXin Li SourceLocation ASTUnit::getStartOfMainFileID() {
2752*67e74705SXin Li FileID FID;
2753*67e74705SXin Li if (SourceMgr)
2754*67e74705SXin Li FID = SourceMgr->getMainFileID();
2755*67e74705SXin Li
2756*67e74705SXin Li if (FID.isInvalid())
2757*67e74705SXin Li return SourceLocation();
2758*67e74705SXin Li
2759*67e74705SXin Li return SourceMgr->getLocForStartOfFile(FID);
2760*67e74705SXin Li }
2761*67e74705SXin Li
2762*67e74705SXin Li llvm::iterator_range<PreprocessingRecord::iterator>
getLocalPreprocessingEntities() const2763*67e74705SXin Li ASTUnit::getLocalPreprocessingEntities() const {
2764*67e74705SXin Li if (isMainFileAST()) {
2765*67e74705SXin Li serialization::ModuleFile &
2766*67e74705SXin Li Mod = Reader->getModuleManager().getPrimaryModule();
2767*67e74705SXin Li return Reader->getModulePreprocessedEntities(Mod);
2768*67e74705SXin Li }
2769*67e74705SXin Li
2770*67e74705SXin Li if (PreprocessingRecord *PPRec = PP->getPreprocessingRecord())
2771*67e74705SXin Li return llvm::make_range(PPRec->local_begin(), PPRec->local_end());
2772*67e74705SXin Li
2773*67e74705SXin Li return llvm::make_range(PreprocessingRecord::iterator(),
2774*67e74705SXin Li PreprocessingRecord::iterator());
2775*67e74705SXin Li }
2776*67e74705SXin Li
visitLocalTopLevelDecls(void * context,DeclVisitorFn Fn)2777*67e74705SXin Li bool ASTUnit::visitLocalTopLevelDecls(void *context, DeclVisitorFn Fn) {
2778*67e74705SXin Li if (isMainFileAST()) {
2779*67e74705SXin Li serialization::ModuleFile &
2780*67e74705SXin Li Mod = Reader->getModuleManager().getPrimaryModule();
2781*67e74705SXin Li for (const Decl *D : Reader->getModuleFileLevelDecls(Mod)) {
2782*67e74705SXin Li if (!Fn(context, D))
2783*67e74705SXin Li return false;
2784*67e74705SXin Li }
2785*67e74705SXin Li
2786*67e74705SXin Li return true;
2787*67e74705SXin Li }
2788*67e74705SXin Li
2789*67e74705SXin Li for (ASTUnit::top_level_iterator TL = top_level_begin(),
2790*67e74705SXin Li TLEnd = top_level_end();
2791*67e74705SXin Li TL != TLEnd; ++TL) {
2792*67e74705SXin Li if (!Fn(context, *TL))
2793*67e74705SXin Li return false;
2794*67e74705SXin Li }
2795*67e74705SXin Li
2796*67e74705SXin Li return true;
2797*67e74705SXin Li }
2798*67e74705SXin Li
getPCHFile()2799*67e74705SXin Li const FileEntry *ASTUnit::getPCHFile() {
2800*67e74705SXin Li if (!Reader)
2801*67e74705SXin Li return nullptr;
2802*67e74705SXin Li
2803*67e74705SXin Li serialization::ModuleFile *Mod = nullptr;
2804*67e74705SXin Li Reader->getModuleManager().visit([&Mod](serialization::ModuleFile &M) {
2805*67e74705SXin Li switch (M.Kind) {
2806*67e74705SXin Li case serialization::MK_ImplicitModule:
2807*67e74705SXin Li case serialization::MK_ExplicitModule:
2808*67e74705SXin Li return true; // skip dependencies.
2809*67e74705SXin Li case serialization::MK_PCH:
2810*67e74705SXin Li Mod = &M;
2811*67e74705SXin Li return true; // found it.
2812*67e74705SXin Li case serialization::MK_Preamble:
2813*67e74705SXin Li return false; // look in dependencies.
2814*67e74705SXin Li case serialization::MK_MainFile:
2815*67e74705SXin Li return false; // look in dependencies.
2816*67e74705SXin Li }
2817*67e74705SXin Li
2818*67e74705SXin Li return true;
2819*67e74705SXin Li });
2820*67e74705SXin Li if (Mod)
2821*67e74705SXin Li return Mod->File;
2822*67e74705SXin Li
2823*67e74705SXin Li return nullptr;
2824*67e74705SXin Li }
2825*67e74705SXin Li
isModuleFile()2826*67e74705SXin Li bool ASTUnit::isModuleFile() {
2827*67e74705SXin Li return isMainFileAST() && ASTFileLangOpts.CompilingModule;
2828*67e74705SXin Li }
2829*67e74705SXin Li
countLines() const2830*67e74705SXin Li void ASTUnit::PreambleData::countLines() const {
2831*67e74705SXin Li NumLines = 0;
2832*67e74705SXin Li if (empty())
2833*67e74705SXin Li return;
2834*67e74705SXin Li
2835*67e74705SXin Li NumLines = std::count(Buffer.begin(), Buffer.end(), '\n');
2836*67e74705SXin Li
2837*67e74705SXin Li if (Buffer.back() != '\n')
2838*67e74705SXin Li ++NumLines;
2839*67e74705SXin Li }
2840*67e74705SXin Li
2841*67e74705SXin Li #ifndef NDEBUG
ConcurrencyState()2842*67e74705SXin Li ASTUnit::ConcurrencyState::ConcurrencyState() {
2843*67e74705SXin Li Mutex = new llvm::sys::MutexImpl(/*recursive=*/true);
2844*67e74705SXin Li }
2845*67e74705SXin Li
~ConcurrencyState()2846*67e74705SXin Li ASTUnit::ConcurrencyState::~ConcurrencyState() {
2847*67e74705SXin Li delete static_cast<llvm::sys::MutexImpl *>(Mutex);
2848*67e74705SXin Li }
2849*67e74705SXin Li
start()2850*67e74705SXin Li void ASTUnit::ConcurrencyState::start() {
2851*67e74705SXin Li bool acquired = static_cast<llvm::sys::MutexImpl *>(Mutex)->tryacquire();
2852*67e74705SXin Li assert(acquired && "Concurrent access to ASTUnit!");
2853*67e74705SXin Li }
2854*67e74705SXin Li
finish()2855*67e74705SXin Li void ASTUnit::ConcurrencyState::finish() {
2856*67e74705SXin Li static_cast<llvm::sys::MutexImpl *>(Mutex)->release();
2857*67e74705SXin Li }
2858*67e74705SXin Li
2859*67e74705SXin Li #else // NDEBUG
2860*67e74705SXin Li
ConcurrencyState()2861*67e74705SXin Li ASTUnit::ConcurrencyState::ConcurrencyState() { Mutex = nullptr; }
~ConcurrencyState()2862*67e74705SXin Li ASTUnit::ConcurrencyState::~ConcurrencyState() {}
start()2863*67e74705SXin Li void ASTUnit::ConcurrencyState::start() {}
finish()2864*67e74705SXin Li void ASTUnit::ConcurrencyState::finish() {}
2865*67e74705SXin Li
2866*67e74705SXin Li #endif // NDEBUG
2867