1 | //===--------- IncrementalParser.cpp - Incremental Compilation -----------===// |
2 | // |
3 | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
4 | // See https://llvm.org/LICENSE.txt for license information. |
5 | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
6 | // |
7 | //===----------------------------------------------------------------------===// |
8 | // |
9 | // This file implements the class which performs incremental code compilation. |
10 | // |
11 | //===----------------------------------------------------------------------===// |
12 | |
13 | #include "IncrementalParser.h" |
14 | |
15 | #include "clang/AST/DeclContextInternals.h" |
16 | #include "clang/Frontend/CompilerInstance.h" |
17 | #include "clang/Interpreter/PartialTranslationUnit.h" |
18 | #include "clang/Parse/Parser.h" |
19 | #include "clang/Sema/Sema.h" |
20 | #include "llvm/Support/CrashRecoveryContext.h" |
21 | #include "llvm/Support/Error.h" |
22 | |
23 | #include <sstream> |
24 | |
25 | namespace clang { |
26 | |
27 | // IncrementalParser::IncrementalParser() {} |
28 | |
29 | IncrementalParser::IncrementalParser(CompilerInstance &Instance, |
30 | llvm::Error &Err) |
31 | : S(Instance.getSema()) { |
32 | llvm::ErrorAsOutParameter EAO(&Err); |
33 | Consumer = &S.getASTConsumer(); |
34 | P.reset(p: new Parser(S.getPreprocessor(), S, /*SkipBodies=*/false)); |
35 | P->Initialize(); |
36 | } |
37 | |
38 | IncrementalParser::~IncrementalParser() { P.reset(); } |
39 | |
40 | llvm::Expected<TranslationUnitDecl *> |
41 | IncrementalParser::ParseOrWrapTopLevelDecl() { |
42 | // Recover resources if we crash before exiting this method. |
43 | llvm::CrashRecoveryContextCleanupRegistrar<Sema> CleanupSema(&S); |
44 | Sema::GlobalEagerInstantiationScope GlobalInstantiations(S, /*Enabled=*/true, |
45 | /*AtEndOfTU=*/true); |
46 | Sema::LocalEagerInstantiationScope LocalInstantiations(S, /*AtEndOfTU=*/true); |
47 | |
48 | // Add a new PTU. |
49 | ASTContext &C = S.getASTContext(); |
50 | C.addTranslationUnitDecl(); |
51 | |
52 | // Skip previous eof due to last incremental input. |
53 | if (P->getCurToken().is(K: tok::annot_repl_input_end)) { |
54 | P->ConsumeAnyToken(); |
55 | // FIXME: Clang does not call ExitScope on finalizing the regular TU, we |
56 | // might want to do that around HandleEndOfTranslationUnit. |
57 | P->ExitScope(); |
58 | S.CurContext = nullptr; |
59 | // Start a new PTU. |
60 | P->EnterScope(ScopeFlags: Scope::DeclScope); |
61 | S.ActOnTranslationUnitScope(S: P->getCurScope()); |
62 | } |
63 | |
64 | Parser::DeclGroupPtrTy ADecl; |
65 | Sema::ModuleImportState ImportState; |
66 | for (bool AtEOF = P->ParseFirstTopLevelDecl(Result&: ADecl, ImportState); !AtEOF; |
67 | AtEOF = P->ParseTopLevelDecl(Result&: ADecl, ImportState)) { |
68 | if (ADecl && !Consumer->HandleTopLevelDecl(D: ADecl.get())) |
69 | return llvm::make_error<llvm::StringError>(Args: "Parsing failed. " |
70 | "The consumer rejected a decl" , |
71 | Args: std::error_code()); |
72 | } |
73 | |
74 | DiagnosticsEngine &Diags = S.getDiagnostics(); |
75 | if (Diags.hasErrorOccurred()) { |
76 | CleanUpPTU(MostRecentTU: C.getTranslationUnitDecl()); |
77 | |
78 | Diags.Reset(/*soft=*/true); |
79 | Diags.getClient()->clear(); |
80 | return llvm::make_error<llvm::StringError>(Args: "Parsing failed." , |
81 | Args: std::error_code()); |
82 | } |
83 | |
84 | // Process any TopLevelDecls generated by #pragma weak. |
85 | for (Decl *D : S.WeakTopLevelDecls()) { |
86 | DeclGroupRef DGR(D); |
87 | Consumer->HandleTopLevelDecl(D: DGR); |
88 | } |
89 | |
90 | LocalInstantiations.perform(); |
91 | GlobalInstantiations.perform(); |
92 | |
93 | Consumer->HandleTranslationUnit(Ctx&: C); |
94 | |
95 | return C.getTranslationUnitDecl(); |
96 | } |
97 | |
98 | llvm::Expected<TranslationUnitDecl *> |
99 | IncrementalParser::Parse(llvm::StringRef input) { |
100 | Preprocessor &PP = S.getPreprocessor(); |
101 | assert(PP.isIncrementalProcessingEnabled() && "Not in incremental mode!?" ); |
102 | |
103 | std::ostringstream SourceName; |
104 | SourceName << "input_line_" << InputCount++; |
105 | |
106 | // Create an uninitialized memory buffer, copy code in and append "\n" |
107 | size_t InputSize = input.size(); // don't include trailing 0 |
108 | // MemBuffer size should *not* include terminating zero |
109 | std::unique_ptr<llvm::MemoryBuffer> MB( |
110 | llvm::WritableMemoryBuffer::getNewUninitMemBuffer(Size: InputSize + 1, |
111 | BufferName: SourceName.str())); |
112 | char *MBStart = const_cast<char *>(MB->getBufferStart()); |
113 | memcpy(dest: MBStart, src: input.data(), n: InputSize); |
114 | MBStart[InputSize] = '\n'; |
115 | |
116 | SourceManager &SM = S.getSourceManager(); |
117 | |
118 | // FIXME: Create SourceLocation, which will allow clang to order the overload |
119 | // candidates for example |
120 | SourceLocation NewLoc = SM.getLocForStartOfFile(FID: SM.getMainFileID()); |
121 | |
122 | // Create FileID for the current buffer. |
123 | FileID FID = SM.createFileID(Buffer: std::move(MB), FileCharacter: SrcMgr::C_User, /*LoadedID=*/0, |
124 | /*LoadedOffset=*/0, IncludeLoc: NewLoc); |
125 | |
126 | // NewLoc only used for diags. |
127 | if (PP.EnterSourceFile(FID, /*DirLookup=*/Dir: nullptr, Loc: NewLoc)) |
128 | return llvm::make_error<llvm::StringError>(Args: "Parsing failed. " |
129 | "Cannot enter source file." , |
130 | Args: std::error_code()); |
131 | |
132 | auto PTU = ParseOrWrapTopLevelDecl(); |
133 | if (!PTU) |
134 | return PTU.takeError(); |
135 | |
136 | if (PP.getLangOpts().DelayedTemplateParsing) { |
137 | // Microsoft-specific: |
138 | // Late parsed templates can leave unswallowed "macro"-like tokens. |
139 | // They will seriously confuse the Parser when entering the next |
140 | // source file. So lex until we are EOF. |
141 | Token Tok; |
142 | do { |
143 | PP.Lex(Result&: Tok); |
144 | } while (Tok.isNot(K: tok::annot_repl_input_end)); |
145 | } else { |
146 | Token AssertTok; |
147 | PP.Lex(Result&: AssertTok); |
148 | assert(AssertTok.is(tok::annot_repl_input_end) && |
149 | "Lexer must be EOF when starting incremental parse!" ); |
150 | } |
151 | |
152 | return PTU; |
153 | } |
154 | |
155 | void IncrementalParser::CleanUpPTU(TranslationUnitDecl *MostRecentTU) { |
156 | if (StoredDeclsMap *Map = MostRecentTU->getPrimaryContext()->getLookupPtr()) { |
157 | for (auto &&[Key, List] : *Map) { |
158 | DeclContextLookupResult R = List.getLookupResult(); |
159 | std::vector<NamedDecl *> NamedDeclsToRemove; |
160 | bool RemoveAll = true; |
161 | for (NamedDecl *D : R) { |
162 | if (D->getTranslationUnitDecl() == MostRecentTU) |
163 | NamedDeclsToRemove.push_back(x: D); |
164 | else |
165 | RemoveAll = false; |
166 | } |
167 | if (LLVM_LIKELY(RemoveAll)) { |
168 | Map->erase(Val: Key); |
169 | } else { |
170 | for (NamedDecl *D : NamedDeclsToRemove) |
171 | List.remove(D); |
172 | } |
173 | } |
174 | } |
175 | |
176 | // FIXME: We should de-allocate MostRecentTU |
177 | for (Decl *D : MostRecentTU->decls()) { |
178 | auto *ND = dyn_cast<NamedDecl>(Val: D); |
179 | if (!ND || ND->getDeclName().isEmpty()) |
180 | continue; |
181 | // Check if we need to clean up the IdResolver chain. |
182 | if (ND->getDeclName().getFETokenInfo() && !D->getLangOpts().ObjC && |
183 | !D->getLangOpts().CPlusPlus) |
184 | S.IdResolver.RemoveDecl(D: ND); |
185 | } |
186 | } |
187 | |
188 | } // end namespace clang |
189 | |