ChainedIncludesSource.cpp 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295
  1. //===- ChainedIncludesSource.cpp - Chained PCHs in Memory -------*- C++ -*-===//
  2. //
  3. // The LLVM Compiler Infrastructure
  4. //
  5. // This file is distributed under the University of Illinois Open Source
  6. // License. See LICENSE.TXT for details.
  7. //
  8. //===----------------------------------------------------------------------===//
  9. //
  10. // This file defines the ChainedIncludesSource class, which converts headers
  11. // to chained PCHs in memory, mainly used for testing.
  12. //
  13. //===----------------------------------------------------------------------===//
  14. #include "clang/Basic/TargetInfo.h"
  15. #include "clang/Frontend/ASTUnit.h"
  16. #include "clang/Frontend/CompilerInstance.h"
  17. #include "clang/Frontend/TextDiagnosticPrinter.h"
  18. #include "clang/Lex/Preprocessor.h"
  19. #include "clang/Parse/ParseAST.h"
  20. #include "clang/Serialization/ASTReader.h"
  21. #include "clang/Serialization/ASTWriter.h"
  22. #include "llvm/Support/MemoryBuffer.h"
  23. using namespace clang;
  24. namespace {
  25. class ChainedIncludesSource : public ExternalSemaSource {
  26. public:
  27. ~ChainedIncludesSource() override;
  28. ExternalSemaSource &getFinalReader() const { return *FinalReader; }
  29. std::vector<CompilerInstance *> CIs;
  30. IntrusiveRefCntPtr<ExternalSemaSource> FinalReader;
  31. protected:
  32. //===----------------------------------------------------------------------===//
  33. // ExternalASTSource interface.
  34. //===----------------------------------------------------------------------===//
  35. Decl *GetExternalDecl(uint32_t ID) override;
  36. Selector GetExternalSelector(uint32_t ID) override;
  37. uint32_t GetNumExternalSelectors() override;
  38. Stmt *GetExternalDeclStmt(uint64_t Offset) override;
  39. CXXCtorInitializer **GetExternalCXXCtorInitializers(uint64_t Offset) override;
  40. CXXBaseSpecifier *GetExternalCXXBaseSpecifiers(uint64_t Offset) override;
  41. bool FindExternalVisibleDeclsByName(const DeclContext *DC,
  42. DeclarationName Name) override;
  43. void
  44. FindExternalLexicalDecls(const DeclContext *DC,
  45. llvm::function_ref<bool(Decl::Kind)> IsKindWeWant,
  46. SmallVectorImpl<Decl *> &Result) override;
  47. void CompleteType(TagDecl *Tag) override;
  48. void CompleteType(ObjCInterfaceDecl *Class) override;
  49. void StartedDeserializing() override;
  50. void FinishedDeserializing() override;
  51. void StartTranslationUnit(ASTConsumer *Consumer) override;
  52. void PrintStats() override;
  53. /// Return the amount of memory used by memory buffers, breaking down
  54. /// by heap-backed versus mmap'ed memory.
  55. void getMemoryBufferSizes(MemoryBufferSizes &sizes) const override;
  56. //===----------------------------------------------------------------------===//
  57. // ExternalSemaSource interface.
  58. //===----------------------------------------------------------------------===//
  59. void InitializeSema(Sema &S) override;
  60. void ForgetSema() override;
  61. void ReadMethodPool(Selector Sel) override;
  62. bool LookupUnqualified(LookupResult &R, Scope *S) override;
  63. };
  64. }
  65. static ASTReader *
  66. createASTReader(CompilerInstance &CI, StringRef pchFile,
  67. SmallVectorImpl<std::unique_ptr<llvm::MemoryBuffer>> &MemBufs,
  68. SmallVectorImpl<std::string> &bufNames,
  69. ASTDeserializationListener *deserialListener = nullptr) {
  70. Preprocessor &PP = CI.getPreprocessor();
  71. std::unique_ptr<ASTReader> Reader;
  72. Reader.reset(new ASTReader(PP, CI.getASTContext(),
  73. CI.getPCHContainerReader(),
  74. /*isysroot=*/"", /*DisableValidation=*/true));
  75. for (unsigned ti = 0; ti < bufNames.size(); ++ti) {
  76. StringRef sr(bufNames[ti]);
  77. Reader->addInMemoryBuffer(sr, std::move(MemBufs[ti]));
  78. }
  79. Reader->setDeserializationListener(deserialListener);
  80. switch (Reader->ReadAST(pchFile, serialization::MK_PCH, SourceLocation(),
  81. ASTReader::ARR_None)) {
  82. case ASTReader::Success:
  83. // Set the predefines buffer as suggested by the PCH reader.
  84. PP.setPredefines(Reader->getSuggestedPredefines());
  85. return Reader.release();
  86. case ASTReader::Failure:
  87. case ASTReader::Missing:
  88. case ASTReader::OutOfDate:
  89. case ASTReader::VersionMismatch:
  90. case ASTReader::ConfigurationMismatch:
  91. case ASTReader::HadErrors:
  92. break;
  93. }
  94. return nullptr;
  95. }
  96. ChainedIncludesSource::~ChainedIncludesSource() {
  97. for (unsigned i = 0, e = CIs.size(); i != e; ++i)
  98. delete CIs[i];
  99. }
  100. IntrusiveRefCntPtr<ExternalSemaSource> clang::createChainedIncludesSource(
  101. CompilerInstance &CI, IntrusiveRefCntPtr<ExternalSemaSource> &Reader) {
  102. std::vector<std::string> &includes = CI.getPreprocessorOpts().ChainedIncludes;
  103. assert(!includes.empty() && "No '-chain-include' in options!");
  104. IntrusiveRefCntPtr<ChainedIncludesSource> source(new ChainedIncludesSource());
  105. InputKind IK = CI.getFrontendOpts().Inputs[0].getKind();
  106. SmallVector<std::unique_ptr<llvm::MemoryBuffer>, 4> SerialBufs;
  107. SmallVector<std::string, 4> serialBufNames;
  108. for (unsigned i = 0, e = includes.size(); i != e; ++i) {
  109. bool firstInclude = (i == 0);
  110. std::unique_ptr<CompilerInvocation> CInvok;
  111. CInvok.reset(new CompilerInvocation(CI.getInvocation()));
  112. CInvok->getPreprocessorOpts().ChainedIncludes.clear();
  113. CInvok->getPreprocessorOpts().ImplicitPCHInclude.clear();
  114. CInvok->getPreprocessorOpts().ImplicitPTHInclude.clear();
  115. CInvok->getPreprocessorOpts().DisablePCHValidation = true;
  116. CInvok->getPreprocessorOpts().Includes.clear();
  117. CInvok->getPreprocessorOpts().MacroIncludes.clear();
  118. CInvok->getPreprocessorOpts().Macros.clear();
  119. CInvok->getFrontendOpts().Inputs.clear();
  120. FrontendInputFile InputFile(includes[i], IK);
  121. CInvok->getFrontendOpts().Inputs.push_back(InputFile);
  122. TextDiagnosticPrinter *DiagClient =
  123. new TextDiagnosticPrinter(llvm::errs(), new DiagnosticOptions());
  124. IntrusiveRefCntPtr<DiagnosticIDs> DiagID(new DiagnosticIDs());
  125. IntrusiveRefCntPtr<DiagnosticsEngine> Diags(
  126. new DiagnosticsEngine(DiagID, &CI.getDiagnosticOpts(), DiagClient));
  127. std::unique_ptr<CompilerInstance> Clang(
  128. new CompilerInstance(CI.getPCHContainerOperations()));
  129. Clang->setInvocation(CInvok.release());
  130. Clang->setDiagnostics(Diags.get());
  131. Clang->setTarget(TargetInfo::CreateTargetInfo(
  132. Clang->getDiagnostics(), Clang->getInvocation().TargetOpts));
  133. Clang->createFileManager();
  134. Clang->createSourceManager(Clang->getFileManager());
  135. Clang->createPreprocessor(TU_Prefix);
  136. Clang->getDiagnosticClient().BeginSourceFile(Clang->getLangOpts(),
  137. &Clang->getPreprocessor());
  138. Clang->createASTContext();
  139. auto Buffer = std::make_shared<PCHBuffer>();
  140. auto consumer = llvm::make_unique<PCHGenerator>(
  141. Clang->getPreprocessor(), "-", nullptr, /*isysroot=*/"", Buffer);
  142. Clang->getASTContext().setASTMutationListener(
  143. consumer->GetASTMutationListener());
  144. Clang->setASTConsumer(std::move(consumer));
  145. Clang->createSema(TU_Prefix, nullptr);
  146. if (firstInclude) {
  147. Preprocessor &PP = Clang->getPreprocessor();
  148. PP.getBuiltinInfo().initializeBuiltins(PP.getIdentifierTable(),
  149. PP.getLangOpts());
  150. } else {
  151. assert(!SerialBufs.empty());
  152. SmallVector<std::unique_ptr<llvm::MemoryBuffer>, 4> Bufs;
  153. // TODO: Pass through the existing MemoryBuffer instances instead of
  154. // allocating new ones.
  155. for (auto &SB : SerialBufs)
  156. Bufs.push_back(llvm::MemoryBuffer::getMemBuffer(SB->getBuffer()));
  157. std::string pchName = includes[i-1];
  158. llvm::raw_string_ostream os(pchName);
  159. os << ".pch" << i-1;
  160. serialBufNames.push_back(os.str());
  161. IntrusiveRefCntPtr<ASTReader> Reader;
  162. Reader = createASTReader(
  163. *Clang, pchName, Bufs, serialBufNames,
  164. Clang->getASTConsumer().GetASTDeserializationListener());
  165. if (!Reader)
  166. return nullptr;
  167. Clang->setModuleManager(Reader);
  168. Clang->getASTContext().setExternalSource(Reader);
  169. }
  170. if (!Clang->InitializeSourceManager(InputFile))
  171. return nullptr;
  172. ParseAST(Clang->getSema());
  173. Clang->getDiagnosticClient().EndSourceFile();
  174. assert(Buffer->IsComplete && "serialization did not complete");
  175. auto &serialAST = Buffer->Data;
  176. SerialBufs.push_back(llvm::MemoryBuffer::getMemBufferCopy(
  177. StringRef(serialAST.data(), serialAST.size())));
  178. serialAST.clear();
  179. source->CIs.push_back(Clang.release());
  180. }
  181. assert(!SerialBufs.empty());
  182. std::string pchName = includes.back() + ".pch-final";
  183. serialBufNames.push_back(pchName);
  184. Reader = createASTReader(CI, pchName, SerialBufs, serialBufNames);
  185. if (!Reader)
  186. return nullptr;
  187. source->FinalReader = Reader;
  188. return source;
  189. }
  190. //===----------------------------------------------------------------------===//
  191. // ExternalASTSource interface.
  192. //===----------------------------------------------------------------------===//
  193. Decl *ChainedIncludesSource::GetExternalDecl(uint32_t ID) {
  194. return getFinalReader().GetExternalDecl(ID);
  195. }
  196. Selector ChainedIncludesSource::GetExternalSelector(uint32_t ID) {
  197. return getFinalReader().GetExternalSelector(ID);
  198. }
  199. uint32_t ChainedIncludesSource::GetNumExternalSelectors() {
  200. return getFinalReader().GetNumExternalSelectors();
  201. }
  202. Stmt *ChainedIncludesSource::GetExternalDeclStmt(uint64_t Offset) {
  203. return getFinalReader().GetExternalDeclStmt(Offset);
  204. }
  205. CXXBaseSpecifier *
  206. ChainedIncludesSource::GetExternalCXXBaseSpecifiers(uint64_t Offset) {
  207. return getFinalReader().GetExternalCXXBaseSpecifiers(Offset);
  208. }
  209. CXXCtorInitializer **
  210. ChainedIncludesSource::GetExternalCXXCtorInitializers(uint64_t Offset) {
  211. return getFinalReader().GetExternalCXXCtorInitializers(Offset);
  212. }
  213. bool
  214. ChainedIncludesSource::FindExternalVisibleDeclsByName(const DeclContext *DC,
  215. DeclarationName Name) {
  216. return getFinalReader().FindExternalVisibleDeclsByName(DC, Name);
  217. }
  218. void ChainedIncludesSource::FindExternalLexicalDecls(
  219. const DeclContext *DC, llvm::function_ref<bool(Decl::Kind)> IsKindWeWant,
  220. SmallVectorImpl<Decl *> &Result) {
  221. return getFinalReader().FindExternalLexicalDecls(DC, IsKindWeWant, Result);
  222. }
  223. void ChainedIncludesSource::CompleteType(TagDecl *Tag) {
  224. return getFinalReader().CompleteType(Tag);
  225. }
  226. void ChainedIncludesSource::CompleteType(ObjCInterfaceDecl *Class) {
  227. return getFinalReader().CompleteType(Class);
  228. }
  229. void ChainedIncludesSource::StartedDeserializing() {
  230. return getFinalReader().StartedDeserializing();
  231. }
  232. void ChainedIncludesSource::FinishedDeserializing() {
  233. return getFinalReader().FinishedDeserializing();
  234. }
  235. void ChainedIncludesSource::StartTranslationUnit(ASTConsumer *Consumer) {
  236. return getFinalReader().StartTranslationUnit(Consumer);
  237. }
  238. void ChainedIncludesSource::PrintStats() {
  239. return getFinalReader().PrintStats();
  240. }
  241. void ChainedIncludesSource::getMemoryBufferSizes(MemoryBufferSizes &sizes)const{
  242. for (unsigned i = 0, e = CIs.size(); i != e; ++i) {
  243. if (const ExternalASTSource *eSrc =
  244. CIs[i]->getASTContext().getExternalSource()) {
  245. eSrc->getMemoryBufferSizes(sizes);
  246. }
  247. }
  248. getFinalReader().getMemoryBufferSizes(sizes);
  249. }
  250. void ChainedIncludesSource::InitializeSema(Sema &S) {
  251. return getFinalReader().InitializeSema(S);
  252. }
  253. void ChainedIncludesSource::ForgetSema() {
  254. return getFinalReader().ForgetSema();
  255. }
  256. void ChainedIncludesSource::ReadMethodPool(Selector Sel) {
  257. getFinalReader().ReadMethodPool(Sel);
  258. }
  259. bool ChainedIncludesSource::LookupUnqualified(LookupResult &R, Scope *S) {
  260. return getFinalReader().LookupUnqualified(R, S);
  261. }