1 //===--- GeneratePCH.cpp - AST Consumer for PCH Generation ------*- 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 CreatePCHGenerate function, which creates an 11 // ASTConsumer that generates a PCH file. 12 // 13 //===----------------------------------------------------------------------===// 14 15 #include "clang/Frontend/ASTConsumers.h" 16 #include "clang/Serialization/ASTWriter.h" 17 #include "clang/Sema/SemaConsumer.h" 18 #include "clang/AST/ASTContext.h" 19 #include "clang/AST/ASTConsumer.h" 20 #include "clang/Lex/Preprocessor.h" 21 #include "clang/Basic/FileManager.h" 22 #include "clang/Basic/FileSystemStatCache.h" 23 #include "llvm/Bitcode/BitstreamWriter.h" 24 #include "llvm/Support/raw_ostream.h" 25 #include <string> 26 27 using namespace clang; 28 29 PCHGenerator::PCHGenerator(const Preprocessor &PP, 30 bool Chaining, 31 const char *isysroot, 32 llvm::raw_ostream *OS) 33 : PP(PP), isysroot(isysroot), Out(OS), SemaPtr(0), 34 StatCalls(0), Stream(Buffer), Writer(Stream), Chaining(Chaining) { 35 // Install a stat() listener to keep track of all of the stat() 36 // calls. 37 StatCalls = new MemorizeStatCalls(); 38 // If we have a chain, we want new stat calls only, so install the memorizer 39 // *after* the already installed ASTReader's stat cache. 40 PP.getFileManager().addStatCache(StatCalls, 41 /*AtBeginning=*/!Chaining); 42 } 43 44 void PCHGenerator::HandleTranslationUnit(ASTContext &Ctx) { 45 if (PP.getDiagnostics().hasErrorOccurred()) 46 return; 47 48 // Set up the serialization listener. 49 Writer.SetSerializationListener(GetASTSerializationListener()); 50 51 // Emit the PCH file 52 assert(SemaPtr && "No Sema?"); 53 Writer.WriteAST(*SemaPtr, StatCalls, isysroot); 54 55 // Write the generated bitstream to "Out". 56 Out->write((char *)&Buffer.front(), Buffer.size()); 57 58 // Make sure it hits disk now. 59 Out->flush(); 60 61 // Free up some memory, in case the process is kept alive. 62 Buffer.clear(); 63 } 64 65 ASTMutationListener *PCHGenerator::GetASTMutationListener() { 66 if (Chaining) 67 return &Writer; 68 return 0; 69 } 70 71 ASTSerializationListener *PCHGenerator::GetASTSerializationListener() { 72 return 0; 73 } 74 75 ASTDeserializationListener *PCHGenerator::GetASTDeserializationListener() { 76 return &Writer; 77 } 78