2010-08-19 07:57:11 +08:00
|
|
|
//===--- ASTReader.cpp - AST File Reader ------------------------*- C++ -*-===//
|
2009-04-10 06:27:44 +08:00
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
2010-08-19 07:56:43 +08:00
|
|
|
// This file defines the ASTReader class, which reads AST files.
|
2009-04-10 06:27:44 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
2009-04-27 13:14:47 +08:00
|
|
|
|
2010-08-19 07:57:17 +08:00
|
|
|
#include "clang/Serialization/ASTReader.h"
|
|
|
|
#include "clang/Serialization/ASTDeserializationListener.h"
|
2010-08-21 00:03:52 +08:00
|
|
|
#include "ASTCommon.h"
|
2009-04-11 04:39:37 +08:00
|
|
|
#include "clang/Frontend/FrontendDiagnostic.h"
|
2009-11-12 07:58:53 +08:00
|
|
|
#include "clang/Frontend/Utils.h"
|
2010-08-13 04:07:10 +08:00
|
|
|
#include "clang/Sema/Sema.h"
|
2010-08-24 16:50:51 +08:00
|
|
|
#include "clang/Sema/Scope.h"
|
2009-04-14 08:24:19 +08:00
|
|
|
#include "clang/AST/ASTConsumer.h"
|
2009-04-10 06:27:44 +08:00
|
|
|
#include "clang/AST/ASTContext.h"
|
2010-08-25 13:32:35 +08:00
|
|
|
#include "clang/AST/DeclTemplate.h"
|
2009-04-15 05:18:50 +08:00
|
|
|
#include "clang/AST/Expr.h"
|
2010-08-24 15:32:53 +08:00
|
|
|
#include "clang/AST/ExprCXX.h"
|
2009-04-10 06:27:44 +08:00
|
|
|
#include "clang/AST/Type.h"
|
2009-10-17 05:56:05 +08:00
|
|
|
#include "clang/AST/TypeLocVisitor.h"
|
2009-04-11 05:41:48 +08:00
|
|
|
#include "clang/Lex/MacroInfo.h"
|
Implement serialization and lazy deserialization of the preprocessing
record (which includes all macro instantiations and definitions). As
with all lay deserialization, this introduces a new external source
(here, an external preprocessing record source) that loads all of the
preprocessed entities prior to iterating over the entities.
The preprocessing record is an optional part of the precompiled header
that is disabled by default (enabled with
-detailed-preprocessing-record). When the preprocessor given to the
PCH writer has a preprocessing record, that record is written into the
PCH file. When the PCH reader is given a PCH file that contains a
preprocessing record, it will be lazily loaded (which, effectively,
implicitly adds -detailed-preprocessing-record). This is the first
case where we have sections of the precompiled header that are
added/removed based on a compilation flag, which is
unfortunate. However, this data consumes ~550k in the PCH file for
Cocoa.h (out of ~9.9MB), and there is a non-trivial cost to gathering
this detailed preprocessing information, so it's too expensive to turn
on by default. In the future, we should investigate a better encoding
of this information.
llvm-svn: 99002
2010-03-20 05:51:54 +08:00
|
|
|
#include "clang/Lex/PreprocessingRecord.h"
|
2009-04-10 11:52:48 +08:00
|
|
|
#include "clang/Lex/Preprocessor.h"
|
2009-04-25 04:03:17 +08:00
|
|
|
#include "clang/Lex/HeaderSearch.h"
|
Lazy deserialization of the declaration chains associated with
identifiers from a precompiled header.
This patch changes the primary name lookup method for entities within
a precompiled header. Previously, we would load all of the names of
declarations at translation unit scope into a large DenseMap (inside
the TranslationUnitDecl's DeclContext), and then perform a special
"last resort" lookup into this DeclContext when we knew there was a
PCH file (see Sema::LookupName). Now, when we see an identifier named
for the first time, we load all of the declarations with that name
that are visible from the translation unit into the IdentifierInfo's
chain of declarations. Thus, the explicit "look into the translation
unit's DeclContext" code is gone, and Sema effectively uses the same
IdentifierInfo-based name lookup mechanism whether we are using a PCH
file or not.
This approach should help PCH scale with the size of the input program
rather than the size of the PCH file. The "Hello, World!" application
with Carbon.h as a PCH file now loads 20% of the identifiers in the
PCH file rather than 85% of the identifiers.
90% of the 20% of identifiers loaded are actually loaded when we
deserialize the preprocessor state. The next step is to make the
preprocessor load macros lazily, which should drastically reduce the
number of types, declarations, and identifiers loaded for "Hello,
World".
llvm-svn: 69737
2009-04-22 06:25:48 +08:00
|
|
|
#include "clang/Basic/OnDiskHashTable.h"
|
2009-04-10 11:52:48 +08:00
|
|
|
#include "clang/Basic/SourceManager.h"
|
2009-04-14 00:31:14 +08:00
|
|
|
#include "clang/Basic/SourceManagerInternals.h"
|
2009-04-10 11:52:48 +08:00
|
|
|
#include "clang/Basic/FileManager.h"
|
2009-04-11 05:16:55 +08:00
|
|
|
#include "clang/Basic/TargetInfo.h"
|
2009-10-06 05:07:28 +08:00
|
|
|
#include "clang/Basic/Version.h"
|
2009-10-18 07:52:28 +08:00
|
|
|
#include "llvm/ADT/StringExtras.h"
|
2009-04-10 06:27:44 +08:00
|
|
|
#include "llvm/Bitcode/BitstreamReader.h"
|
|
|
|
#include "llvm/Support/MemoryBuffer.h"
|
2009-10-29 16:12:44 +08:00
|
|
|
#include "llvm/Support/ErrorHandling.h"
|
2009-11-19 03:50:41 +08:00
|
|
|
#include "llvm/System/Path.h"
|
2009-04-10 06:27:44 +08:00
|
|
|
#include <algorithm>
|
2009-04-29 02:58:38 +08:00
|
|
|
#include <iterator>
|
2009-04-10 06:27:44 +08:00
|
|
|
#include <cstdio>
|
2009-04-28 02:38:38 +08:00
|
|
|
#include <sys/stat.h>
|
2009-04-10 06:27:44 +08:00
|
|
|
using namespace clang;
|
2010-08-19 07:57:32 +08:00
|
|
|
using namespace clang::serialization;
|
2009-04-10 06:27:44 +08:00
|
|
|
|
2009-06-19 08:03:23 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2010-08-19 07:57:06 +08:00
|
|
|
// PCH validator implementation
|
2009-06-19 08:03:23 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2010-08-19 07:56:56 +08:00
|
|
|
ASTReaderListener::~ASTReaderListener() {}
|
2009-06-19 08:03:23 +08:00
|
|
|
|
|
|
|
bool
|
|
|
|
PCHValidator::ReadLanguageOptions(const LangOptions &LangOpts) {
|
|
|
|
const LangOptions &PPLangOpts = PP.getLangOptions();
|
|
|
|
#define PARSE_LANGOPT_BENIGN(Option)
|
|
|
|
#define PARSE_LANGOPT_IMPORTANT(Option, DiagID) \
|
|
|
|
if (PPLangOpts.Option != LangOpts.Option) { \
|
|
|
|
Reader.Diag(DiagID) << LangOpts.Option << PPLangOpts.Option; \
|
|
|
|
return true; \
|
|
|
|
}
|
|
|
|
|
|
|
|
PARSE_LANGOPT_BENIGN(Trigraphs);
|
|
|
|
PARSE_LANGOPT_BENIGN(BCPLComment);
|
|
|
|
PARSE_LANGOPT_BENIGN(DollarIdents);
|
|
|
|
PARSE_LANGOPT_BENIGN(AsmPreprocessor);
|
|
|
|
PARSE_LANGOPT_IMPORTANT(GNUMode, diag::warn_pch_gnu_extensions);
|
2010-04-18 04:17:31 +08:00
|
|
|
PARSE_LANGOPT_IMPORTANT(GNUKeywords, diag::warn_pch_gnu_keywords);
|
2009-06-19 08:03:23 +08:00
|
|
|
PARSE_LANGOPT_BENIGN(ImplicitInt);
|
|
|
|
PARSE_LANGOPT_BENIGN(Digraphs);
|
|
|
|
PARSE_LANGOPT_BENIGN(HexFloats);
|
|
|
|
PARSE_LANGOPT_IMPORTANT(C99, diag::warn_pch_c99);
|
|
|
|
PARSE_LANGOPT_IMPORTANT(Microsoft, diag::warn_pch_microsoft_extensions);
|
|
|
|
PARSE_LANGOPT_IMPORTANT(CPlusPlus, diag::warn_pch_cplusplus);
|
|
|
|
PARSE_LANGOPT_IMPORTANT(CPlusPlus0x, diag::warn_pch_cplusplus0x);
|
|
|
|
PARSE_LANGOPT_BENIGN(CXXOperatorName);
|
|
|
|
PARSE_LANGOPT_IMPORTANT(ObjC1, diag::warn_pch_objective_c);
|
|
|
|
PARSE_LANGOPT_IMPORTANT(ObjC2, diag::warn_pch_objective_c2);
|
|
|
|
PARSE_LANGOPT_IMPORTANT(ObjCNonFragileABI, diag::warn_pch_nonfragile_abi);
|
2010-02-10 03:31:38 +08:00
|
|
|
PARSE_LANGOPT_IMPORTANT(ObjCNonFragileABI2, diag::warn_pch_nonfragile_abi2);
|
2010-04-23 05:01:59 +08:00
|
|
|
PARSE_LANGOPT_IMPORTANT(NoConstantCFStrings,
|
|
|
|
diag::warn_pch_no_constant_cfstrings);
|
2009-06-19 08:03:23 +08:00
|
|
|
PARSE_LANGOPT_BENIGN(PascalStrings);
|
|
|
|
PARSE_LANGOPT_BENIGN(WritableStrings);
|
2009-09-09 23:08:12 +08:00
|
|
|
PARSE_LANGOPT_IMPORTANT(LaxVectorConversions,
|
2009-06-19 08:03:23 +08:00
|
|
|
diag::warn_pch_lax_vector_conversions);
|
2009-06-26 06:57:40 +08:00
|
|
|
PARSE_LANGOPT_IMPORTANT(AltiVec, diag::warn_pch_altivec);
|
2009-06-19 08:03:23 +08:00
|
|
|
PARSE_LANGOPT_IMPORTANT(Exceptions, diag::warn_pch_exceptions);
|
2010-02-11 02:48:44 +08:00
|
|
|
PARSE_LANGOPT_IMPORTANT(SjLjExceptions, diag::warn_pch_sjlj_exceptions);
|
2009-06-19 08:03:23 +08:00
|
|
|
PARSE_LANGOPT_IMPORTANT(NeXTRuntime, diag::warn_pch_objc_runtime);
|
|
|
|
PARSE_LANGOPT_IMPORTANT(Freestanding, diag::warn_pch_freestanding);
|
|
|
|
PARSE_LANGOPT_IMPORTANT(NoBuiltin, diag::warn_pch_builtins);
|
2009-09-09 23:08:12 +08:00
|
|
|
PARSE_LANGOPT_IMPORTANT(ThreadsafeStatics,
|
2009-06-19 08:03:23 +08:00
|
|
|
diag::warn_pch_thread_safe_statics);
|
2009-09-03 12:54:28 +08:00
|
|
|
PARSE_LANGOPT_IMPORTANT(POSIXThreads, diag::warn_pch_posix_threads);
|
2009-06-19 08:03:23 +08:00
|
|
|
PARSE_LANGOPT_IMPORTANT(Blocks, diag::warn_pch_blocks);
|
|
|
|
PARSE_LANGOPT_BENIGN(EmitAllDecls);
|
|
|
|
PARSE_LANGOPT_IMPORTANT(MathErrno, diag::warn_pch_math_errno);
|
2010-06-27 05:25:03 +08:00
|
|
|
PARSE_LANGOPT_BENIGN(getSignedOverflowBehavior());
|
2009-09-09 23:08:12 +08:00
|
|
|
PARSE_LANGOPT_IMPORTANT(HeinousExtensions,
|
2009-06-19 08:03:23 +08:00
|
|
|
diag::warn_pch_heinous_extensions);
|
|
|
|
// FIXME: Most of the options below are benign if the macro wasn't
|
|
|
|
// used. Unfortunately, this means that a PCH compiled without
|
|
|
|
// optimization can't be used with optimization turned on, even
|
|
|
|
// though the only thing that changes is whether __OPTIMIZE__ was
|
|
|
|
// defined... but if __OPTIMIZE__ never showed up in the header, it
|
|
|
|
// doesn't matter. We could consider making this some special kind
|
|
|
|
// of check.
|
|
|
|
PARSE_LANGOPT_IMPORTANT(Optimize, diag::warn_pch_optimize);
|
|
|
|
PARSE_LANGOPT_IMPORTANT(OptimizeSize, diag::warn_pch_optimize_size);
|
|
|
|
PARSE_LANGOPT_IMPORTANT(Static, diag::warn_pch_static);
|
|
|
|
PARSE_LANGOPT_IMPORTANT(PICLevel, diag::warn_pch_pic_level);
|
|
|
|
PARSE_LANGOPT_IMPORTANT(GNUInline, diag::warn_pch_gnu_inline);
|
|
|
|
PARSE_LANGOPT_IMPORTANT(NoInline, diag::warn_pch_no_inline);
|
|
|
|
PARSE_LANGOPT_IMPORTANT(AccessControl, diag::warn_pch_access_control);
|
|
|
|
PARSE_LANGOPT_IMPORTANT(CharIsSigned, diag::warn_pch_char_signed);
|
2009-11-06 04:14:16 +08:00
|
|
|
PARSE_LANGOPT_IMPORTANT(ShortWChar, diag::warn_pch_short_wchar);
|
2009-06-19 08:03:23 +08:00
|
|
|
if ((PPLangOpts.getGCMode() != 0) != (LangOpts.getGCMode() != 0)) {
|
2009-09-09 23:08:12 +08:00
|
|
|
Reader.Diag(diag::warn_pch_gc_mode)
|
2009-06-19 08:03:23 +08:00
|
|
|
<< LangOpts.getGCMode() << PPLangOpts.getGCMode();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
PARSE_LANGOPT_BENIGN(getVisibilityMode());
|
2009-09-21 12:16:19 +08:00
|
|
|
PARSE_LANGOPT_IMPORTANT(getStackProtectorMode(),
|
|
|
|
diag::warn_pch_stack_protector);
|
2009-06-19 08:03:23 +08:00
|
|
|
PARSE_LANGOPT_BENIGN(InstantiationDepth);
|
2009-06-26 06:57:40 +08:00
|
|
|
PARSE_LANGOPT_IMPORTANT(OpenCL, diag::warn_pch_opencl);
|
2009-12-12 09:27:46 +08:00
|
|
|
PARSE_LANGOPT_BENIGN(CatchUndefined);
|
2009-09-21 12:16:19 +08:00
|
|
|
PARSE_LANGOPT_IMPORTANT(ElideConstructors, diag::warn_pch_elide_constructors);
|
2010-07-10 01:35:33 +08:00
|
|
|
PARSE_LANGOPT_BENIGN(SpellChecking);
|
2010-03-08 03:10:13 +08:00
|
|
|
#undef PARSE_LANGOPT_IMPORTANT
|
2009-06-19 08:03:23 +08:00
|
|
|
#undef PARSE_LANGOPT_BENIGN
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2009-11-11 08:52:11 +08:00
|
|
|
bool PCHValidator::ReadTargetTriple(llvm::StringRef Triple) {
|
|
|
|
if (Triple == PP.getTargetInfo().getTriple().str())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
Reader.Diag(diag::warn_pch_target_triple)
|
|
|
|
<< Triple << PP.getTargetInfo().getTriple().str();
|
|
|
|
return true;
|
2009-06-19 08:03:23 +08:00
|
|
|
}
|
|
|
|
|
2010-07-15 01:49:11 +08:00
|
|
|
struct EmptyStringRef {
|
2010-07-15 07:19:41 +08:00
|
|
|
bool operator ()(llvm::StringRef r) const { return r.empty(); }
|
2010-07-15 01:49:11 +08:00
|
|
|
};
|
|
|
|
struct EmptyBlock {
|
|
|
|
bool operator ()(const PCHPredefinesBlock &r) const { return r.Data.empty(); }
|
|
|
|
};
|
|
|
|
|
|
|
|
static bool EqualConcatenations(llvm::SmallVector<llvm::StringRef, 2> L,
|
|
|
|
PCHPredefinesBlocks R) {
|
|
|
|
// First, sum up the lengths.
|
|
|
|
unsigned LL = 0, RL = 0;
|
|
|
|
for (unsigned I = 0, N = L.size(); I != N; ++I) {
|
|
|
|
LL += L[I].size();
|
|
|
|
}
|
|
|
|
for (unsigned I = 0, N = R.size(); I != N; ++I) {
|
|
|
|
RL += R[I].Data.size();
|
|
|
|
}
|
|
|
|
if (LL != RL)
|
|
|
|
return false;
|
|
|
|
if (LL == 0 && RL == 0)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
// Kick out empty parts, they confuse the algorithm below.
|
|
|
|
L.erase(std::remove_if(L.begin(), L.end(), EmptyStringRef()), L.end());
|
|
|
|
R.erase(std::remove_if(R.begin(), R.end(), EmptyBlock()), R.end());
|
|
|
|
|
|
|
|
// Do it the hard way. At this point, both vectors must be non-empty.
|
|
|
|
llvm::StringRef LR = L[0], RR = R[0].Data;
|
|
|
|
unsigned LI = 0, RI = 0, LN = L.size(), RN = R.size();
|
2010-07-16 08:00:11 +08:00
|
|
|
(void) RN;
|
2010-07-15 01:49:11 +08:00
|
|
|
for (;;) {
|
|
|
|
// Compare the current pieces.
|
|
|
|
if (LR.size() == RR.size()) {
|
|
|
|
// If they're the same length, it's pretty easy.
|
|
|
|
if (LR != RR)
|
|
|
|
return false;
|
|
|
|
// Both pieces are done, advance.
|
|
|
|
++LI;
|
|
|
|
++RI;
|
|
|
|
// If either string is done, they're both done, since they're the same
|
|
|
|
// length.
|
|
|
|
if (LI == LN) {
|
|
|
|
assert(RI == RN && "Strings not the same length after all?");
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
LR = L[LI];
|
|
|
|
RR = R[RI].Data;
|
|
|
|
} else if (LR.size() < RR.size()) {
|
|
|
|
// Right piece is longer.
|
|
|
|
if (!RR.startswith(LR))
|
|
|
|
return false;
|
|
|
|
++LI;
|
|
|
|
assert(LI != LN && "Strings not the same length after all?");
|
|
|
|
RR = RR.substr(LR.size());
|
|
|
|
LR = L[LI];
|
|
|
|
} else {
|
|
|
|
// Left piece is longer.
|
|
|
|
if (!LR.startswith(RR))
|
|
|
|
return false;
|
|
|
|
++RI;
|
|
|
|
assert(RI != RN && "Strings not the same length after all?");
|
|
|
|
LR = LR.substr(RR.size());
|
|
|
|
RR = R[RI].Data;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static std::pair<FileID, llvm::StringRef::size_type>
|
|
|
|
FindMacro(const PCHPredefinesBlocks &Buffers, llvm::StringRef MacroDef) {
|
|
|
|
std::pair<FileID, llvm::StringRef::size_type> Res;
|
|
|
|
for (unsigned I = 0, N = Buffers.size(); I != N; ++I) {
|
|
|
|
Res.second = Buffers[I].Data.find(MacroDef);
|
|
|
|
if (Res.second != llvm::StringRef::npos) {
|
|
|
|
Res.first = Buffers[I].BufferID;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return Res;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool PCHValidator::ReadPredefinesBuffer(const PCHPredefinesBlocks &Buffers,
|
2009-11-11 13:29:04 +08:00
|
|
|
llvm::StringRef OriginalFileName,
|
2009-06-19 08:03:23 +08:00
|
|
|
std::string &SuggestedPredefines) {
|
2009-11-12 07:58:53 +08:00
|
|
|
// We are in the context of an implicit include, so the predefines buffer will
|
|
|
|
// have a #include entry for the PCH file itself (as normalized by the
|
|
|
|
// preprocessor initialization). Find it and skip over it in the checking
|
|
|
|
// below.
|
2009-11-11 13:29:04 +08:00
|
|
|
llvm::SmallString<256> PCHInclude;
|
|
|
|
PCHInclude += "#include \"";
|
2009-11-12 07:58:53 +08:00
|
|
|
PCHInclude += NormalizeDashIncludePath(OriginalFileName);
|
2009-11-11 13:29:04 +08:00
|
|
|
PCHInclude += "\"\n";
|
|
|
|
std::pair<llvm::StringRef,llvm::StringRef> Split =
|
|
|
|
llvm::StringRef(PP.getPredefines()).split(PCHInclude.str());
|
|
|
|
llvm::StringRef Left = Split.first, Right = Split.second;
|
2010-03-18 08:56:54 +08:00
|
|
|
if (Left == PP.getPredefines()) {
|
|
|
|
Error("Missing PCH include entry!");
|
|
|
|
return true;
|
|
|
|
}
|
2009-11-11 13:29:04 +08:00
|
|
|
|
2010-07-15 01:49:11 +08:00
|
|
|
// If the concatenation of all the PCH buffers is equal to the adjusted
|
|
|
|
// command line, we're done.
|
|
|
|
llvm::SmallVector<llvm::StringRef, 2> CommandLine;
|
|
|
|
CommandLine.push_back(Left);
|
|
|
|
CommandLine.push_back(Right);
|
|
|
|
if (EqualConcatenations(CommandLine, Buffers))
|
2009-06-19 08:03:23 +08:00
|
|
|
return false;
|
|
|
|
|
|
|
|
SourceManager &SourceMgr = PP.getSourceManager();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-11-11 11:45:59 +08:00
|
|
|
// The predefines buffers are different. Determine what the differences are,
|
|
|
|
// and whether they require us to reject the PCH file.
|
2009-11-14 00:46:11 +08:00
|
|
|
llvm::SmallVector<llvm::StringRef, 8> PCHLines;
|
2010-07-15 01:49:11 +08:00
|
|
|
for (unsigned I = 0, N = Buffers.size(); I != N; ++I)
|
|
|
|
Buffers[I].Data.split(PCHLines, "\n", /*MaxSplit=*/-1, /*KeepEmpty=*/false);
|
2009-11-14 00:46:11 +08:00
|
|
|
|
|
|
|
llvm::SmallVector<llvm::StringRef, 8> CmdLineLines;
|
|
|
|
Left.split(CmdLineLines, "\n", /*MaxSplit=*/-1, /*KeepEmpty=*/false);
|
|
|
|
Right.split(CmdLineLines, "\n", /*MaxSplit=*/-1, /*KeepEmpty=*/false);
|
2009-06-19 08:03:23 +08:00
|
|
|
|
2009-11-11 13:26:28 +08:00
|
|
|
// Sort both sets of predefined buffer lines, since we allow some extra
|
|
|
|
// definitions and they may appear at any point in the output.
|
2009-06-19 08:03:23 +08:00
|
|
|
std::sort(CmdLineLines.begin(), CmdLineLines.end());
|
|
|
|
std::sort(PCHLines.begin(), PCHLines.end());
|
|
|
|
|
2009-11-11 13:26:28 +08:00
|
|
|
// Determine which predefines that were used to build the PCH file are missing
|
|
|
|
// from the command line.
|
|
|
|
std::vector<llvm::StringRef> MissingPredefines;
|
2009-06-19 08:03:23 +08:00
|
|
|
std::set_difference(PCHLines.begin(), PCHLines.end(),
|
|
|
|
CmdLineLines.begin(), CmdLineLines.end(),
|
|
|
|
std::back_inserter(MissingPredefines));
|
|
|
|
|
|
|
|
bool MissingDefines = false;
|
|
|
|
bool ConflictingDefines = false;
|
|
|
|
for (unsigned I = 0, N = MissingPredefines.size(); I != N; ++I) {
|
2009-11-11 13:26:28 +08:00
|
|
|
llvm::StringRef Missing = MissingPredefines[I];
|
|
|
|
if (!Missing.startswith("#define ")) {
|
2009-06-19 08:03:23 +08:00
|
|
|
Reader.Diag(diag::warn_pch_compiler_options_mismatch);
|
|
|
|
return true;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-11-11 11:45:59 +08:00
|
|
|
// This is a macro definition. Determine the name of the macro we're
|
|
|
|
// defining.
|
2009-06-19 08:03:23 +08:00
|
|
|
std::string::size_type StartOfMacroName = strlen("#define ");
|
2009-09-09 23:08:12 +08:00
|
|
|
std::string::size_type EndOfMacroName
|
2009-06-19 08:03:23 +08:00
|
|
|
= Missing.find_first_of("( \n\r", StartOfMacroName);
|
|
|
|
assert(EndOfMacroName != std::string::npos &&
|
|
|
|
"Couldn't find the end of the macro name");
|
2009-11-11 13:26:28 +08:00
|
|
|
llvm::StringRef MacroName = Missing.slice(StartOfMacroName, EndOfMacroName);
|
2009-06-19 08:03:23 +08:00
|
|
|
|
2009-11-11 11:45:59 +08:00
|
|
|
// Determine whether this macro was given a different definition on the
|
|
|
|
// command line.
|
2009-11-11 13:26:28 +08:00
|
|
|
std::string MacroDefStart = "#define " + MacroName.str();
|
2009-06-19 08:03:23 +08:00
|
|
|
std::string::size_type MacroDefLen = MacroDefStart.size();
|
2009-11-14 00:46:11 +08:00
|
|
|
llvm::SmallVector<llvm::StringRef, 8>::iterator ConflictPos
|
2009-06-19 08:03:23 +08:00
|
|
|
= std::lower_bound(CmdLineLines.begin(), CmdLineLines.end(),
|
|
|
|
MacroDefStart);
|
|
|
|
for (; ConflictPos != CmdLineLines.end(); ++ConflictPos) {
|
2009-11-11 11:45:59 +08:00
|
|
|
if (!ConflictPos->startswith(MacroDefStart)) {
|
2009-06-19 08:03:23 +08:00
|
|
|
// Different macro; we're done.
|
|
|
|
ConflictPos = CmdLineLines.end();
|
2009-09-09 23:08:12 +08:00
|
|
|
break;
|
2009-06-19 08:03:23 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
|
|
|
assert(ConflictPos->size() > MacroDefLen &&
|
2009-06-19 08:03:23 +08:00
|
|
|
"Invalid #define in predefines buffer?");
|
2009-09-09 23:08:12 +08:00
|
|
|
if ((*ConflictPos)[MacroDefLen] != ' ' &&
|
2009-06-19 08:03:23 +08:00
|
|
|
(*ConflictPos)[MacroDefLen] != '(')
|
|
|
|
continue; // Longer macro name; keep trying.
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-06-19 08:03:23 +08:00
|
|
|
// We found a conflicting macro definition.
|
|
|
|
break;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-06-19 08:03:23 +08:00
|
|
|
if (ConflictPos != CmdLineLines.end()) {
|
|
|
|
Reader.Diag(diag::warn_cmdline_conflicting_macro_def)
|
|
|
|
<< MacroName;
|
|
|
|
|
|
|
|
// Show the definition of this macro within the PCH file.
|
2010-07-15 01:49:11 +08:00
|
|
|
std::pair<FileID, llvm::StringRef::size_type> MacroLoc =
|
|
|
|
FindMacro(Buffers, Missing);
|
|
|
|
assert(MacroLoc.second!=llvm::StringRef::npos && "Unable to find macro!");
|
|
|
|
SourceLocation PCHMissingLoc =
|
|
|
|
SourceMgr.getLocForStartOfFile(MacroLoc.first)
|
|
|
|
.getFileLocWithOffset(MacroLoc.second);
|
2009-11-11 13:26:28 +08:00
|
|
|
Reader.Diag(PCHMissingLoc, diag::note_pch_macro_defined_as) << MacroName;
|
2009-06-19 08:03:23 +08:00
|
|
|
|
|
|
|
ConflictingDefines = true;
|
|
|
|
continue;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-11-11 11:45:59 +08:00
|
|
|
// If the macro doesn't conflict, then we'll just pick up the macro
|
|
|
|
// definition from the PCH file. Warn the user that they made a mistake.
|
2009-06-19 08:03:23 +08:00
|
|
|
if (ConflictingDefines)
|
|
|
|
continue; // Don't complain if there are already conflicting defs
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-06-19 08:03:23 +08:00
|
|
|
if (!MissingDefines) {
|
|
|
|
Reader.Diag(diag::warn_cmdline_missing_macro_defs);
|
|
|
|
MissingDefines = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Show the definition of this macro within the PCH file.
|
2010-07-15 01:49:11 +08:00
|
|
|
std::pair<FileID, llvm::StringRef::size_type> MacroLoc =
|
|
|
|
FindMacro(Buffers, Missing);
|
|
|
|
assert(MacroLoc.second!=llvm::StringRef::npos && "Unable to find macro!");
|
|
|
|
SourceLocation PCHMissingLoc =
|
|
|
|
SourceMgr.getLocForStartOfFile(MacroLoc.first)
|
|
|
|
.getFileLocWithOffset(MacroLoc.second);
|
2009-06-19 08:03:23 +08:00
|
|
|
Reader.Diag(PCHMissingLoc, diag::note_using_macro_def_from_pch);
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-06-19 08:03:23 +08:00
|
|
|
if (ConflictingDefines)
|
|
|
|
return true;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-06-19 08:03:23 +08:00
|
|
|
// Determine what predefines were introduced based on command-line
|
|
|
|
// parameters that were not present when building the PCH
|
|
|
|
// file. Extra #defines are okay, so long as the identifiers being
|
|
|
|
// defined were not used within the precompiled header.
|
2009-11-11 13:26:28 +08:00
|
|
|
std::vector<llvm::StringRef> ExtraPredefines;
|
2009-06-19 08:03:23 +08:00
|
|
|
std::set_difference(CmdLineLines.begin(), CmdLineLines.end(),
|
|
|
|
PCHLines.begin(), PCHLines.end(),
|
2009-09-09 23:08:12 +08:00
|
|
|
std::back_inserter(ExtraPredefines));
|
2009-06-19 08:03:23 +08:00
|
|
|
for (unsigned I = 0, N = ExtraPredefines.size(); I != N; ++I) {
|
2009-11-11 13:26:28 +08:00
|
|
|
llvm::StringRef &Extra = ExtraPredefines[I];
|
|
|
|
if (!Extra.startswith("#define ")) {
|
2009-06-19 08:03:23 +08:00
|
|
|
Reader.Diag(diag::warn_pch_compiler_options_mismatch);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// This is an extra macro definition. Determine the name of the
|
|
|
|
// macro we're defining.
|
|
|
|
std::string::size_type StartOfMacroName = strlen("#define ");
|
2009-09-09 23:08:12 +08:00
|
|
|
std::string::size_type EndOfMacroName
|
2009-06-19 08:03:23 +08:00
|
|
|
= Extra.find_first_of("( \n\r", StartOfMacroName);
|
|
|
|
assert(EndOfMacroName != std::string::npos &&
|
|
|
|
"Couldn't find the end of the macro name");
|
2009-11-11 13:26:28 +08:00
|
|
|
llvm::StringRef MacroName = Extra.slice(StartOfMacroName, EndOfMacroName);
|
2009-06-19 08:03:23 +08:00
|
|
|
|
|
|
|
// Check whether this name was used somewhere in the PCH file. If
|
|
|
|
// so, defining it as a macro could change behavior, so we reject
|
|
|
|
// the PCH file.
|
2009-11-11 13:26:28 +08:00
|
|
|
if (IdentifierInfo *II = Reader.get(MacroName)) {
|
2009-11-11 08:52:00 +08:00
|
|
|
Reader.Diag(diag::warn_macro_name_used_in_pch) << II;
|
2009-06-19 08:03:23 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add this definition to the suggested predefines buffer.
|
|
|
|
SuggestedPredefines += Extra;
|
|
|
|
SuggestedPredefines += '\n';
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we get here, it's because the predefines buffer had compatible
|
|
|
|
// contents. Accept the PCH file.
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2010-03-17 00:35:32 +08:00
|
|
|
void PCHValidator::ReadHeaderFileInfo(const HeaderFileInfo &HFI,
|
|
|
|
unsigned ID) {
|
|
|
|
PP.getHeaderSearchInfo().setHeaderFileInfoForUID(HFI, ID);
|
|
|
|
++NumHeaderInfos;
|
2009-06-19 08:03:23 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void PCHValidator::ReadCounter(unsigned Value) {
|
|
|
|
PP.setCounterValue(Value);
|
|
|
|
}
|
|
|
|
|
2009-04-22 14:29:42 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2010-08-19 07:56:43 +08:00
|
|
|
// AST reader implementation
|
2009-04-27 13:14:47 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2009-04-27 02:52:16 +08:00
|
|
|
|
2010-07-30 08:29:29 +08:00
|
|
|
void
|
2010-08-19 07:56:56 +08:00
|
|
|
ASTReader::setDeserializationListener(ASTDeserializationListener *Listener) {
|
2010-07-30 08:29:29 +08:00
|
|
|
DeserializationListener = Listener;
|
|
|
|
if (DeserializationListener)
|
|
|
|
DeserializationListener->SetReader(this);
|
|
|
|
}
|
|
|
|
|
Lazy deserialization of the declaration chains associated with
identifiers from a precompiled header.
This patch changes the primary name lookup method for entities within
a precompiled header. Previously, we would load all of the names of
declarations at translation unit scope into a large DenseMap (inside
the TranslationUnitDecl's DeclContext), and then perform a special
"last resort" lookup into this DeclContext when we knew there was a
PCH file (see Sema::LookupName). Now, when we see an identifier named
for the first time, we load all of the declarations with that name
that are visible from the translation unit into the IdentifierInfo's
chain of declarations. Thus, the explicit "look into the translation
unit's DeclContext" code is gone, and Sema effectively uses the same
IdentifierInfo-based name lookup mechanism whether we are using a PCH
file or not.
This approach should help PCH scale with the size of the input program
rather than the size of the PCH file. The "Hello, World!" application
with Carbon.h as a PCH file now loads 20% of the identifiers in the
PCH file rather than 85% of the identifiers.
90% of the 20% of identifiers loaded are actually loaded when we
deserialize the preprocessor state. The next step is to make the
preprocessor load macros lazily, which should drastically reduce the
number of types, declarations, and identifiers loaded for "Hello,
World".
llvm-svn: 69737
2009-04-22 06:25:48 +08:00
|
|
|
|
2009-04-25 05:10:55 +08:00
|
|
|
namespace {
|
2010-08-19 07:57:06 +08:00
|
|
|
class ASTSelectorLookupTrait {
|
2010-08-19 07:56:43 +08:00
|
|
|
ASTReader &Reader;
|
2009-04-25 05:10:55 +08:00
|
|
|
|
|
|
|
public:
|
2010-08-05 01:20:04 +08:00
|
|
|
struct data_type {
|
2010-08-19 07:57:32 +08:00
|
|
|
SelectorID ID;
|
2010-08-05 01:20:04 +08:00
|
|
|
ObjCMethodList Instance, Factory;
|
|
|
|
};
|
2009-04-25 05:10:55 +08:00
|
|
|
|
|
|
|
typedef Selector external_key_type;
|
|
|
|
typedef external_key_type internal_key_type;
|
|
|
|
|
2010-08-19 07:57:06 +08:00
|
|
|
explicit ASTSelectorLookupTrait(ASTReader &Reader) : Reader(Reader) { }
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-25 05:10:55 +08:00
|
|
|
static bool EqualKey(const internal_key_type& a,
|
|
|
|
const internal_key_type& b) {
|
|
|
|
return a == b;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-25 05:10:55 +08:00
|
|
|
static unsigned ComputeHash(Selector Sel) {
|
2010-08-21 00:03:52 +08:00
|
|
|
return serialization::ComputeHash(Sel);
|
2009-04-25 05:10:55 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-25 05:10:55 +08:00
|
|
|
// This hopefully will just get inlined and removed by the optimizer.
|
|
|
|
static const internal_key_type&
|
|
|
|
GetInternalKey(const external_key_type& x) { return x; }
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-25 05:10:55 +08:00
|
|
|
static std::pair<unsigned, unsigned>
|
|
|
|
ReadKeyDataLength(const unsigned char*& d) {
|
|
|
|
using namespace clang::io;
|
|
|
|
unsigned KeyLen = ReadUnalignedLE16(d);
|
|
|
|
unsigned DataLen = ReadUnalignedLE16(d);
|
|
|
|
return std::make_pair(KeyLen, DataLen);
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-26 01:48:32 +08:00
|
|
|
internal_key_type ReadKey(const unsigned char* d, unsigned) {
|
2009-04-25 05:10:55 +08:00
|
|
|
using namespace clang::io;
|
2009-04-28 05:45:14 +08:00
|
|
|
SelectorTable &SelTable = Reader.getContext()->Selectors;
|
2009-04-25 05:10:55 +08:00
|
|
|
unsigned N = ReadUnalignedLE16(d);
|
2009-09-09 23:08:12 +08:00
|
|
|
IdentifierInfo *FirstII
|
2009-04-25 05:10:55 +08:00
|
|
|
= Reader.DecodeIdentifierInfo(ReadUnalignedLE32(d));
|
|
|
|
if (N == 0)
|
|
|
|
return SelTable.getNullarySelector(FirstII);
|
|
|
|
else if (N == 1)
|
|
|
|
return SelTable.getUnarySelector(FirstII);
|
|
|
|
|
|
|
|
llvm::SmallVector<IdentifierInfo *, 16> Args;
|
|
|
|
Args.push_back(FirstII);
|
|
|
|
for (unsigned I = 1; I != N; ++I)
|
|
|
|
Args.push_back(Reader.DecodeIdentifierInfo(ReadUnalignedLE32(d)));
|
|
|
|
|
2009-05-23 06:45:36 +08:00
|
|
|
return SelTable.getSelector(N, Args.data());
|
2009-04-25 05:10:55 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-25 05:10:55 +08:00
|
|
|
data_type ReadData(Selector, const unsigned char* d, unsigned DataLen) {
|
|
|
|
using namespace clang::io;
|
|
|
|
|
|
|
|
data_type Result;
|
|
|
|
|
2010-08-05 01:20:04 +08:00
|
|
|
Result.ID = ReadUnalignedLE32(d);
|
|
|
|
unsigned NumInstanceMethods = ReadUnalignedLE16(d);
|
|
|
|
unsigned NumFactoryMethods = ReadUnalignedLE16(d);
|
|
|
|
|
2009-04-25 05:10:55 +08:00
|
|
|
// Load instance methods
|
|
|
|
ObjCMethodList *Prev = 0;
|
|
|
|
for (unsigned I = 0; I != NumInstanceMethods; ++I) {
|
2009-09-09 23:08:12 +08:00
|
|
|
ObjCMethodDecl *Method
|
2009-04-25 05:10:55 +08:00
|
|
|
= cast<ObjCMethodDecl>(Reader.GetDecl(ReadUnalignedLE32(d)));
|
2010-08-05 01:20:04 +08:00
|
|
|
if (!Result.Instance.Method) {
|
2009-04-25 05:10:55 +08:00
|
|
|
// This is the first method, which is the easy case.
|
2010-08-05 01:20:04 +08:00
|
|
|
Result.Instance.Method = Method;
|
|
|
|
Prev = &Result.Instance;
|
2009-04-25 05:10:55 +08:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2010-02-11 08:53:01 +08:00
|
|
|
ObjCMethodList *Mem =
|
|
|
|
Reader.getSema()->BumpAlloc.Allocate<ObjCMethodList>();
|
|
|
|
Prev->Next = new (Mem) ObjCMethodList(Method, 0);
|
2009-04-25 05:10:55 +08:00
|
|
|
Prev = Prev->Next;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Load factory methods
|
|
|
|
Prev = 0;
|
|
|
|
for (unsigned I = 0; I != NumFactoryMethods; ++I) {
|
2009-09-09 23:08:12 +08:00
|
|
|
ObjCMethodDecl *Method
|
2009-04-25 05:10:55 +08:00
|
|
|
= cast<ObjCMethodDecl>(Reader.GetDecl(ReadUnalignedLE32(d)));
|
2010-08-05 01:20:04 +08:00
|
|
|
if (!Result.Factory.Method) {
|
2009-04-25 05:10:55 +08:00
|
|
|
// This is the first method, which is the easy case.
|
2010-08-05 01:20:04 +08:00
|
|
|
Result.Factory.Method = Method;
|
|
|
|
Prev = &Result.Factory;
|
2009-04-25 05:10:55 +08:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2010-02-11 08:53:01 +08:00
|
|
|
ObjCMethodList *Mem =
|
|
|
|
Reader.getSema()->BumpAlloc.Allocate<ObjCMethodList>();
|
|
|
|
Prev->Next = new (Mem) ObjCMethodList(Method, 0);
|
2009-04-25 05:10:55 +08:00
|
|
|
Prev = Prev->Next;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
};
|
2009-09-09 23:08:12 +08:00
|
|
|
|
|
|
|
} // end anonymous namespace
|
2009-04-25 05:10:55 +08:00
|
|
|
|
|
|
|
/// \brief The on-disk hash table used for the global method pool.
|
2010-08-19 07:57:06 +08:00
|
|
|
typedef OnDiskChainedHashTable<ASTSelectorLookupTrait>
|
|
|
|
ASTSelectorLookupTable;
|
2009-04-25 05:10:55 +08:00
|
|
|
|
Lazy deserialization of the declaration chains associated with
identifiers from a precompiled header.
This patch changes the primary name lookup method for entities within
a precompiled header. Previously, we would load all of the names of
declarations at translation unit scope into a large DenseMap (inside
the TranslationUnitDecl's DeclContext), and then perform a special
"last resort" lookup into this DeclContext when we knew there was a
PCH file (see Sema::LookupName). Now, when we see an identifier named
for the first time, we load all of the declarations with that name
that are visible from the translation unit into the IdentifierInfo's
chain of declarations. Thus, the explicit "look into the translation
unit's DeclContext" code is gone, and Sema effectively uses the same
IdentifierInfo-based name lookup mechanism whether we are using a PCH
file or not.
This approach should help PCH scale with the size of the input program
rather than the size of the PCH file. The "Hello, World!" application
with Carbon.h as a PCH file now loads 20% of the identifiers in the
PCH file rather than 85% of the identifiers.
90% of the 20% of identifiers loaded are actually loaded when we
deserialize the preprocessor state. The next step is to make the
preprocessor load macros lazily, which should drastically reduce the
number of types, declarations, and identifiers loaded for "Hello,
World".
llvm-svn: 69737
2009-04-22 06:25:48 +08:00
|
|
|
namespace {
|
2010-08-19 07:57:06 +08:00
|
|
|
class ASTIdentifierLookupTrait {
|
2010-08-19 07:56:43 +08:00
|
|
|
ASTReader &Reader;
|
2010-07-22 06:31:37 +08:00
|
|
|
llvm::BitstreamCursor &Stream;
|
Lazy deserialization of the declaration chains associated with
identifiers from a precompiled header.
This patch changes the primary name lookup method for entities within
a precompiled header. Previously, we would load all of the names of
declarations at translation unit scope into a large DenseMap (inside
the TranslationUnitDecl's DeclContext), and then perform a special
"last resort" lookup into this DeclContext when we knew there was a
PCH file (see Sema::LookupName). Now, when we see an identifier named
for the first time, we load all of the declarations with that name
that are visible from the translation unit into the IdentifierInfo's
chain of declarations. Thus, the explicit "look into the translation
unit's DeclContext" code is gone, and Sema effectively uses the same
IdentifierInfo-based name lookup mechanism whether we are using a PCH
file or not.
This approach should help PCH scale with the size of the input program
rather than the size of the PCH file. The "Hello, World!" application
with Carbon.h as a PCH file now loads 20% of the identifiers in the
PCH file rather than 85% of the identifiers.
90% of the 20% of identifiers loaded are actually loaded when we
deserialize the preprocessor state. The next step is to make the
preprocessor load macros lazily, which should drastically reduce the
number of types, declarations, and identifiers loaded for "Hello,
World".
llvm-svn: 69737
2009-04-22 06:25:48 +08:00
|
|
|
|
|
|
|
// If we know the IdentifierInfo in advance, it is here and we will
|
|
|
|
// not build a new one. Used when deserializing information about an
|
2010-08-19 07:57:06 +08:00
|
|
|
// identifier that was constructed before the AST file was read.
|
Lazy deserialization of the declaration chains associated with
identifiers from a precompiled header.
This patch changes the primary name lookup method for entities within
a precompiled header. Previously, we would load all of the names of
declarations at translation unit scope into a large DenseMap (inside
the TranslationUnitDecl's DeclContext), and then perform a special
"last resort" lookup into this DeclContext when we knew there was a
PCH file (see Sema::LookupName). Now, when we see an identifier named
for the first time, we load all of the declarations with that name
that are visible from the translation unit into the IdentifierInfo's
chain of declarations. Thus, the explicit "look into the translation
unit's DeclContext" code is gone, and Sema effectively uses the same
IdentifierInfo-based name lookup mechanism whether we are using a PCH
file or not.
This approach should help PCH scale with the size of the input program
rather than the size of the PCH file. The "Hello, World!" application
with Carbon.h as a PCH file now loads 20% of the identifiers in the
PCH file rather than 85% of the identifiers.
90% of the 20% of identifiers loaded are actually loaded when we
deserialize the preprocessor state. The next step is to make the
preprocessor load macros lazily, which should drastically reduce the
number of types, declarations, and identifiers loaded for "Hello,
World".
llvm-svn: 69737
2009-04-22 06:25:48 +08:00
|
|
|
IdentifierInfo *KnownII;
|
|
|
|
|
|
|
|
public:
|
|
|
|
typedef IdentifierInfo * data_type;
|
|
|
|
|
|
|
|
typedef const std::pair<const char*, unsigned> external_key_type;
|
|
|
|
|
|
|
|
typedef external_key_type internal_key_type;
|
|
|
|
|
2010-08-19 07:57:06 +08:00
|
|
|
ASTIdentifierLookupTrait(ASTReader &Reader, llvm::BitstreamCursor &Stream,
|
2010-07-22 06:31:37 +08:00
|
|
|
IdentifierInfo *II = 0)
|
|
|
|
: Reader(Reader), Stream(Stream), KnownII(II) { }
|
2009-09-09 23:08:12 +08:00
|
|
|
|
Lazy deserialization of the declaration chains associated with
identifiers from a precompiled header.
This patch changes the primary name lookup method for entities within
a precompiled header. Previously, we would load all of the names of
declarations at translation unit scope into a large DenseMap (inside
the TranslationUnitDecl's DeclContext), and then perform a special
"last resort" lookup into this DeclContext when we knew there was a
PCH file (see Sema::LookupName). Now, when we see an identifier named
for the first time, we load all of the declarations with that name
that are visible from the translation unit into the IdentifierInfo's
chain of declarations. Thus, the explicit "look into the translation
unit's DeclContext" code is gone, and Sema effectively uses the same
IdentifierInfo-based name lookup mechanism whether we are using a PCH
file or not.
This approach should help PCH scale with the size of the input program
rather than the size of the PCH file. The "Hello, World!" application
with Carbon.h as a PCH file now loads 20% of the identifiers in the
PCH file rather than 85% of the identifiers.
90% of the 20% of identifiers loaded are actually loaded when we
deserialize the preprocessor state. The next step is to make the
preprocessor load macros lazily, which should drastically reduce the
number of types, declarations, and identifiers loaded for "Hello,
World".
llvm-svn: 69737
2009-04-22 06:25:48 +08:00
|
|
|
static bool EqualKey(const internal_key_type& a,
|
|
|
|
const internal_key_type& b) {
|
|
|
|
return (a.second == b.second) ? memcmp(a.first, b.first, a.second) == 0
|
|
|
|
: false;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
Lazy deserialization of the declaration chains associated with
identifiers from a precompiled header.
This patch changes the primary name lookup method for entities within
a precompiled header. Previously, we would load all of the names of
declarations at translation unit scope into a large DenseMap (inside
the TranslationUnitDecl's DeclContext), and then perform a special
"last resort" lookup into this DeclContext when we knew there was a
PCH file (see Sema::LookupName). Now, when we see an identifier named
for the first time, we load all of the declarations with that name
that are visible from the translation unit into the IdentifierInfo's
chain of declarations. Thus, the explicit "look into the translation
unit's DeclContext" code is gone, and Sema effectively uses the same
IdentifierInfo-based name lookup mechanism whether we are using a PCH
file or not.
This approach should help PCH scale with the size of the input program
rather than the size of the PCH file. The "Hello, World!" application
with Carbon.h as a PCH file now loads 20% of the identifiers in the
PCH file rather than 85% of the identifiers.
90% of the 20% of identifiers loaded are actually loaded when we
deserialize the preprocessor state. The next step is to make the
preprocessor load macros lazily, which should drastically reduce the
number of types, declarations, and identifiers loaded for "Hello,
World".
llvm-svn: 69737
2009-04-22 06:25:48 +08:00
|
|
|
static unsigned ComputeHash(const internal_key_type& a) {
|
2009-10-18 07:52:28 +08:00
|
|
|
return llvm::HashString(llvm::StringRef(a.first, a.second));
|
Lazy deserialization of the declaration chains associated with
identifiers from a precompiled header.
This patch changes the primary name lookup method for entities within
a precompiled header. Previously, we would load all of the names of
declarations at translation unit scope into a large DenseMap (inside
the TranslationUnitDecl's DeclContext), and then perform a special
"last resort" lookup into this DeclContext when we knew there was a
PCH file (see Sema::LookupName). Now, when we see an identifier named
for the first time, we load all of the declarations with that name
that are visible from the translation unit into the IdentifierInfo's
chain of declarations. Thus, the explicit "look into the translation
unit's DeclContext" code is gone, and Sema effectively uses the same
IdentifierInfo-based name lookup mechanism whether we are using a PCH
file or not.
This approach should help PCH scale with the size of the input program
rather than the size of the PCH file. The "Hello, World!" application
with Carbon.h as a PCH file now loads 20% of the identifiers in the
PCH file rather than 85% of the identifiers.
90% of the 20% of identifiers loaded are actually loaded when we
deserialize the preprocessor state. The next step is to make the
preprocessor load macros lazily, which should drastically reduce the
number of types, declarations, and identifiers loaded for "Hello,
World".
llvm-svn: 69737
2009-04-22 06:25:48 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
Lazy deserialization of the declaration chains associated with
identifiers from a precompiled header.
This patch changes the primary name lookup method for entities within
a precompiled header. Previously, we would load all of the names of
declarations at translation unit scope into a large DenseMap (inside
the TranslationUnitDecl's DeclContext), and then perform a special
"last resort" lookup into this DeclContext when we knew there was a
PCH file (see Sema::LookupName). Now, when we see an identifier named
for the first time, we load all of the declarations with that name
that are visible from the translation unit into the IdentifierInfo's
chain of declarations. Thus, the explicit "look into the translation
unit's DeclContext" code is gone, and Sema effectively uses the same
IdentifierInfo-based name lookup mechanism whether we are using a PCH
file or not.
This approach should help PCH scale with the size of the input program
rather than the size of the PCH file. The "Hello, World!" application
with Carbon.h as a PCH file now loads 20% of the identifiers in the
PCH file rather than 85% of the identifiers.
90% of the 20% of identifiers loaded are actually loaded when we
deserialize the preprocessor state. The next step is to make the
preprocessor load macros lazily, which should drastically reduce the
number of types, declarations, and identifiers loaded for "Hello,
World".
llvm-svn: 69737
2009-04-22 06:25:48 +08:00
|
|
|
// This hopefully will just get inlined and removed by the optimizer.
|
|
|
|
static const internal_key_type&
|
|
|
|
GetInternalKey(const external_key_type& x) { return x; }
|
2009-09-09 23:08:12 +08:00
|
|
|
|
Lazy deserialization of the declaration chains associated with
identifiers from a precompiled header.
This patch changes the primary name lookup method for entities within
a precompiled header. Previously, we would load all of the names of
declarations at translation unit scope into a large DenseMap (inside
the TranslationUnitDecl's DeclContext), and then perform a special
"last resort" lookup into this DeclContext when we knew there was a
PCH file (see Sema::LookupName). Now, when we see an identifier named
for the first time, we load all of the declarations with that name
that are visible from the translation unit into the IdentifierInfo's
chain of declarations. Thus, the explicit "look into the translation
unit's DeclContext" code is gone, and Sema effectively uses the same
IdentifierInfo-based name lookup mechanism whether we are using a PCH
file or not.
This approach should help PCH scale with the size of the input program
rather than the size of the PCH file. The "Hello, World!" application
with Carbon.h as a PCH file now loads 20% of the identifiers in the
PCH file rather than 85% of the identifiers.
90% of the 20% of identifiers loaded are actually loaded when we
deserialize the preprocessor state. The next step is to make the
preprocessor load macros lazily, which should drastically reduce the
number of types, declarations, and identifiers loaded for "Hello,
World".
llvm-svn: 69737
2009-04-22 06:25:48 +08:00
|
|
|
static std::pair<unsigned, unsigned>
|
|
|
|
ReadKeyDataLength(const unsigned char*& d) {
|
|
|
|
using namespace clang::io;
|
2009-04-26 04:26:24 +08:00
|
|
|
unsigned DataLen = ReadUnalignedLE16(d);
|
2009-04-26 05:04:17 +08:00
|
|
|
unsigned KeyLen = ReadUnalignedLE16(d);
|
Lazy deserialization of the declaration chains associated with
identifiers from a precompiled header.
This patch changes the primary name lookup method for entities within
a precompiled header. Previously, we would load all of the names of
declarations at translation unit scope into a large DenseMap (inside
the TranslationUnitDecl's DeclContext), and then perform a special
"last resort" lookup into this DeclContext when we knew there was a
PCH file (see Sema::LookupName). Now, when we see an identifier named
for the first time, we load all of the declarations with that name
that are visible from the translation unit into the IdentifierInfo's
chain of declarations. Thus, the explicit "look into the translation
unit's DeclContext" code is gone, and Sema effectively uses the same
IdentifierInfo-based name lookup mechanism whether we are using a PCH
file or not.
This approach should help PCH scale with the size of the input program
rather than the size of the PCH file. The "Hello, World!" application
with Carbon.h as a PCH file now loads 20% of the identifiers in the
PCH file rather than 85% of the identifiers.
90% of the 20% of identifiers loaded are actually loaded when we
deserialize the preprocessor state. The next step is to make the
preprocessor load macros lazily, which should drastically reduce the
number of types, declarations, and identifiers loaded for "Hello,
World".
llvm-svn: 69737
2009-04-22 06:25:48 +08:00
|
|
|
return std::make_pair(KeyLen, DataLen);
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
Lazy deserialization of the declaration chains associated with
identifiers from a precompiled header.
This patch changes the primary name lookup method for entities within
a precompiled header. Previously, we would load all of the names of
declarations at translation unit scope into a large DenseMap (inside
the TranslationUnitDecl's DeclContext), and then perform a special
"last resort" lookup into this DeclContext when we knew there was a
PCH file (see Sema::LookupName). Now, when we see an identifier named
for the first time, we load all of the declarations with that name
that are visible from the translation unit into the IdentifierInfo's
chain of declarations. Thus, the explicit "look into the translation
unit's DeclContext" code is gone, and Sema effectively uses the same
IdentifierInfo-based name lookup mechanism whether we are using a PCH
file or not.
This approach should help PCH scale with the size of the input program
rather than the size of the PCH file. The "Hello, World!" application
with Carbon.h as a PCH file now loads 20% of the identifiers in the
PCH file rather than 85% of the identifiers.
90% of the 20% of identifiers loaded are actually loaded when we
deserialize the preprocessor state. The next step is to make the
preprocessor load macros lazily, which should drastically reduce the
number of types, declarations, and identifiers loaded for "Hello,
World".
llvm-svn: 69737
2009-04-22 06:25:48 +08:00
|
|
|
static std::pair<const char*, unsigned>
|
|
|
|
ReadKey(const unsigned char* d, unsigned n) {
|
|
|
|
assert(n >= 2 && d[n-1] == '\0');
|
|
|
|
return std::make_pair((const char*) d, n-1);
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
|
|
|
IdentifierInfo *ReadData(const internal_key_type& k,
|
Lazy deserialization of the declaration chains associated with
identifiers from a precompiled header.
This patch changes the primary name lookup method for entities within
a precompiled header. Previously, we would load all of the names of
declarations at translation unit scope into a large DenseMap (inside
the TranslationUnitDecl's DeclContext), and then perform a special
"last resort" lookup into this DeclContext when we knew there was a
PCH file (see Sema::LookupName). Now, when we see an identifier named
for the first time, we load all of the declarations with that name
that are visible from the translation unit into the IdentifierInfo's
chain of declarations. Thus, the explicit "look into the translation
unit's DeclContext" code is gone, and Sema effectively uses the same
IdentifierInfo-based name lookup mechanism whether we are using a PCH
file or not.
This approach should help PCH scale with the size of the input program
rather than the size of the PCH file. The "Hello, World!" application
with Carbon.h as a PCH file now loads 20% of the identifiers in the
PCH file rather than 85% of the identifiers.
90% of the 20% of identifiers loaded are actually loaded when we
deserialize the preprocessor state. The next step is to make the
preprocessor load macros lazily, which should drastically reduce the
number of types, declarations, and identifiers loaded for "Hello,
World".
llvm-svn: 69737
2009-04-22 06:25:48 +08:00
|
|
|
const unsigned char* d,
|
|
|
|
unsigned DataLen) {
|
|
|
|
using namespace clang::io;
|
2010-08-19 07:57:32 +08:00
|
|
|
IdentID ID = ReadUnalignedLE32(d);
|
2009-04-29 05:18:29 +08:00
|
|
|
bool IsInteresting = ID & 0x01;
|
|
|
|
|
|
|
|
// Wipe out the "is interesting" bit.
|
|
|
|
ID = ID >> 1;
|
|
|
|
|
|
|
|
if (!IsInteresting) {
|
2010-07-28 07:01:28 +08:00
|
|
|
// For uninteresting identifiers, just build the IdentifierInfo
|
2009-04-29 05:18:29 +08:00
|
|
|
// and associate it with the persistent ID.
|
|
|
|
IdentifierInfo *II = KnownII;
|
|
|
|
if (!II)
|
2010-07-30 08:29:29 +08:00
|
|
|
II = &Reader.getIdentifierTable().getOwn(k.first, k.first + k.second);
|
2009-04-29 05:18:29 +08:00
|
|
|
Reader.SetIdentifierInfo(ID, II);
|
2010-08-19 07:57:06 +08:00
|
|
|
II->setIsFromAST();
|
2009-04-29 05:18:29 +08:00
|
|
|
return II;
|
|
|
|
}
|
|
|
|
|
2009-04-29 05:32:13 +08:00
|
|
|
unsigned Bits = ReadUnalignedLE16(d);
|
2009-04-23 02:49:13 +08:00
|
|
|
bool CPlusPlusOperatorKeyword = Bits & 0x01;
|
|
|
|
Bits >>= 1;
|
2010-08-12 06:55:12 +08:00
|
|
|
bool HasRevertedTokenIDToIdentifier = Bits & 0x01;
|
|
|
|
Bits >>= 1;
|
2009-04-23 02:49:13 +08:00
|
|
|
bool Poisoned = Bits & 0x01;
|
|
|
|
Bits >>= 1;
|
|
|
|
bool ExtensionToken = Bits & 0x01;
|
|
|
|
Bits >>= 1;
|
|
|
|
bool hasMacroDefinition = Bits & 0x01;
|
|
|
|
Bits >>= 1;
|
|
|
|
unsigned ObjCOrBuiltinID = Bits & 0x3FF;
|
|
|
|
Bits >>= 10;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-23 02:49:13 +08:00
|
|
|
assert(Bits == 0 && "Extra bits in the identifier?");
|
2009-04-29 05:32:13 +08:00
|
|
|
DataLen -= 6;
|
Lazy deserialization of the declaration chains associated with
identifiers from a precompiled header.
This patch changes the primary name lookup method for entities within
a precompiled header. Previously, we would load all of the names of
declarations at translation unit scope into a large DenseMap (inside
the TranslationUnitDecl's DeclContext), and then perform a special
"last resort" lookup into this DeclContext when we knew there was a
PCH file (see Sema::LookupName). Now, when we see an identifier named
for the first time, we load all of the declarations with that name
that are visible from the translation unit into the IdentifierInfo's
chain of declarations. Thus, the explicit "look into the translation
unit's DeclContext" code is gone, and Sema effectively uses the same
IdentifierInfo-based name lookup mechanism whether we are using a PCH
file or not.
This approach should help PCH scale with the size of the input program
rather than the size of the PCH file. The "Hello, World!" application
with Carbon.h as a PCH file now loads 20% of the identifiers in the
PCH file rather than 85% of the identifiers.
90% of the 20% of identifiers loaded are actually loaded when we
deserialize the preprocessor state. The next step is to make the
preprocessor load macros lazily, which should drastically reduce the
number of types, declarations, and identifiers loaded for "Hello,
World".
llvm-svn: 69737
2009-04-22 06:25:48 +08:00
|
|
|
|
|
|
|
// Build the IdentifierInfo itself and link the identifier ID with
|
|
|
|
// the new IdentifierInfo.
|
|
|
|
IdentifierInfo *II = KnownII;
|
|
|
|
if (!II)
|
2010-07-30 08:29:29 +08:00
|
|
|
II = &Reader.getIdentifierTable().getOwn(k.first, k.first + k.second);
|
Lazy deserialization of the declaration chains associated with
identifiers from a precompiled header.
This patch changes the primary name lookup method for entities within
a precompiled header. Previously, we would load all of the names of
declarations at translation unit scope into a large DenseMap (inside
the TranslationUnitDecl's DeclContext), and then perform a special
"last resort" lookup into this DeclContext when we knew there was a
PCH file (see Sema::LookupName). Now, when we see an identifier named
for the first time, we load all of the declarations with that name
that are visible from the translation unit into the IdentifierInfo's
chain of declarations. Thus, the explicit "look into the translation
unit's DeclContext" code is gone, and Sema effectively uses the same
IdentifierInfo-based name lookup mechanism whether we are using a PCH
file or not.
This approach should help PCH scale with the size of the input program
rather than the size of the PCH file. The "Hello, World!" application
with Carbon.h as a PCH file now loads 20% of the identifiers in the
PCH file rather than 85% of the identifiers.
90% of the 20% of identifiers loaded are actually loaded when we
deserialize the preprocessor state. The next step is to make the
preprocessor load macros lazily, which should drastically reduce the
number of types, declarations, and identifiers loaded for "Hello,
World".
llvm-svn: 69737
2009-04-22 06:25:48 +08:00
|
|
|
Reader.SetIdentifierInfo(ID, II);
|
|
|
|
|
2009-04-23 02:49:13 +08:00
|
|
|
// Set or check the various bits in the IdentifierInfo structure.
|
2010-08-12 06:55:12 +08:00
|
|
|
// Token IDs are read-only.
|
|
|
|
if (HasRevertedTokenIDToIdentifier)
|
|
|
|
II->RevertTokenIDToIdentifier();
|
2009-04-23 02:49:13 +08:00
|
|
|
II->setObjCOrBuiltinID(ObjCOrBuiltinID);
|
2009-09-09 23:08:12 +08:00
|
|
|
assert(II->isExtensionToken() == ExtensionToken &&
|
2009-04-23 02:49:13 +08:00
|
|
|
"Incorrect extension token flag");
|
|
|
|
(void)ExtensionToken;
|
|
|
|
II->setIsPoisoned(Poisoned);
|
|
|
|
assert(II->isCPlusPlusOperatorKeyword() == CPlusPlusOperatorKeyword &&
|
|
|
|
"Incorrect C++ operator keyword flag");
|
|
|
|
(void)CPlusPlusOperatorKeyword;
|
|
|
|
|
2009-04-22 07:56:24 +08:00
|
|
|
// If this identifier is a macro, deserialize the macro
|
|
|
|
// definition.
|
|
|
|
if (hasMacroDefinition) {
|
2009-04-29 05:32:13 +08:00
|
|
|
uint32_t Offset = ReadUnalignedLE32(d);
|
2010-07-22 06:31:37 +08:00
|
|
|
Reader.ReadMacroRecord(Stream, Offset);
|
2009-04-29 05:32:13 +08:00
|
|
|
DataLen -= 4;
|
2009-04-22 07:56:24 +08:00
|
|
|
}
|
Lazy deserialization of the declaration chains associated with
identifiers from a precompiled header.
This patch changes the primary name lookup method for entities within
a precompiled header. Previously, we would load all of the names of
declarations at translation unit scope into a large DenseMap (inside
the TranslationUnitDecl's DeclContext), and then perform a special
"last resort" lookup into this DeclContext when we knew there was a
PCH file (see Sema::LookupName). Now, when we see an identifier named
for the first time, we load all of the declarations with that name
that are visible from the translation unit into the IdentifierInfo's
chain of declarations. Thus, the explicit "look into the translation
unit's DeclContext" code is gone, and Sema effectively uses the same
IdentifierInfo-based name lookup mechanism whether we are using a PCH
file or not.
This approach should help PCH scale with the size of the input program
rather than the size of the PCH file. The "Hello, World!" application
with Carbon.h as a PCH file now loads 20% of the identifiers in the
PCH file rather than 85% of the identifiers.
90% of the 20% of identifiers loaded are actually loaded when we
deserialize the preprocessor state. The next step is to make the
preprocessor load macros lazily, which should drastically reduce the
number of types, declarations, and identifiers loaded for "Hello,
World".
llvm-svn: 69737
2009-04-22 06:25:48 +08:00
|
|
|
|
|
|
|
// Read all of the declarations visible at global scope with this
|
|
|
|
// name.
|
2009-04-28 06:17:41 +08:00
|
|
|
if (Reader.getContext() == 0) return II;
|
2009-07-07 02:54:52 +08:00
|
|
|
if (DataLen > 0) {
|
|
|
|
llvm::SmallVector<uint32_t, 4> DeclIDs;
|
|
|
|
for (; DataLen > 0; DataLen -= 4)
|
|
|
|
DeclIDs.push_back(ReadUnalignedLE32(d));
|
|
|
|
Reader.SetGloballyVisibleDecls(II, DeclIDs);
|
2009-06-19 08:03:23 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-08-19 07:57:06 +08:00
|
|
|
II->setIsFromAST();
|
2009-06-19 08:03:23 +08:00
|
|
|
return II;
|
2009-04-29 02:58:38 +08:00
|
|
|
}
|
2009-06-19 08:03:23 +08:00
|
|
|
};
|
2009-09-09 23:08:12 +08:00
|
|
|
|
|
|
|
} // end anonymous namespace
|
2009-04-29 02:58:38 +08:00
|
|
|
|
2009-06-19 08:03:23 +08:00
|
|
|
/// \brief The on-disk hash table used to contain information about
|
|
|
|
/// all of the identifiers in the program.
|
2010-08-19 07:57:06 +08:00
|
|
|
typedef OnDiskChainedHashTable<ASTIdentifierLookupTrait>
|
|
|
|
ASTIdentifierLookupTable;
|
2009-04-29 02:58:38 +08:00
|
|
|
|
2010-08-21 00:04:27 +08:00
|
|
|
namespace {
|
|
|
|
class ASTDeclContextNameLookupTrait {
|
|
|
|
ASTReader &Reader;
|
|
|
|
|
|
|
|
public:
|
|
|
|
/// \brief Pair of begin/end iterators for DeclIDs.
|
|
|
|
typedef std::pair<DeclID *, DeclID *> data_type;
|
|
|
|
|
|
|
|
/// \brief Special internal key for declaration names.
|
|
|
|
/// The hash table creates keys for comparison; we do not create
|
|
|
|
/// a DeclarationName for the internal key to avoid deserializing types.
|
|
|
|
struct DeclNameKey {
|
|
|
|
DeclarationName::NameKind Kind;
|
|
|
|
uint64_t Data;
|
|
|
|
DeclNameKey() : Kind((DeclarationName::NameKind)0), Data(0) { }
|
|
|
|
};
|
|
|
|
|
|
|
|
typedef DeclarationName external_key_type;
|
|
|
|
typedef DeclNameKey internal_key_type;
|
|
|
|
|
|
|
|
explicit ASTDeclContextNameLookupTrait(ASTReader &Reader) : Reader(Reader) { }
|
|
|
|
|
|
|
|
static bool EqualKey(const internal_key_type& a,
|
|
|
|
const internal_key_type& b) {
|
|
|
|
return a.Kind == b.Kind && a.Data == b.Data;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned ComputeHash(const DeclNameKey &Key) const {
|
|
|
|
llvm::FoldingSetNodeID ID;
|
|
|
|
ID.AddInteger(Key.Kind);
|
|
|
|
|
|
|
|
switch (Key.Kind) {
|
|
|
|
case DeclarationName::Identifier:
|
|
|
|
case DeclarationName::CXXLiteralOperatorName:
|
|
|
|
ID.AddString(((IdentifierInfo*)Key.Data)->getName());
|
|
|
|
break;
|
|
|
|
case DeclarationName::ObjCZeroArgSelector:
|
|
|
|
case DeclarationName::ObjCOneArgSelector:
|
|
|
|
case DeclarationName::ObjCMultiArgSelector:
|
|
|
|
ID.AddInteger(serialization::ComputeHash(Selector(Key.Data)));
|
|
|
|
break;
|
|
|
|
case DeclarationName::CXXConstructorName:
|
|
|
|
case DeclarationName::CXXDestructorName:
|
|
|
|
case DeclarationName::CXXConversionFunctionName:
|
|
|
|
ID.AddInteger((TypeID)Key.Data);
|
|
|
|
break;
|
|
|
|
case DeclarationName::CXXOperatorName:
|
|
|
|
ID.AddInteger((OverloadedOperatorKind)Key.Data);
|
|
|
|
break;
|
|
|
|
case DeclarationName::CXXUsingDirective:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ID.ComputeHash();
|
|
|
|
}
|
|
|
|
|
|
|
|
internal_key_type GetInternalKey(const external_key_type& Name) const {
|
|
|
|
DeclNameKey Key;
|
|
|
|
Key.Kind = Name.getNameKind();
|
|
|
|
switch (Name.getNameKind()) {
|
|
|
|
case DeclarationName::Identifier:
|
|
|
|
Key.Data = (uint64_t)Name.getAsIdentifierInfo();
|
|
|
|
break;
|
|
|
|
case DeclarationName::ObjCZeroArgSelector:
|
|
|
|
case DeclarationName::ObjCOneArgSelector:
|
|
|
|
case DeclarationName::ObjCMultiArgSelector:
|
|
|
|
Key.Data = (uint64_t)Name.getObjCSelector().getAsOpaquePtr();
|
|
|
|
break;
|
|
|
|
case DeclarationName::CXXConstructorName:
|
|
|
|
case DeclarationName::CXXDestructorName:
|
|
|
|
case DeclarationName::CXXConversionFunctionName:
|
|
|
|
Key.Data = Reader.GetTypeID(Name.getCXXNameType());
|
|
|
|
break;
|
|
|
|
case DeclarationName::CXXOperatorName:
|
|
|
|
Key.Data = Name.getCXXOverloadedOperator();
|
|
|
|
break;
|
|
|
|
case DeclarationName::CXXLiteralOperatorName:
|
|
|
|
Key.Data = (uint64_t)Name.getCXXLiteralIdentifier();
|
|
|
|
break;
|
|
|
|
case DeclarationName::CXXUsingDirective:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Key;
|
|
|
|
}
|
|
|
|
|
2010-08-21 07:35:55 +08:00
|
|
|
external_key_type GetExternalKey(const internal_key_type& Key) const {
|
|
|
|
ASTContext *Context = Reader.getContext();
|
|
|
|
switch (Key.Kind) {
|
|
|
|
case DeclarationName::Identifier:
|
|
|
|
return DeclarationName((IdentifierInfo*)Key.Data);
|
|
|
|
|
|
|
|
case DeclarationName::ObjCZeroArgSelector:
|
|
|
|
case DeclarationName::ObjCOneArgSelector:
|
|
|
|
case DeclarationName::ObjCMultiArgSelector:
|
|
|
|
return DeclarationName(Selector(Key.Data));
|
|
|
|
|
|
|
|
case DeclarationName::CXXConstructorName:
|
|
|
|
return Context->DeclarationNames.getCXXConstructorName(
|
|
|
|
Context->getCanonicalType(Reader.GetType(Key.Data)));
|
|
|
|
|
|
|
|
case DeclarationName::CXXDestructorName:
|
|
|
|
return Context->DeclarationNames.getCXXDestructorName(
|
|
|
|
Context->getCanonicalType(Reader.GetType(Key.Data)));
|
|
|
|
|
|
|
|
case DeclarationName::CXXConversionFunctionName:
|
|
|
|
return Context->DeclarationNames.getCXXConversionFunctionName(
|
|
|
|
Context->getCanonicalType(Reader.GetType(Key.Data)));
|
|
|
|
|
|
|
|
case DeclarationName::CXXOperatorName:
|
|
|
|
return Context->DeclarationNames.getCXXOperatorName(
|
|
|
|
(OverloadedOperatorKind)Key.Data);
|
|
|
|
|
|
|
|
case DeclarationName::CXXLiteralOperatorName:
|
|
|
|
return Context->DeclarationNames.getCXXLiteralOperatorName(
|
|
|
|
(IdentifierInfo*)Key.Data);
|
|
|
|
|
|
|
|
case DeclarationName::CXXUsingDirective:
|
|
|
|
return DeclarationName::getUsingDirectiveName();
|
|
|
|
}
|
|
|
|
|
|
|
|
llvm_unreachable("Invalid Name Kind ?");
|
|
|
|
}
|
|
|
|
|
2010-08-21 00:04:27 +08:00
|
|
|
static std::pair<unsigned, unsigned>
|
|
|
|
ReadKeyDataLength(const unsigned char*& d) {
|
|
|
|
using namespace clang::io;
|
|
|
|
unsigned KeyLen = ReadUnalignedLE16(d);
|
|
|
|
unsigned DataLen = ReadUnalignedLE16(d);
|
|
|
|
return std::make_pair(KeyLen, DataLen);
|
|
|
|
}
|
|
|
|
|
|
|
|
internal_key_type ReadKey(const unsigned char* d, unsigned) {
|
|
|
|
using namespace clang::io;
|
|
|
|
|
|
|
|
DeclNameKey Key;
|
|
|
|
Key.Kind = (DeclarationName::NameKind)*d++;
|
|
|
|
switch (Key.Kind) {
|
|
|
|
case DeclarationName::Identifier:
|
|
|
|
Key.Data = (uint64_t)Reader.DecodeIdentifierInfo(ReadUnalignedLE32(d));
|
|
|
|
break;
|
|
|
|
case DeclarationName::ObjCZeroArgSelector:
|
|
|
|
case DeclarationName::ObjCOneArgSelector:
|
|
|
|
case DeclarationName::ObjCMultiArgSelector:
|
|
|
|
Key.Data =
|
|
|
|
(uint64_t)Reader.DecodeSelector(ReadUnalignedLE32(d)).getAsOpaquePtr();
|
|
|
|
break;
|
|
|
|
case DeclarationName::CXXConstructorName:
|
|
|
|
case DeclarationName::CXXDestructorName:
|
|
|
|
case DeclarationName::CXXConversionFunctionName:
|
|
|
|
Key.Data = ReadUnalignedLE32(d); // TypeID
|
|
|
|
break;
|
|
|
|
case DeclarationName::CXXOperatorName:
|
|
|
|
Key.Data = *d++; // OverloadedOperatorKind
|
|
|
|
break;
|
|
|
|
case DeclarationName::CXXLiteralOperatorName:
|
|
|
|
Key.Data = (uint64_t)Reader.DecodeIdentifierInfo(ReadUnalignedLE32(d));
|
|
|
|
break;
|
|
|
|
case DeclarationName::CXXUsingDirective:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Key;
|
|
|
|
}
|
|
|
|
|
|
|
|
data_type ReadData(internal_key_type, const unsigned char* d,
|
|
|
|
unsigned DataLen) {
|
|
|
|
using namespace clang::io;
|
|
|
|
unsigned NumDecls = ReadUnalignedLE16(d);
|
|
|
|
DeclID *Start = (DeclID *)d;
|
|
|
|
return std::make_pair(Start, Start + NumDecls);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
} // end anonymous namespace
|
|
|
|
|
|
|
|
/// \brief The on-disk hash table used for the DeclContext's Name lookup table.
|
|
|
|
typedef OnDiskChainedHashTable<ASTDeclContextNameLookupTrait>
|
|
|
|
ASTDeclContextNameLookupTable;
|
|
|
|
|
2010-08-21 00:04:35 +08:00
|
|
|
bool ASTReader::ReadDeclContextStorage(llvm::BitstreamCursor &Cursor,
|
|
|
|
const std::pair<uint64_t, uint64_t> &Offsets,
|
|
|
|
DeclContextInfo &Info) {
|
|
|
|
SavedStreamPosition SavedPosition(Cursor);
|
|
|
|
// First the lexical decls.
|
|
|
|
if (Offsets.first != 0) {
|
|
|
|
Cursor.JumpToBit(Offsets.first);
|
|
|
|
|
|
|
|
RecordData Record;
|
|
|
|
const char *Blob;
|
|
|
|
unsigned BlobLen;
|
|
|
|
unsigned Code = Cursor.ReadCode();
|
|
|
|
unsigned RecCode = Cursor.ReadRecord(Code, Record, &Blob, &BlobLen);
|
|
|
|
if (RecCode != DECL_CONTEXT_LEXICAL) {
|
|
|
|
Error("Expected lexical block");
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
Info.LexicalDecls = reinterpret_cast<const DeclID*>(Blob);
|
|
|
|
Info.NumLexicalDecls = BlobLen / sizeof(DeclID);
|
|
|
|
} else {
|
|
|
|
Info.LexicalDecls = 0;
|
|
|
|
Info.NumLexicalDecls = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Now the lookup table.
|
|
|
|
if (Offsets.second != 0) {
|
|
|
|
Cursor.JumpToBit(Offsets.second);
|
|
|
|
|
|
|
|
RecordData Record;
|
|
|
|
const char *Blob;
|
|
|
|
unsigned BlobLen;
|
|
|
|
unsigned Code = Cursor.ReadCode();
|
|
|
|
unsigned RecCode = Cursor.ReadRecord(Code, Record, &Blob, &BlobLen);
|
|
|
|
if (RecCode != DECL_CONTEXT_VISIBLE) {
|
|
|
|
Error("Expected visible lookup table block");
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
Info.NameLookupTableData
|
|
|
|
= ASTDeclContextNameLookupTable::Create(
|
|
|
|
(const unsigned char *)Blob + Record[0],
|
|
|
|
(const unsigned char *)Blob,
|
|
|
|
ASTDeclContextNameLookupTrait(*this));
|
2010-08-24 08:50:00 +08:00
|
|
|
} else {
|
|
|
|
Info.NameLookupTableData = 0;
|
2010-08-21 00:04:35 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:43 +08:00
|
|
|
void ASTReader::Error(const char *Msg) {
|
2010-03-18 08:56:54 +08:00
|
|
|
Diag(diag::err_fe_pch_malformed) << Msg;
|
2009-04-29 02:58:38 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:57:06 +08:00
|
|
|
/// \brief Tell the AST listener about the predefines buffers in the chain.
|
2010-08-19 07:56:43 +08:00
|
|
|
bool ASTReader::CheckPredefinesBuffers() {
|
2009-06-19 08:03:23 +08:00
|
|
|
if (Listener)
|
2010-07-15 01:49:11 +08:00
|
|
|
return Listener->ReadPredefinesBuffer(PCHPredefinesBuffers,
|
2009-11-11 13:29:04 +08:00
|
|
|
ActualOriginalFileName,
|
2009-06-19 08:03:23 +08:00
|
|
|
SuggestedPredefines);
|
2009-04-29 02:58:38 +08:00
|
|
|
return false;
|
2009-04-11 07:10:45 +08:00
|
|
|
}
|
|
|
|
|
2009-04-28 02:38:38 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Source Manager Deserialization
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2009-04-14 00:31:14 +08:00
|
|
|
/// \brief Read the line table in the source manager block.
|
|
|
|
/// \returns true if ther was an error.
|
2010-08-19 07:56:43 +08:00
|
|
|
bool ASTReader::ParseLineTable(llvm::SmallVectorImpl<uint64_t> &Record) {
|
2009-04-14 00:31:14 +08:00
|
|
|
unsigned Idx = 0;
|
|
|
|
LineTableInfo &LineTable = SourceMgr.getLineTable();
|
|
|
|
|
|
|
|
// Parse the file names
|
2009-04-14 01:12:42 +08:00
|
|
|
std::map<int, int> FileIDs;
|
|
|
|
for (int I = 0, N = Record[Idx++]; I != N; ++I) {
|
2009-04-14 00:31:14 +08:00
|
|
|
// Extract the file name
|
|
|
|
unsigned FilenameLen = Record[Idx++];
|
|
|
|
std::string Filename(&Record[Idx], &Record[Idx] + FilenameLen);
|
|
|
|
Idx += FilenameLen;
|
2009-07-07 08:12:59 +08:00
|
|
|
MaybeAddSystemRootToFilename(Filename);
|
2009-09-09 23:08:12 +08:00
|
|
|
FileIDs[I] = LineTable.getLineTableFilenameID(Filename.c_str(),
|
2009-04-14 01:12:42 +08:00
|
|
|
Filename.size());
|
2009-04-14 00:31:14 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Parse the line entries
|
|
|
|
std::vector<LineEntry> Entries;
|
|
|
|
while (Idx < Record.size()) {
|
2010-07-02 19:55:05 +08:00
|
|
|
int FID = Record[Idx++];
|
2009-04-14 00:31:14 +08:00
|
|
|
|
|
|
|
// Extract the line entries
|
|
|
|
unsigned NumEntries = Record[Idx++];
|
2010-07-02 19:55:05 +08:00
|
|
|
assert(NumEntries && "Numentries is 00000");
|
2009-04-14 00:31:14 +08:00
|
|
|
Entries.clear();
|
|
|
|
Entries.reserve(NumEntries);
|
|
|
|
for (unsigned I = 0; I != NumEntries; ++I) {
|
|
|
|
unsigned FileOffset = Record[Idx++];
|
|
|
|
unsigned LineNo = Record[Idx++];
|
2010-07-02 19:55:05 +08:00
|
|
|
int FilenameID = FileIDs[Record[Idx++]];
|
2009-09-09 23:08:12 +08:00
|
|
|
SrcMgr::CharacteristicKind FileKind
|
2009-04-14 00:31:14 +08:00
|
|
|
= (SrcMgr::CharacteristicKind)Record[Idx++];
|
|
|
|
unsigned IncludeOffset = Record[Idx++];
|
|
|
|
Entries.push_back(LineEntry::get(FileOffset, LineNo, FilenameID,
|
|
|
|
FileKind, IncludeOffset));
|
|
|
|
}
|
|
|
|
LineTable.AddEntry(FID, Entries);
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2009-04-28 02:38:38 +08:00
|
|
|
namespace {
|
|
|
|
|
2010-08-19 07:57:06 +08:00
|
|
|
class ASTStatData {
|
2009-04-28 02:38:38 +08:00
|
|
|
public:
|
|
|
|
const bool hasStat;
|
|
|
|
const ino_t ino;
|
|
|
|
const dev_t dev;
|
|
|
|
const mode_t mode;
|
|
|
|
const time_t mtime;
|
|
|
|
const off_t size;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-08-19 07:57:06 +08:00
|
|
|
ASTStatData(ino_t i, dev_t d, mode_t mo, time_t m, off_t s)
|
2009-09-09 23:08:12 +08:00
|
|
|
: hasStat(true), ino(i), dev(d), mode(mo), mtime(m), size(s) {}
|
|
|
|
|
2010-08-19 07:57:06 +08:00
|
|
|
ASTStatData()
|
2009-04-28 02:38:38 +08:00
|
|
|
: hasStat(false), ino(0), dev(0), mode(0), mtime(0), size(0) {}
|
|
|
|
};
|
|
|
|
|
2010-08-19 07:57:06 +08:00
|
|
|
class ASTStatLookupTrait {
|
2009-04-28 02:38:38 +08:00
|
|
|
public:
|
|
|
|
typedef const char *external_key_type;
|
|
|
|
typedef const char *internal_key_type;
|
|
|
|
|
2010-08-19 07:57:06 +08:00
|
|
|
typedef ASTStatData data_type;
|
2009-04-28 02:38:38 +08:00
|
|
|
|
|
|
|
static unsigned ComputeHash(const char *path) {
|
2009-10-18 07:52:28 +08:00
|
|
|
return llvm::HashString(path);
|
2009-04-28 02:38:38 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static internal_key_type GetInternalKey(const char *path) { return path; }
|
|
|
|
|
|
|
|
static bool EqualKey(internal_key_type a, internal_key_type b) {
|
|
|
|
return strcmp(a, b) == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static std::pair<unsigned, unsigned>
|
|
|
|
ReadKeyDataLength(const unsigned char*& d) {
|
|
|
|
unsigned KeyLen = (unsigned) clang::io::ReadUnalignedLE16(d);
|
|
|
|
unsigned DataLen = (unsigned) *d++;
|
|
|
|
return std::make_pair(KeyLen + 1, DataLen);
|
|
|
|
}
|
|
|
|
|
|
|
|
static internal_key_type ReadKey(const unsigned char *d, unsigned) {
|
|
|
|
return (const char *)d;
|
|
|
|
}
|
|
|
|
|
|
|
|
static data_type ReadData(const internal_key_type, const unsigned char *d,
|
|
|
|
unsigned /*DataLen*/) {
|
|
|
|
using namespace clang::io;
|
|
|
|
|
|
|
|
if (*d++ == 1)
|
|
|
|
return data_type();
|
|
|
|
|
|
|
|
ino_t ino = (ino_t) ReadUnalignedLE32(d);
|
|
|
|
dev_t dev = (dev_t) ReadUnalignedLE32(d);
|
|
|
|
mode_t mode = (mode_t) ReadUnalignedLE16(d);
|
2009-09-09 23:08:12 +08:00
|
|
|
time_t mtime = (time_t) ReadUnalignedLE64(d);
|
2009-04-28 02:38:38 +08:00
|
|
|
off_t size = (off_t) ReadUnalignedLE64(d);
|
|
|
|
return data_type(ino, dev, mode, mtime, size);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/// \brief stat() cache for precompiled headers.
|
|
|
|
///
|
|
|
|
/// This cache is very similar to the stat cache used by pretokenized
|
|
|
|
/// headers.
|
2010-08-19 07:57:06 +08:00
|
|
|
class ASTStatCache : public StatSysCallCache {
|
|
|
|
typedef OnDiskChainedHashTable<ASTStatLookupTrait> CacheTy;
|
2009-04-28 02:38:38 +08:00
|
|
|
CacheTy *Cache;
|
|
|
|
|
|
|
|
unsigned &NumStatHits, &NumStatMisses;
|
2009-09-09 23:08:12 +08:00
|
|
|
public:
|
2010-08-19 07:57:06 +08:00
|
|
|
ASTStatCache(const unsigned char *Buckets,
|
2009-04-28 02:38:38 +08:00
|
|
|
const unsigned char *Base,
|
|
|
|
unsigned &NumStatHits,
|
2009-09-09 23:08:12 +08:00
|
|
|
unsigned &NumStatMisses)
|
2009-04-28 02:38:38 +08:00
|
|
|
: Cache(0), NumStatHits(NumStatHits), NumStatMisses(NumStatMisses) {
|
|
|
|
Cache = CacheTy::Create(Buckets, Base);
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:57:06 +08:00
|
|
|
~ASTStatCache() { delete Cache; }
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-28 02:38:38 +08:00
|
|
|
int stat(const char *path, struct stat *buf) {
|
2010-08-19 07:57:06 +08:00
|
|
|
// Do the lookup for the file's data in the AST file.
|
2009-04-28 02:38:38 +08:00
|
|
|
CacheTy::iterator I = Cache->find(path);
|
|
|
|
|
2010-08-19 07:57:06 +08:00
|
|
|
// If we don't get a hit in the AST file just forward to 'stat'.
|
2009-04-28 02:38:38 +08:00
|
|
|
if (I == Cache->end()) {
|
|
|
|
++NumStatMisses;
|
2009-10-17 02:18:30 +08:00
|
|
|
return StatSysCallCache::stat(path, buf);
|
2009-04-28 02:38:38 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-28 02:38:38 +08:00
|
|
|
++NumStatHits;
|
2010-08-19 07:57:06 +08:00
|
|
|
ASTStatData Data = *I;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-28 02:38:38 +08:00
|
|
|
if (!Data.hasStat)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
buf->st_ino = Data.ino;
|
|
|
|
buf->st_dev = Data.dev;
|
|
|
|
buf->st_mtime = Data.mtime;
|
|
|
|
buf->st_mode = Data.mode;
|
|
|
|
buf->st_size = Data.size;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
} // end anonymous namespace
|
|
|
|
|
|
|
|
|
2010-07-20 04:52:06 +08:00
|
|
|
/// \brief Read a source manager block
|
2010-08-19 07:56:43 +08:00
|
|
|
ASTReader::ASTReadResult ASTReader::ReadSourceManagerBlock(PerFileData &F) {
|
2009-04-10 11:52:48 +08:00
|
|
|
using namespace SrcMgr;
|
2009-04-27 14:38:32 +08:00
|
|
|
|
2010-07-20 04:52:06 +08:00
|
|
|
llvm::BitstreamCursor &SLocEntryCursor = F.SLocEntryCursor;
|
2010-07-17 01:50:48 +08:00
|
|
|
|
2009-04-27 14:38:32 +08:00
|
|
|
// Set the source-location entry cursor to the current position in
|
|
|
|
// the stream. This cursor will be used to read the contents of the
|
|
|
|
// source manager block initially, and then lazily read
|
|
|
|
// source-location entries as needed.
|
2010-07-20 04:52:06 +08:00
|
|
|
SLocEntryCursor = F.Stream;
|
2009-04-27 14:38:32 +08:00
|
|
|
|
|
|
|
// The stream itself is going to skip over the source manager block.
|
2010-07-20 04:52:06 +08:00
|
|
|
if (F.Stream.SkipBlock()) {
|
2010-08-19 07:57:06 +08:00
|
|
|
Error("malformed block record in AST file");
|
2009-04-27 14:38:32 +08:00
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Enter the source manager block.
|
2010-08-19 07:57:32 +08:00
|
|
|
if (SLocEntryCursor.EnterSubBlock(SOURCE_MANAGER_BLOCK_ID)) {
|
2010-08-19 07:57:06 +08:00
|
|
|
Error("malformed source manager block record in AST file");
|
2009-04-11 07:10:45 +08:00
|
|
|
return Failure;
|
|
|
|
}
|
2009-04-10 11:52:48 +08:00
|
|
|
|
|
|
|
RecordData Record;
|
|
|
|
while (true) {
|
2009-04-27 14:38:32 +08:00
|
|
|
unsigned Code = SLocEntryCursor.ReadCode();
|
2009-04-10 11:52:48 +08:00
|
|
|
if (Code == llvm::bitc::END_BLOCK) {
|
2009-04-27 14:38:32 +08:00
|
|
|
if (SLocEntryCursor.ReadBlockEnd()) {
|
2010-08-19 07:57:06 +08:00
|
|
|
Error("error at end of Source Manager block in AST file");
|
2009-04-11 07:10:45 +08:00
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
return Success;
|
2009-04-10 11:52:48 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-10 11:52:48 +08:00
|
|
|
if (Code == llvm::bitc::ENTER_SUBBLOCK) {
|
|
|
|
// No known subblocks, always skip them.
|
2009-04-27 14:38:32 +08:00
|
|
|
SLocEntryCursor.ReadSubBlockID();
|
|
|
|
if (SLocEntryCursor.SkipBlock()) {
|
2010-08-19 07:57:06 +08:00
|
|
|
Error("malformed block record in AST file");
|
2009-04-11 07:10:45 +08:00
|
|
|
return Failure;
|
|
|
|
}
|
2009-04-10 11:52:48 +08:00
|
|
|
continue;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-10 11:52:48 +08:00
|
|
|
if (Code == llvm::bitc::DEFINE_ABBREV) {
|
2009-04-27 14:38:32 +08:00
|
|
|
SLocEntryCursor.ReadAbbrevRecord();
|
2009-04-10 11:52:48 +08:00
|
|
|
continue;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-10 11:52:48 +08:00
|
|
|
// Read a record.
|
|
|
|
const char *BlobStart;
|
|
|
|
unsigned BlobLen;
|
|
|
|
Record.clear();
|
2009-04-27 14:38:32 +08:00
|
|
|
switch (SLocEntryCursor.ReadRecord(Code, Record, &BlobStart, &BlobLen)) {
|
2009-04-10 11:52:48 +08:00
|
|
|
default: // Default behavior: ignore.
|
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case SM_LINE_TABLE:
|
2010-07-21 05:20:32 +08:00
|
|
|
if (ParseLineTable(Record))
|
2009-04-14 00:31:14 +08:00
|
|
|
return Failure;
|
2009-04-15 07:22:57 +08:00
|
|
|
break;
|
2009-04-26 08:07:37 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case SM_SLOC_FILE_ENTRY:
|
|
|
|
case SM_SLOC_BUFFER_ENTRY:
|
|
|
|
case SM_SLOC_INSTANTIATION_ENTRY:
|
2009-04-27 14:38:32 +08:00
|
|
|
// Once we hit one of the source location entries, we're done.
|
|
|
|
return Success;
|
2009-04-10 11:52:48 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-07-21 05:50:20 +08:00
|
|
|
/// \brief Get a cursor that's correctly positioned for reading the source
|
|
|
|
/// location entry with the given ID.
|
2010-08-19 07:56:43 +08:00
|
|
|
llvm::BitstreamCursor &ASTReader::SLocCursorForID(unsigned ID) {
|
2010-07-21 05:50:20 +08:00
|
|
|
assert(ID != 0 && ID <= TotalNumSLocEntries &&
|
|
|
|
"SLocCursorForID should only be called for real IDs.");
|
|
|
|
|
|
|
|
ID -= 1;
|
|
|
|
PerFileData *F = 0;
|
|
|
|
for (unsigned I = 0, N = Chain.size(); I != N; ++I) {
|
|
|
|
F = Chain[N - I - 1];
|
|
|
|
if (ID < F->LocalNumSLocEntries)
|
|
|
|
break;
|
|
|
|
ID -= F->LocalNumSLocEntries;
|
|
|
|
}
|
|
|
|
assert(F && F->LocalNumSLocEntries > ID && "Chain corrupted");
|
|
|
|
|
|
|
|
F->SLocEntryCursor.JumpToBit(F->SLocOffsets[ID]);
|
|
|
|
return F->SLocEntryCursor;
|
|
|
|
}
|
|
|
|
|
2009-04-27 14:38:32 +08:00
|
|
|
/// \brief Read in the source location entry with the given ID.
|
2010-08-19 07:56:43 +08:00
|
|
|
ASTReader::ASTReadResult ASTReader::ReadSLocEntryRecord(unsigned ID) {
|
2009-04-27 14:38:32 +08:00
|
|
|
if (ID == 0)
|
|
|
|
return Success;
|
|
|
|
|
|
|
|
if (ID > TotalNumSLocEntries) {
|
2010-08-19 07:57:06 +08:00
|
|
|
Error("source location entry ID out-of-range for AST file");
|
2009-04-27 14:38:32 +08:00
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
|
2010-07-21 05:50:20 +08:00
|
|
|
llvm::BitstreamCursor &SLocEntryCursor = SLocCursorForID(ID);
|
2010-07-17 01:50:48 +08:00
|
|
|
|
2009-04-27 14:38:32 +08:00
|
|
|
++NumSLocEntriesRead;
|
|
|
|
unsigned Code = SLocEntryCursor.ReadCode();
|
|
|
|
if (Code == llvm::bitc::END_BLOCK ||
|
|
|
|
Code == llvm::bitc::ENTER_SUBBLOCK ||
|
|
|
|
Code == llvm::bitc::DEFINE_ABBREV) {
|
2010-08-19 07:57:06 +08:00
|
|
|
Error("incorrectly-formatted source location entry in AST file");
|
2009-04-27 14:38:32 +08:00
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
|
|
|
|
RecordData Record;
|
|
|
|
const char *BlobStart;
|
|
|
|
unsigned BlobLen;
|
|
|
|
switch (SLocEntryCursor.ReadRecord(Code, Record, &BlobStart, &BlobLen)) {
|
|
|
|
default:
|
2010-08-19 07:57:06 +08:00
|
|
|
Error("incorrectly-formatted source location entry in AST file");
|
2009-04-27 14:38:32 +08:00
|
|
|
return Failure;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case SM_SLOC_FILE_ENTRY: {
|
2009-07-07 08:12:59 +08:00
|
|
|
std::string Filename(BlobStart, BlobStart + BlobLen);
|
|
|
|
MaybeAddSystemRootToFilename(Filename);
|
|
|
|
const FileEntry *File = FileMgr.getFile(Filename);
|
2009-06-15 12:35:16 +08:00
|
|
|
if (File == 0) {
|
|
|
|
std::string ErrorStr = "could not find file '";
|
2009-07-07 08:12:59 +08:00
|
|
|
ErrorStr += Filename;
|
2010-08-19 07:57:06 +08:00
|
|
|
ErrorStr += "' referenced by AST file";
|
2009-06-15 12:35:16 +08:00
|
|
|
Error(ErrorStr.c_str());
|
|
|
|
return Failure;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-03-22 06:49:54 +08:00
|
|
|
if (Record.size() < 10) {
|
2010-03-19 05:23:05 +08:00
|
|
|
Error("source location entry is incorrect");
|
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
|
2010-07-27 08:27:13 +08:00
|
|
|
if (!DisableValidation &&
|
|
|
|
((off_t)Record[4] != File->getSize()
|
2010-04-09 23:54:22 +08:00
|
|
|
#if !defined(LLVM_ON_WIN32)
|
|
|
|
// In our regression testing, the Windows file system seems to
|
|
|
|
// have inconsistent modification times that sometimes
|
|
|
|
// erroneously trigger this error-handling path.
|
2010-07-27 08:27:13 +08:00
|
|
|
|| (time_t)Record[5] != File->getModificationTime()
|
2010-04-09 23:54:22 +08:00
|
|
|
#endif
|
2010-07-27 08:27:13 +08:00
|
|
|
)) {
|
2010-03-22 06:49:54 +08:00
|
|
|
Diag(diag::err_fe_pch_file_modified)
|
|
|
|
<< Filename;
|
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
|
2009-04-27 14:38:32 +08:00
|
|
|
FileID FID = SourceMgr.createFileID(File,
|
|
|
|
SourceLocation::getFromRawEncoding(Record[1]),
|
|
|
|
(SrcMgr::CharacteristicKind)Record[2],
|
|
|
|
ID, Record[0]);
|
|
|
|
if (Record[3])
|
|
|
|
const_cast<SrcMgr::FileInfo&>(SourceMgr.getSLocEntry(FID).getFile())
|
|
|
|
.setHasLineDirectives();
|
|
|
|
|
2010-03-17 00:35:32 +08:00
|
|
|
// Reconstruct header-search information for this file.
|
|
|
|
HeaderFileInfo HFI;
|
2010-03-22 06:49:54 +08:00
|
|
|
HFI.isImport = Record[6];
|
|
|
|
HFI.DirInfo = Record[7];
|
|
|
|
HFI.NumIncludes = Record[8];
|
|
|
|
HFI.ControllingMacroID = Record[9];
|
2010-03-17 00:35:32 +08:00
|
|
|
if (Listener)
|
|
|
|
Listener->ReadHeaderFileInfo(HFI, File->getUID());
|
2009-04-27 14:38:32 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case SM_SLOC_BUFFER_ENTRY: {
|
2009-04-27 14:38:32 +08:00
|
|
|
const char *Name = BlobStart;
|
|
|
|
unsigned Offset = Record[0];
|
|
|
|
unsigned Code = SLocEntryCursor.ReadCode();
|
|
|
|
Record.clear();
|
2009-09-09 23:08:12 +08:00
|
|
|
unsigned RecCode
|
2009-04-27 14:38:32 +08:00
|
|
|
= SLocEntryCursor.ReadRecord(Code, Record, &BlobStart, &BlobLen);
|
2010-03-18 08:56:54 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
if (RecCode != SM_SLOC_BUFFER_BLOB) {
|
2010-08-19 07:57:06 +08:00
|
|
|
Error("AST record has invalid code");
|
2010-03-18 08:56:54 +08:00
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
|
2009-04-27 14:38:32 +08:00
|
|
|
llvm::MemoryBuffer *Buffer
|
2010-04-06 06:42:27 +08:00
|
|
|
= llvm::MemoryBuffer::getMemBuffer(llvm::StringRef(BlobStart, BlobLen - 1),
|
|
|
|
Name);
|
2009-04-27 14:38:32 +08:00
|
|
|
FileID BufferID = SourceMgr.createFileIDForMemBuffer(Buffer, ID, Offset);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-29 04:33:11 +08:00
|
|
|
if (strcmp(Name, "<built-in>") == 0) {
|
2010-07-15 01:49:11 +08:00
|
|
|
PCHPredefinesBlock Block = {
|
|
|
|
BufferID,
|
|
|
|
llvm::StringRef(BlobStart, BlobLen - 1)
|
|
|
|
};
|
|
|
|
PCHPredefinesBuffers.push_back(Block);
|
2009-04-29 04:33:11 +08:00
|
|
|
}
|
2009-04-27 14:38:32 +08:00
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case SM_SLOC_INSTANTIATION_ENTRY: {
|
2009-09-09 23:08:12 +08:00
|
|
|
SourceLocation SpellingLoc
|
2009-04-27 14:38:32 +08:00
|
|
|
= SourceLocation::getFromRawEncoding(Record[1]);
|
|
|
|
SourceMgr.createInstantiationLoc(SpellingLoc,
|
|
|
|
SourceLocation::getFromRawEncoding(Record[2]),
|
|
|
|
SourceLocation::getFromRawEncoding(Record[3]),
|
|
|
|
Record[4],
|
|
|
|
ID,
|
|
|
|
Record[0]);
|
|
|
|
break;
|
2009-09-09 23:08:12 +08:00
|
|
|
}
|
2009-04-27 14:38:32 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return Success;
|
|
|
|
}
|
|
|
|
|
2009-04-27 09:05:14 +08:00
|
|
|
/// ReadBlockAbbrevs - Enter a subblock of the specified BlockID with the
|
|
|
|
/// specified cursor. Read the abbreviations that are at the top of the block
|
|
|
|
/// and then leave the cursor pointing into the block.
|
2010-08-19 07:56:43 +08:00
|
|
|
bool ASTReader::ReadBlockAbbrevs(llvm::BitstreamCursor &Cursor,
|
2009-04-27 09:05:14 +08:00
|
|
|
unsigned BlockID) {
|
|
|
|
if (Cursor.EnterSubBlock(BlockID)) {
|
2010-08-19 07:57:06 +08:00
|
|
|
Error("malformed block record in AST file");
|
2009-04-27 09:05:14 +08:00
|
|
|
return Failure;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-27 09:05:14 +08:00
|
|
|
while (true) {
|
|
|
|
unsigned Code = Cursor.ReadCode();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-27 09:05:14 +08:00
|
|
|
// We expect all abbrevs to be at the start of the block.
|
|
|
|
if (Code != llvm::bitc::DEFINE_ABBREV)
|
|
|
|
return false;
|
|
|
|
Cursor.ReadAbbrevRecord();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:43 +08:00
|
|
|
void ASTReader::ReadMacroRecord(llvm::BitstreamCursor &Stream, uint64_t Offset){
|
2009-06-19 08:03:23 +08:00
|
|
|
assert(PP && "Forgot to set Preprocessor ?");
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-22 07:56:24 +08:00
|
|
|
// Keep track of where we are in the stream, then jump back there
|
|
|
|
// after reading this macro.
|
|
|
|
SavedStreamPosition SavedPosition(Stream);
|
|
|
|
|
|
|
|
Stream.JumpToBit(Offset);
|
2009-04-11 05:41:48 +08:00
|
|
|
RecordData Record;
|
|
|
|
llvm::SmallVector<IdentifierInfo*, 16> MacroArgs;
|
2009-04-22 07:56:24 +08:00
|
|
|
MacroInfo *Macro = 0;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-11 05:41:48 +08:00
|
|
|
while (true) {
|
|
|
|
unsigned Code = Stream.ReadCode();
|
|
|
|
switch (Code) {
|
|
|
|
case llvm::bitc::END_BLOCK:
|
2009-04-22 07:56:24 +08:00
|
|
|
return;
|
|
|
|
|
2009-04-11 05:41:48 +08:00
|
|
|
case llvm::bitc::ENTER_SUBBLOCK:
|
|
|
|
// No known subblocks, always skip them.
|
|
|
|
Stream.ReadSubBlockID();
|
2009-04-22 07:56:24 +08:00
|
|
|
if (Stream.SkipBlock()) {
|
2010-08-19 07:57:06 +08:00
|
|
|
Error("malformed block record in AST file");
|
2009-04-22 07:56:24 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-04-11 05:41:48 +08:00
|
|
|
continue;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-11 05:41:48 +08:00
|
|
|
case llvm::bitc::DEFINE_ABBREV:
|
|
|
|
Stream.ReadAbbrevRecord();
|
|
|
|
continue;
|
|
|
|
default: break;
|
|
|
|
}
|
2009-04-22 07:56:24 +08:00
|
|
|
|
2009-04-11 05:41:48 +08:00
|
|
|
// Read a record.
|
|
|
|
Record.clear();
|
2010-08-19 07:57:32 +08:00
|
|
|
PreprocessorRecordTypes RecType =
|
|
|
|
(PreprocessorRecordTypes)Stream.ReadRecord(Code, Record);
|
2009-04-11 05:41:48 +08:00
|
|
|
switch (RecType) {
|
2010-08-19 07:57:32 +08:00
|
|
|
case PP_MACRO_OBJECT_LIKE:
|
|
|
|
case PP_MACRO_FUNCTION_LIKE: {
|
2009-04-22 07:56:24 +08:00
|
|
|
// If we already have a macro, that means that we've hit the end
|
|
|
|
// of the definition of the macro we were looking for. We're
|
|
|
|
// done.
|
|
|
|
if (Macro)
|
|
|
|
return;
|
|
|
|
|
2009-04-12 05:15:38 +08:00
|
|
|
IdentifierInfo *II = DecodeIdentifierInfo(Record[0]);
|
2009-04-22 07:56:24 +08:00
|
|
|
if (II == 0) {
|
2010-08-19 07:57:06 +08:00
|
|
|
Error("macro must have a name in AST file");
|
2009-04-22 07:56:24 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-04-11 05:41:48 +08:00
|
|
|
SourceLocation Loc = SourceLocation::getFromRawEncoding(Record[1]);
|
|
|
|
bool isUsed = Record[2];
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-06-19 08:03:23 +08:00
|
|
|
MacroInfo *MI = PP->AllocateMacroInfo(Loc);
|
2009-04-11 05:41:48 +08:00
|
|
|
MI->setIsUsed(isUsed);
|
2010-08-19 07:57:06 +08:00
|
|
|
MI->setIsFromAST();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
Implement serialization and lazy deserialization of the preprocessing
record (which includes all macro instantiations and definitions). As
with all lay deserialization, this introduces a new external source
(here, an external preprocessing record source) that loads all of the
preprocessed entities prior to iterating over the entities.
The preprocessing record is an optional part of the precompiled header
that is disabled by default (enabled with
-detailed-preprocessing-record). When the preprocessor given to the
PCH writer has a preprocessing record, that record is written into the
PCH file. When the PCH reader is given a PCH file that contains a
preprocessing record, it will be lazily loaded (which, effectively,
implicitly adds -detailed-preprocessing-record). This is the first
case where we have sections of the precompiled header that are
added/removed based on a compilation flag, which is
unfortunate. However, this data consumes ~550k in the PCH file for
Cocoa.h (out of ~9.9MB), and there is a non-trivial cost to gathering
this detailed preprocessing information, so it's too expensive to turn
on by default. In the future, we should investigate a better encoding
of this information.
llvm-svn: 99002
2010-03-20 05:51:54 +08:00
|
|
|
unsigned NextIndex = 3;
|
2010-08-19 07:57:32 +08:00
|
|
|
if (RecType == PP_MACRO_FUNCTION_LIKE) {
|
2009-04-11 05:41:48 +08:00
|
|
|
// Decode function-like macro info.
|
|
|
|
bool isC99VarArgs = Record[3];
|
|
|
|
bool isGNUVarArgs = Record[4];
|
|
|
|
MacroArgs.clear();
|
|
|
|
unsigned NumArgs = Record[5];
|
Implement serialization and lazy deserialization of the preprocessing
record (which includes all macro instantiations and definitions). As
with all lay deserialization, this introduces a new external source
(here, an external preprocessing record source) that loads all of the
preprocessed entities prior to iterating over the entities.
The preprocessing record is an optional part of the precompiled header
that is disabled by default (enabled with
-detailed-preprocessing-record). When the preprocessor given to the
PCH writer has a preprocessing record, that record is written into the
PCH file. When the PCH reader is given a PCH file that contains a
preprocessing record, it will be lazily loaded (which, effectively,
implicitly adds -detailed-preprocessing-record). This is the first
case where we have sections of the precompiled header that are
added/removed based on a compilation flag, which is
unfortunate. However, this data consumes ~550k in the PCH file for
Cocoa.h (out of ~9.9MB), and there is a non-trivial cost to gathering
this detailed preprocessing information, so it's too expensive to turn
on by default. In the future, we should investigate a better encoding
of this information.
llvm-svn: 99002
2010-03-20 05:51:54 +08:00
|
|
|
NextIndex = 6 + NumArgs;
|
2009-04-11 05:41:48 +08:00
|
|
|
for (unsigned i = 0; i != NumArgs; ++i)
|
2009-04-12 05:15:38 +08:00
|
|
|
MacroArgs.push_back(DecodeIdentifierInfo(Record[6+i]));
|
2009-04-11 05:41:48 +08:00
|
|
|
|
|
|
|
// Install function-like macro info.
|
|
|
|
MI->setIsFunctionLike();
|
|
|
|
if (isC99VarArgs) MI->setIsC99Varargs();
|
|
|
|
if (isGNUVarArgs) MI->setIsGNUVarargs();
|
2009-05-23 06:45:36 +08:00
|
|
|
MI->setArgumentList(MacroArgs.data(), MacroArgs.size(),
|
2009-06-19 08:03:23 +08:00
|
|
|
PP->getPreprocessorAllocator());
|
2009-04-11 05:41:48 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Finally, install the macro.
|
2009-06-19 08:03:23 +08:00
|
|
|
PP->setMacroInfo(II, MI);
|
2009-04-11 05:41:48 +08:00
|
|
|
|
|
|
|
// Remember that we saw this macro last so that we add the tokens that
|
|
|
|
// form its body to it.
|
2009-04-22 07:56:24 +08:00
|
|
|
Macro = MI;
|
Implement serialization and lazy deserialization of the preprocessing
record (which includes all macro instantiations and definitions). As
with all lay deserialization, this introduces a new external source
(here, an external preprocessing record source) that loads all of the
preprocessed entities prior to iterating over the entities.
The preprocessing record is an optional part of the precompiled header
that is disabled by default (enabled with
-detailed-preprocessing-record). When the preprocessor given to the
PCH writer has a preprocessing record, that record is written into the
PCH file. When the PCH reader is given a PCH file that contains a
preprocessing record, it will be lazily loaded (which, effectively,
implicitly adds -detailed-preprocessing-record). This is the first
case where we have sections of the precompiled header that are
added/removed based on a compilation flag, which is
unfortunate. However, this data consumes ~550k in the PCH file for
Cocoa.h (out of ~9.9MB), and there is a non-trivial cost to gathering
this detailed preprocessing information, so it's too expensive to turn
on by default. In the future, we should investigate a better encoding
of this information.
llvm-svn: 99002
2010-03-20 05:51:54 +08:00
|
|
|
|
|
|
|
if (NextIndex + 1 == Record.size() && PP->getPreprocessingRecord()) {
|
|
|
|
// We have a macro definition. Load it now.
|
|
|
|
PP->getPreprocessingRecord()->RegisterMacroDefinition(Macro,
|
|
|
|
getMacroDefinition(Record[NextIndex]));
|
|
|
|
}
|
|
|
|
|
2009-04-22 07:56:24 +08:00
|
|
|
++NumMacrosRead;
|
2009-04-11 05:41:48 +08:00
|
|
|
break;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case PP_TOKEN: {
|
2009-04-22 07:56:24 +08:00
|
|
|
// If we see a TOKEN before a PP_MACRO_*, then the file is
|
|
|
|
// erroneous, just pretend we didn't see this.
|
|
|
|
if (Macro == 0) break;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-11 05:41:48 +08:00
|
|
|
Token Tok;
|
|
|
|
Tok.startToken();
|
|
|
|
Tok.setLocation(SourceLocation::getFromRawEncoding(Record[0]));
|
|
|
|
Tok.setLength(Record[1]);
|
2009-04-12 05:15:38 +08:00
|
|
|
if (IdentifierInfo *II = DecodeIdentifierInfo(Record[2]))
|
|
|
|
Tok.setIdentifierInfo(II);
|
2009-04-11 05:41:48 +08:00
|
|
|
Tok.setKind((tok::TokenKind)Record[3]);
|
|
|
|
Tok.setFlag((Token::TokenFlags)Record[4]);
|
2009-04-22 07:56:24 +08:00
|
|
|
Macro->AddTokenToBody(Tok);
|
2009-04-11 05:41:48 +08:00
|
|
|
break;
|
|
|
|
}
|
Implement serialization and lazy deserialization of the preprocessing
record (which includes all macro instantiations and definitions). As
with all lay deserialization, this introduces a new external source
(here, an external preprocessing record source) that loads all of the
preprocessed entities prior to iterating over the entities.
The preprocessing record is an optional part of the precompiled header
that is disabled by default (enabled with
-detailed-preprocessing-record). When the preprocessor given to the
PCH writer has a preprocessing record, that record is written into the
PCH file. When the PCH reader is given a PCH file that contains a
preprocessing record, it will be lazily loaded (which, effectively,
implicitly adds -detailed-preprocessing-record). This is the first
case where we have sections of the precompiled header that are
added/removed based on a compilation flag, which is
unfortunate. However, this data consumes ~550k in the PCH file for
Cocoa.h (out of ~9.9MB), and there is a non-trivial cost to gathering
this detailed preprocessing information, so it's too expensive to turn
on by default. In the future, we should investigate a better encoding
of this information.
llvm-svn: 99002
2010-03-20 05:51:54 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case PP_MACRO_INSTANTIATION: {
|
Implement serialization and lazy deserialization of the preprocessing
record (which includes all macro instantiations and definitions). As
with all lay deserialization, this introduces a new external source
(here, an external preprocessing record source) that loads all of the
preprocessed entities prior to iterating over the entities.
The preprocessing record is an optional part of the precompiled header
that is disabled by default (enabled with
-detailed-preprocessing-record). When the preprocessor given to the
PCH writer has a preprocessing record, that record is written into the
PCH file. When the PCH reader is given a PCH file that contains a
preprocessing record, it will be lazily loaded (which, effectively,
implicitly adds -detailed-preprocessing-record). This is the first
case where we have sections of the precompiled header that are
added/removed based on a compilation flag, which is
unfortunate. However, this data consumes ~550k in the PCH file for
Cocoa.h (out of ~9.9MB), and there is a non-trivial cost to gathering
this detailed preprocessing information, so it's too expensive to turn
on by default. In the future, we should investigate a better encoding
of this information.
llvm-svn: 99002
2010-03-20 05:51:54 +08:00
|
|
|
// If we already have a macro, that means that we've hit the end
|
|
|
|
// of the definition of the macro we were looking for. We're
|
|
|
|
// done.
|
|
|
|
if (Macro)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (!PP->getPreprocessingRecord()) {
|
2010-08-19 07:57:06 +08:00
|
|
|
Error("missing preprocessing record in AST file");
|
Implement serialization and lazy deserialization of the preprocessing
record (which includes all macro instantiations and definitions). As
with all lay deserialization, this introduces a new external source
(here, an external preprocessing record source) that loads all of the
preprocessed entities prior to iterating over the entities.
The preprocessing record is an optional part of the precompiled header
that is disabled by default (enabled with
-detailed-preprocessing-record). When the preprocessor given to the
PCH writer has a preprocessing record, that record is written into the
PCH file. When the PCH reader is given a PCH file that contains a
preprocessing record, it will be lazily loaded (which, effectively,
implicitly adds -detailed-preprocessing-record). This is the first
case where we have sections of the precompiled header that are
added/removed based on a compilation flag, which is
unfortunate. However, this data consumes ~550k in the PCH file for
Cocoa.h (out of ~9.9MB), and there is a non-trivial cost to gathering
this detailed preprocessing information, so it's too expensive to turn
on by default. In the future, we should investigate a better encoding
of this information.
llvm-svn: 99002
2010-03-20 05:51:54 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
PreprocessingRecord &PPRec = *PP->getPreprocessingRecord();
|
|
|
|
if (PPRec.getPreprocessedEntity(Record[0]))
|
|
|
|
return;
|
|
|
|
|
|
|
|
MacroInstantiation *MI
|
|
|
|
= new (PPRec) MacroInstantiation(DecodeIdentifierInfo(Record[3]),
|
|
|
|
SourceRange(
|
|
|
|
SourceLocation::getFromRawEncoding(Record[1]),
|
|
|
|
SourceLocation::getFromRawEncoding(Record[2])),
|
|
|
|
getMacroDefinition(Record[4]));
|
|
|
|
PPRec.SetPreallocatedEntity(Record[0], MI);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case PP_MACRO_DEFINITION: {
|
Implement serialization and lazy deserialization of the preprocessing
record (which includes all macro instantiations and definitions). As
with all lay deserialization, this introduces a new external source
(here, an external preprocessing record source) that loads all of the
preprocessed entities prior to iterating over the entities.
The preprocessing record is an optional part of the precompiled header
that is disabled by default (enabled with
-detailed-preprocessing-record). When the preprocessor given to the
PCH writer has a preprocessing record, that record is written into the
PCH file. When the PCH reader is given a PCH file that contains a
preprocessing record, it will be lazily loaded (which, effectively,
implicitly adds -detailed-preprocessing-record). This is the first
case where we have sections of the precompiled header that are
added/removed based on a compilation flag, which is
unfortunate. However, this data consumes ~550k in the PCH file for
Cocoa.h (out of ~9.9MB), and there is a non-trivial cost to gathering
this detailed preprocessing information, so it's too expensive to turn
on by default. In the future, we should investigate a better encoding
of this information.
llvm-svn: 99002
2010-03-20 05:51:54 +08:00
|
|
|
// If we already have a macro, that means that we've hit the end
|
|
|
|
// of the definition of the macro we were looking for. We're
|
|
|
|
// done.
|
|
|
|
if (Macro)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (!PP->getPreprocessingRecord()) {
|
2010-08-19 07:57:06 +08:00
|
|
|
Error("missing preprocessing record in AST file");
|
Implement serialization and lazy deserialization of the preprocessing
record (which includes all macro instantiations and definitions). As
with all lay deserialization, this introduces a new external source
(here, an external preprocessing record source) that loads all of the
preprocessed entities prior to iterating over the entities.
The preprocessing record is an optional part of the precompiled header
that is disabled by default (enabled with
-detailed-preprocessing-record). When the preprocessor given to the
PCH writer has a preprocessing record, that record is written into the
PCH file. When the PCH reader is given a PCH file that contains a
preprocessing record, it will be lazily loaded (which, effectively,
implicitly adds -detailed-preprocessing-record). This is the first
case where we have sections of the precompiled header that are
added/removed based on a compilation flag, which is
unfortunate. However, this data consumes ~550k in the PCH file for
Cocoa.h (out of ~9.9MB), and there is a non-trivial cost to gathering
this detailed preprocessing information, so it's too expensive to turn
on by default. In the future, we should investigate a better encoding
of this information.
llvm-svn: 99002
2010-03-20 05:51:54 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
PreprocessingRecord &PPRec = *PP->getPreprocessingRecord();
|
|
|
|
if (PPRec.getPreprocessedEntity(Record[0]))
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (Record[1] >= MacroDefinitionsLoaded.size()) {
|
|
|
|
Error("out-of-bounds macro definition record");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
MacroDefinition *MD
|
|
|
|
= new (PPRec) MacroDefinition(DecodeIdentifierInfo(Record[4]),
|
|
|
|
SourceLocation::getFromRawEncoding(Record[5]),
|
|
|
|
SourceRange(
|
|
|
|
SourceLocation::getFromRawEncoding(Record[2]),
|
|
|
|
SourceLocation::getFromRawEncoding(Record[3])));
|
|
|
|
PPRec.SetPreallocatedEntity(Record[0], MD);
|
|
|
|
MacroDefinitionsLoaded[Record[1]] = MD;
|
|
|
|
return;
|
|
|
|
}
|
2010-09-28 06:18:47 +08:00
|
|
|
}
|
2009-04-11 05:41:48 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:43 +08:00
|
|
|
void ASTReader::ReadDefinedMacros() {
|
2010-07-22 06:31:37 +08:00
|
|
|
for (unsigned I = 0, N = Chain.size(); I != N; ++I) {
|
|
|
|
llvm::BitstreamCursor &MacroCursor = Chain[N - I - 1]->MacroCursor;
|
2010-03-08 03:10:13 +08:00
|
|
|
|
2010-07-22 06:31:37 +08:00
|
|
|
// If there was no preprocessor block, skip this file.
|
|
|
|
if (!MacroCursor.getBitStreamReader())
|
|
|
|
continue;
|
2010-03-08 03:10:13 +08:00
|
|
|
|
2010-07-22 06:31:37 +08:00
|
|
|
llvm::BitstreamCursor Cursor = MacroCursor;
|
2010-08-19 07:57:32 +08:00
|
|
|
if (Cursor.EnterSubBlock(PREPROCESSOR_BLOCK_ID)) {
|
2010-08-19 07:57:06 +08:00
|
|
|
Error("malformed preprocessor block record in AST file");
|
2010-01-05 03:18:44 +08:00
|
|
|
return;
|
|
|
|
}
|
2010-03-08 03:10:13 +08:00
|
|
|
|
2010-07-22 06:31:37 +08:00
|
|
|
RecordData Record;
|
|
|
|
while (true) {
|
|
|
|
unsigned Code = Cursor.ReadCode();
|
|
|
|
if (Code == llvm::bitc::END_BLOCK) {
|
|
|
|
if (Cursor.ReadBlockEnd()) {
|
2010-08-19 07:57:06 +08:00
|
|
|
Error("error at end of preprocessor block in AST file");
|
2010-07-22 06:31:37 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
break;
|
2010-01-05 03:18:44 +08:00
|
|
|
}
|
2010-03-08 03:10:13 +08:00
|
|
|
|
2010-07-22 06:31:37 +08:00
|
|
|
if (Code == llvm::bitc::ENTER_SUBBLOCK) {
|
|
|
|
// No known subblocks, always skip them.
|
|
|
|
Cursor.ReadSubBlockID();
|
|
|
|
if (Cursor.SkipBlock()) {
|
2010-08-19 07:57:06 +08:00
|
|
|
Error("malformed block record in AST file");
|
2010-07-22 06:31:37 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
2010-03-08 03:10:13 +08:00
|
|
|
|
2010-07-22 06:31:37 +08:00
|
|
|
if (Code == llvm::bitc::DEFINE_ABBREV) {
|
|
|
|
Cursor.ReadAbbrevRecord();
|
|
|
|
continue;
|
|
|
|
}
|
2010-03-08 03:10:13 +08:00
|
|
|
|
2010-07-22 06:31:37 +08:00
|
|
|
// Read a record.
|
|
|
|
const char *BlobStart;
|
|
|
|
unsigned BlobLen;
|
|
|
|
Record.clear();
|
|
|
|
switch (Cursor.ReadRecord(Code, Record, &BlobStart, &BlobLen)) {
|
|
|
|
default: // Default behavior: ignore.
|
|
|
|
break;
|
2010-01-05 03:18:44 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case PP_MACRO_OBJECT_LIKE:
|
|
|
|
case PP_MACRO_FUNCTION_LIKE:
|
2010-07-22 06:31:37 +08:00
|
|
|
DecodeIdentifierInfo(Record[0]);
|
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case PP_TOKEN:
|
2010-07-22 06:31:37 +08:00
|
|
|
// Ignore tokens.
|
|
|
|
break;
|
Implement serialization and lazy deserialization of the preprocessing
record (which includes all macro instantiations and definitions). As
with all lay deserialization, this introduces a new external source
(here, an external preprocessing record source) that loads all of the
preprocessed entities prior to iterating over the entities.
The preprocessing record is an optional part of the precompiled header
that is disabled by default (enabled with
-detailed-preprocessing-record). When the preprocessor given to the
PCH writer has a preprocessing record, that record is written into the
PCH file. When the PCH reader is given a PCH file that contains a
preprocessing record, it will be lazily loaded (which, effectively,
implicitly adds -detailed-preprocessing-record). This is the first
case where we have sections of the precompiled header that are
added/removed based on a compilation flag, which is
unfortunate. However, this data consumes ~550k in the PCH file for
Cocoa.h (out of ~9.9MB), and there is a non-trivial cost to gathering
this detailed preprocessing information, so it's too expensive to turn
on by default. In the future, we should investigate a better encoding
of this information.
llvm-svn: 99002
2010-03-20 05:51:54 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case PP_MACRO_INSTANTIATION:
|
|
|
|
case PP_MACRO_DEFINITION:
|
2010-07-22 06:31:37 +08:00
|
|
|
// Read the macro record.
|
|
|
|
ReadMacroRecord(Chain[N - I - 1]->Stream, Cursor.GetCurrentBitNo());
|
|
|
|
break;
|
|
|
|
}
|
2010-01-05 03:18:44 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-09-16 03:54:06 +08:00
|
|
|
MacroDefinition *ASTReader::getMacroDefinition(MacroID ID) {
|
Implement serialization and lazy deserialization of the preprocessing
record (which includes all macro instantiations and definitions). As
with all lay deserialization, this introduces a new external source
(here, an external preprocessing record source) that loads all of the
preprocessed entities prior to iterating over the entities.
The preprocessing record is an optional part of the precompiled header
that is disabled by default (enabled with
-detailed-preprocessing-record). When the preprocessor given to the
PCH writer has a preprocessing record, that record is written into the
PCH file. When the PCH reader is given a PCH file that contains a
preprocessing record, it will be lazily loaded (which, effectively,
implicitly adds -detailed-preprocessing-record). This is the first
case where we have sections of the precompiled header that are
added/removed based on a compilation flag, which is
unfortunate. However, this data consumes ~550k in the PCH file for
Cocoa.h (out of ~9.9MB), and there is a non-trivial cost to gathering
this detailed preprocessing information, so it's too expensive to turn
on by default. In the future, we should investigate a better encoding
of this information.
llvm-svn: 99002
2010-03-20 05:51:54 +08:00
|
|
|
if (ID == 0 || ID >= MacroDefinitionsLoaded.size())
|
|
|
|
return 0;
|
2010-07-22 06:31:37 +08:00
|
|
|
|
|
|
|
if (!MacroDefinitionsLoaded[ID]) {
|
|
|
|
unsigned Index = ID;
|
|
|
|
for (unsigned I = 0, N = Chain.size(); I != N; ++I) {
|
|
|
|
PerFileData &F = *Chain[N - I - 1];
|
|
|
|
if (Index < F.LocalNumMacroDefinitions) {
|
|
|
|
ReadMacroRecord(F.Stream, F.MacroDefinitionOffsets[Index]);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
Index -= F.LocalNumMacroDefinitions;
|
|
|
|
}
|
|
|
|
assert(MacroDefinitionsLoaded[ID] && "Broken chain");
|
|
|
|
}
|
|
|
|
|
Implement serialization and lazy deserialization of the preprocessing
record (which includes all macro instantiations and definitions). As
with all lay deserialization, this introduces a new external source
(here, an external preprocessing record source) that loads all of the
preprocessed entities prior to iterating over the entities.
The preprocessing record is an optional part of the precompiled header
that is disabled by default (enabled with
-detailed-preprocessing-record). When the preprocessor given to the
PCH writer has a preprocessing record, that record is written into the
PCH file. When the PCH reader is given a PCH file that contains a
preprocessing record, it will be lazily loaded (which, effectively,
implicitly adds -detailed-preprocessing-record). This is the first
case where we have sections of the precompiled header that are
added/removed based on a compilation flag, which is
unfortunate. However, this data consumes ~550k in the PCH file for
Cocoa.h (out of ~9.9MB), and there is a non-trivial cost to gathering
this detailed preprocessing information, so it's too expensive to turn
on by default. In the future, we should investigate a better encoding
of this information.
llvm-svn: 99002
2010-03-20 05:51:54 +08:00
|
|
|
return MacroDefinitionsLoaded[ID];
|
|
|
|
}
|
|
|
|
|
2009-07-07 08:12:59 +08:00
|
|
|
/// \brief If we are loading a relocatable PCH file, and the filename is
|
|
|
|
/// not an absolute path, add the system root to the beginning of the file
|
|
|
|
/// name.
|
2010-08-19 07:56:43 +08:00
|
|
|
void ASTReader::MaybeAddSystemRootToFilename(std::string &Filename) {
|
2009-07-07 08:12:59 +08:00
|
|
|
// If this is not a relocatable PCH file, there's nothing to do.
|
|
|
|
if (!RelocatablePCH)
|
|
|
|
return;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-11-19 03:50:41 +08:00
|
|
|
if (Filename.empty() || llvm::sys::Path(Filename).isAbsolute())
|
2009-07-07 08:12:59 +08:00
|
|
|
return;
|
|
|
|
|
|
|
|
if (isysroot == 0) {
|
|
|
|
// If no system root was given, default to '/'
|
|
|
|
Filename.insert(Filename.begin(), '/');
|
|
|
|
return;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-07-07 08:12:59 +08:00
|
|
|
unsigned Length = strlen(isysroot);
|
|
|
|
if (isysroot[Length - 1] != '/')
|
|
|
|
Filename.insert(Filename.begin(), '/');
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-07-07 08:12:59 +08:00
|
|
|
Filename.insert(Filename.begin(), isysroot, isysroot + Length);
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:43 +08:00
|
|
|
ASTReader::ASTReadResult
|
2010-08-19 07:56:56 +08:00
|
|
|
ASTReader::ReadASTBlock(PerFileData &F) {
|
2010-07-17 01:50:48 +08:00
|
|
|
llvm::BitstreamCursor &Stream = F.Stream;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
if (Stream.EnterSubBlock(AST_BLOCK_ID)) {
|
2010-08-19 07:57:06 +08:00
|
|
|
Error("malformed block record in AST file");
|
2009-04-11 04:39:37 +08:00
|
|
|
return Failure;
|
|
|
|
}
|
2009-04-10 06:27:44 +08:00
|
|
|
|
2010-08-19 07:57:06 +08:00
|
|
|
// Read all of the records and blocks for the ASt file.
|
2009-04-10 06:27:44 +08:00
|
|
|
RecordData Record;
|
2010-07-20 04:52:06 +08:00
|
|
|
bool First = true;
|
2009-04-11 01:25:41 +08:00
|
|
|
while (!Stream.AtEndOfStream()) {
|
2009-04-10 06:27:44 +08:00
|
|
|
unsigned Code = Stream.ReadCode();
|
|
|
|
if (Code == llvm::bitc::END_BLOCK) {
|
2009-04-11 04:39:37 +08:00
|
|
|
if (Stream.ReadBlockEnd()) {
|
2010-08-19 07:57:06 +08:00
|
|
|
Error("error at end of module block in AST file");
|
2009-04-11 04:39:37 +08:00
|
|
|
return Failure;
|
|
|
|
}
|
2009-04-12 05:15:38 +08:00
|
|
|
|
2009-04-11 04:39:37 +08:00
|
|
|
return Success;
|
2009-04-10 06:27:44 +08:00
|
|
|
}
|
2009-04-11 01:25:41 +08:00
|
|
|
|
2009-04-10 06:27:44 +08:00
|
|
|
if (Code == llvm::bitc::ENTER_SUBBLOCK) {
|
2009-04-11 01:25:41 +08:00
|
|
|
switch (Stream.ReadSubBlockID()) {
|
2010-08-19 07:57:32 +08:00
|
|
|
case DECLTYPES_BLOCK_ID:
|
2009-04-27 09:05:14 +08:00
|
|
|
// We lazily load the decls block, but we want to set up the
|
|
|
|
// DeclsCursor cursor to point into it. Clone our current bitcode
|
|
|
|
// cursor to it, enter the block and read the abbrevs in that block.
|
|
|
|
// With the main cursor, we just skip over it.
|
2010-07-17 01:50:48 +08:00
|
|
|
F.DeclsCursor = Stream;
|
2009-04-27 09:05:14 +08:00
|
|
|
if (Stream.SkipBlock() || // Skip with the main cursor.
|
|
|
|
// Read the abbrevs.
|
2010-08-19 07:57:32 +08:00
|
|
|
ReadBlockAbbrevs(F.DeclsCursor, DECLTYPES_BLOCK_ID)) {
|
2010-08-19 07:57:06 +08:00
|
|
|
Error("malformed block record in AST file");
|
2009-04-27 09:05:14 +08:00
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
break;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case PREPROCESSOR_BLOCK_ID:
|
2010-07-17 01:50:48 +08:00
|
|
|
F.MacroCursor = Stream;
|
2010-01-05 03:18:44 +08:00
|
|
|
if (PP)
|
|
|
|
PP->setExternalSource(this);
|
|
|
|
|
2009-04-12 05:15:38 +08:00
|
|
|
if (Stream.SkipBlock()) {
|
2010-08-19 07:57:06 +08:00
|
|
|
Error("malformed block record in AST file");
|
2009-04-12 05:15:38 +08:00
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
break;
|
2009-04-23 18:39:46 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case SOURCE_MANAGER_BLOCK_ID:
|
2010-07-20 04:52:06 +08:00
|
|
|
switch (ReadSourceManagerBlock(F)) {
|
2009-04-11 07:10:45 +08:00
|
|
|
case Success:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Failure:
|
2010-08-19 07:57:06 +08:00
|
|
|
Error("malformed source manager block in AST file");
|
2009-04-11 04:39:37 +08:00
|
|
|
return Failure;
|
2009-04-11 07:10:45 +08:00
|
|
|
|
|
|
|
case IgnorePCH:
|
|
|
|
return IgnorePCH;
|
2009-04-11 04:39:37 +08:00
|
|
|
}
|
2009-04-11 01:25:41 +08:00
|
|
|
break;
|
|
|
|
}
|
2010-07-20 04:52:06 +08:00
|
|
|
First = false;
|
2009-04-10 06:27:44 +08:00
|
|
|
continue;
|
|
|
|
}
|
2009-04-11 01:25:41 +08:00
|
|
|
|
2009-04-10 06:27:44 +08:00
|
|
|
if (Code == llvm::bitc::DEFINE_ABBREV) {
|
|
|
|
Stream.ReadAbbrevRecord();
|
|
|
|
continue;
|
|
|
|
}
|
2009-04-11 01:25:41 +08:00
|
|
|
|
|
|
|
// Read and process a record.
|
2009-04-10 06:27:44 +08:00
|
|
|
Record.clear();
|
2009-04-11 05:16:55 +08:00
|
|
|
const char *BlobStart = 0;
|
|
|
|
unsigned BlobLen = 0;
|
2010-08-19 07:57:32 +08:00
|
|
|
switch ((ASTRecordTypes)Stream.ReadRecord(Code, Record,
|
2009-04-11 05:16:55 +08:00
|
|
|
&BlobStart, &BlobLen)) {
|
2009-04-10 06:27:44 +08:00
|
|
|
default: // Default behavior: ignore.
|
|
|
|
break;
|
2009-04-11 01:25:41 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case METADATA: {
|
|
|
|
if (Record[0] != VERSION_MAJOR && !DisableValidation) {
|
|
|
|
Diag(Record[0] < VERSION_MAJOR? diag::warn_pch_version_too_old
|
2010-07-17 08:12:06 +08:00
|
|
|
: diag::warn_pch_version_too_new);
|
|
|
|
return IgnorePCH;
|
|
|
|
}
|
|
|
|
|
|
|
|
RelocatablePCH = Record[4];
|
|
|
|
if (Listener) {
|
|
|
|
std::string TargetTriple(BlobStart, BlobLen);
|
|
|
|
if (Listener->ReadTargetTriple(TargetTriple))
|
|
|
|
return IgnorePCH;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case CHAINED_METADATA: {
|
2010-07-20 04:52:06 +08:00
|
|
|
if (!First) {
|
|
|
|
Error("CHAINED_METADATA is not first record in block");
|
|
|
|
return Failure;
|
|
|
|
}
|
2010-08-19 07:57:32 +08:00
|
|
|
if (Record[0] != VERSION_MAJOR && !DisableValidation) {
|
|
|
|
Diag(Record[0] < VERSION_MAJOR? diag::warn_pch_version_too_old
|
2010-07-17 08:12:06 +08:00
|
|
|
: diag::warn_pch_version_too_new);
|
|
|
|
return IgnorePCH;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Load the chained file.
|
2010-08-19 07:56:56 +08:00
|
|
|
switch(ReadASTCore(llvm::StringRef(BlobStart, BlobLen))) {
|
2010-07-17 08:12:06 +08:00
|
|
|
case Failure: return Failure;
|
|
|
|
// If we have to ignore the dependency, we'll have to ignore this too.
|
|
|
|
case IgnorePCH: return IgnorePCH;
|
|
|
|
case Success: break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TYPE_OFFSET:
|
2010-07-20 06:06:55 +08:00
|
|
|
if (F.LocalNumTypes != 0) {
|
2010-08-19 07:57:06 +08:00
|
|
|
Error("duplicate TYPE_OFFSET record in AST file");
|
2009-04-11 04:39:37 +08:00
|
|
|
return Failure;
|
|
|
|
}
|
2010-07-20 06:06:55 +08:00
|
|
|
F.TypeOffsets = (const uint32_t *)BlobStart;
|
|
|
|
F.LocalNumTypes = Record[0];
|
2009-04-10 06:27:44 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case DECL_OFFSET:
|
2010-07-20 06:06:55 +08:00
|
|
|
if (F.LocalNumDecls != 0) {
|
2010-08-19 07:57:06 +08:00
|
|
|
Error("duplicate DECL_OFFSET record in AST file");
|
2009-04-11 04:39:37 +08:00
|
|
|
return Failure;
|
|
|
|
}
|
2010-07-20 06:06:55 +08:00
|
|
|
F.DeclOffsets = (const uint32_t *)BlobStart;
|
|
|
|
F.LocalNumDecls = Record[0];
|
2009-04-10 06:27:44 +08:00
|
|
|
break;
|
2009-04-11 04:39:37 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TU_UPDATE_LEXICAL: {
|
2010-07-28 02:24:41 +08:00
|
|
|
DeclContextInfo Info = {
|
2010-08-21 00:04:35 +08:00
|
|
|
/* No visible information */ 0,
|
2010-08-19 07:57:32 +08:00
|
|
|
reinterpret_cast<const DeclID *>(BlobStart),
|
|
|
|
BlobLen / sizeof(DeclID)
|
2010-07-28 02:24:41 +08:00
|
|
|
};
|
|
|
|
DeclContextOffsets[Context->getTranslationUnitDecl()].push_back(Info);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2010-08-24 08:50:04 +08:00
|
|
|
case UPDATE_VISIBLE: {
|
|
|
|
serialization::DeclID ID = Record[0];
|
|
|
|
void *Table = ASTDeclContextNameLookupTable::Create(
|
|
|
|
(const unsigned char *)BlobStart + Record[1],
|
|
|
|
(const unsigned char *)BlobStart,
|
|
|
|
ASTDeclContextNameLookupTrait(*this));
|
|
|
|
if (ID == 1) { // Is it the TU?
|
|
|
|
DeclContextInfo Info = {
|
|
|
|
Table, /* No lexical inforamtion */ 0, 0
|
|
|
|
};
|
|
|
|
DeclContextOffsets[Context->getTranslationUnitDecl()].push_back(Info);
|
|
|
|
} else
|
|
|
|
PendingVisibleUpdates[ID].push_back(Table);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case REDECLS_UPDATE_LATEST: {
|
2010-08-04 01:30:10 +08:00
|
|
|
assert(Record.size() % 2 == 0 && "Expected pairs of DeclIDs");
|
|
|
|
for (unsigned i = 0, e = Record.size(); i < e; i += 2) {
|
2010-08-19 07:57:32 +08:00
|
|
|
DeclID First = Record[i], Latest = Record[i+1];
|
2010-08-04 01:30:10 +08:00
|
|
|
assert((FirstLatestDeclIDs.find(First) == FirstLatestDeclIDs.end() ||
|
|
|
|
Latest > FirstLatestDeclIDs[First]) &&
|
|
|
|
"The new latest is supposed to come after the previous latest");
|
|
|
|
FirstLatestDeclIDs[First] = Latest;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case LANGUAGE_OPTIONS:
|
2010-07-27 08:27:13 +08:00
|
|
|
if (ParseLanguageOptions(Record) && !DisableValidation)
|
2009-04-11 04:39:37 +08:00
|
|
|
return IgnorePCH;
|
|
|
|
break;
|
2009-04-11 05:16:55 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case IDENTIFIER_TABLE:
|
2010-07-20 04:52:06 +08:00
|
|
|
F.IdentifierTableData = BlobStart;
|
2009-04-26 03:10:14 +08:00
|
|
|
if (Record[0]) {
|
2010-07-20 04:52:06 +08:00
|
|
|
F.IdentifierLookupTable
|
2010-08-19 07:57:06 +08:00
|
|
|
= ASTIdentifierLookupTable::Create(
|
2010-07-20 04:52:06 +08:00
|
|
|
(const unsigned char *)F.IdentifierTableData + Record[0],
|
|
|
|
(const unsigned char *)F.IdentifierTableData,
|
2010-08-19 07:57:06 +08:00
|
|
|
ASTIdentifierLookupTrait(*this, F.Stream));
|
2009-06-19 08:03:23 +08:00
|
|
|
if (PP)
|
|
|
|
PP->getIdentifierTable().setExternalIdentifierLookup(this);
|
2009-04-26 03:10:14 +08:00
|
|
|
}
|
2009-04-11 08:14:32 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case IDENTIFIER_OFFSET:
|
2010-07-20 06:28:42 +08:00
|
|
|
if (F.LocalNumIdentifiers != 0) {
|
2010-08-19 07:57:06 +08:00
|
|
|
Error("duplicate IDENTIFIER_OFFSET record in AST file");
|
2009-04-11 08:14:32 +08:00
|
|
|
return Failure;
|
|
|
|
}
|
2010-07-20 06:28:42 +08:00
|
|
|
F.IdentifierOffsets = (const uint32_t *)BlobStart;
|
|
|
|
F.LocalNumIdentifiers = Record[0];
|
2009-04-11 08:14:32 +08:00
|
|
|
break;
|
2009-04-14 08:24:19 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXTERNAL_DEFINITIONS:
|
2010-07-21 05:20:32 +08:00
|
|
|
// Optimization for the first block.
|
|
|
|
if (ExternalDefinitions.empty())
|
|
|
|
ExternalDefinitions.swap(Record);
|
|
|
|
else
|
|
|
|
ExternalDefinitions.insert(ExternalDefinitions.end(),
|
|
|
|
Record.begin(), Record.end());
|
2009-04-14 08:24:19 +08:00
|
|
|
break;
|
2009-04-18 06:13:46 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case SPECIAL_TYPES:
|
2010-07-21 05:20:32 +08:00
|
|
|
// Optimization for the first block
|
|
|
|
if (SpecialTypes.empty())
|
|
|
|
SpecialTypes.swap(Record);
|
|
|
|
else
|
|
|
|
SpecialTypes.insert(SpecialTypes.end(), Record.begin(), Record.end());
|
2009-04-18 13:55:16 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case STATISTICS:
|
2010-07-21 05:20:32 +08:00
|
|
|
TotalNumStatements += Record[0];
|
|
|
|
TotalNumMacros += Record[1];
|
|
|
|
TotalLexicalDeclContexts += Record[2];
|
|
|
|
TotalVisibleDeclContexts += Record[3];
|
2009-04-18 06:13:46 +08:00
|
|
|
break;
|
2009-04-27 14:38:32 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TENTATIVE_DEFINITIONS:
|
2010-07-21 05:20:32 +08:00
|
|
|
// Optimization for the first block.
|
|
|
|
if (TentativeDefinitions.empty())
|
|
|
|
TentativeDefinitions.swap(Record);
|
|
|
|
else
|
|
|
|
TentativeDefinitions.insert(TentativeDefinitions.end(),
|
|
|
|
Record.begin(), Record.end());
|
2009-04-23 06:02:47 +08:00
|
|
|
break;
|
2009-04-23 06:18:58 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case UNUSED_FILESCOPED_DECLS:
|
2010-07-21 05:20:32 +08:00
|
|
|
// Optimization for the first block.
|
2010-08-14 02:42:17 +08:00
|
|
|
if (UnusedFileScopedDecls.empty())
|
|
|
|
UnusedFileScopedDecls.swap(Record);
|
2010-07-21 05:20:32 +08:00
|
|
|
else
|
2010-08-14 02:42:17 +08:00
|
|
|
UnusedFileScopedDecls.insert(UnusedFileScopedDecls.end(),
|
|
|
|
Record.begin(), Record.end());
|
2010-02-12 08:07:30 +08:00
|
|
|
break;
|
2010-03-08 03:10:13 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case WEAK_UNDECLARED_IDENTIFIERS:
|
2010-08-06 02:21:25 +08:00
|
|
|
// Later blocks overwrite earlier ones.
|
|
|
|
WeakUndeclaredIdentifiers.swap(Record);
|
2010-08-05 17:48:08 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case LOCALLY_SCOPED_EXTERNAL_DECLS:
|
2010-07-21 05:20:32 +08:00
|
|
|
// Optimization for the first block.
|
|
|
|
if (LocallyScopedExternalDecls.empty())
|
|
|
|
LocallyScopedExternalDecls.swap(Record);
|
|
|
|
else
|
|
|
|
LocallyScopedExternalDecls.insert(LocallyScopedExternalDecls.end(),
|
|
|
|
Record.begin(), Record.end());
|
2009-04-23 06:18:58 +08:00
|
|
|
break;
|
2009-04-25 05:10:55 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case SELECTOR_OFFSETS:
|
2010-08-04 05:58:15 +08:00
|
|
|
F.SelectorOffsets = (const uint32_t *)BlobStart;
|
2010-08-05 04:40:17 +08:00
|
|
|
F.LocalNumSelectors = Record[0];
|
2009-04-26 01:48:32 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case METHOD_POOL:
|
2010-08-05 04:40:17 +08:00
|
|
|
F.SelectorLookupTableData = (const unsigned char *)BlobStart;
|
2009-04-26 01:48:32 +08:00
|
|
|
if (Record[0])
|
2010-08-05 04:40:17 +08:00
|
|
|
F.SelectorLookupTable
|
2010-08-19 07:57:06 +08:00
|
|
|
= ASTSelectorLookupTable::Create(
|
2010-08-05 04:40:17 +08:00
|
|
|
F.SelectorLookupTableData + Record[0],
|
|
|
|
F.SelectorLookupTableData,
|
2010-08-19 07:57:06 +08:00
|
|
|
ASTSelectorLookupTrait(*this));
|
2010-08-05 05:22:45 +08:00
|
|
|
TotalNumMethodPoolEntries += Record[1];
|
2009-04-25 05:10:55 +08:00
|
|
|
break;
|
2009-04-26 08:07:37 +08:00
|
|
|
|
2010-09-22 08:42:30 +08:00
|
|
|
case REFERENCED_SELECTOR_POOL:
|
|
|
|
if (ReferencedSelectorsData.empty())
|
|
|
|
ReferencedSelectorsData.swap(Record);
|
|
|
|
else
|
|
|
|
ReferencedSelectorsData.insert(ReferencedSelectorsData.end(),
|
|
|
|
Record.begin(), Record.end());
|
2010-07-24 03:11:11 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case PP_COUNTER_VALUE:
|
2009-06-19 08:03:23 +08:00
|
|
|
if (!Record.empty() && Listener)
|
|
|
|
Listener->ReadCounter(Record[0]);
|
2009-04-26 08:07:37 +08:00
|
|
|
break;
|
2009-04-27 14:38:32 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case SOURCE_LOCATION_OFFSETS:
|
2010-07-21 05:20:32 +08:00
|
|
|
F.SLocOffsets = (const uint32_t *)BlobStart;
|
|
|
|
F.LocalNumSLocEntries = Record[0];
|
2010-09-23 04:19:08 +08:00
|
|
|
F.LocalSLocSize = Record[1];
|
2009-04-27 14:38:32 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case SOURCE_LOCATION_PRELOADS:
|
2010-09-22 08:42:30 +08:00
|
|
|
if (PreloadSLocEntries.empty())
|
|
|
|
PreloadSLocEntries.swap(Record);
|
|
|
|
else
|
|
|
|
PreloadSLocEntries.insert(PreloadSLocEntries.end(),
|
|
|
|
Record.begin(), Record.end());
|
2009-04-27 14:38:32 +08:00
|
|
|
break;
|
2009-04-28 02:38:38 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case STAT_CACHE: {
|
2010-08-19 07:57:06 +08:00
|
|
|
ASTStatCache *MyStatCache =
|
|
|
|
new ASTStatCache((const unsigned char *)BlobStart + Record[0],
|
2009-10-17 02:18:30 +08:00
|
|
|
(const unsigned char *)BlobStart,
|
|
|
|
NumStatHits, NumStatMisses);
|
|
|
|
FileMgr.addStatCache(MyStatCache);
|
2010-07-17 01:50:48 +08:00
|
|
|
F.StatCache = MyStatCache;
|
2009-04-28 02:38:38 +08:00
|
|
|
break;
|
2009-10-17 02:18:30 +08:00
|
|
|
}
|
2010-03-08 03:10:13 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXT_VECTOR_DECLS:
|
2010-07-29 05:38:49 +08:00
|
|
|
// Optimization for the first block.
|
|
|
|
if (ExtVectorDecls.empty())
|
|
|
|
ExtVectorDecls.swap(Record);
|
|
|
|
else
|
|
|
|
ExtVectorDecls.insert(ExtVectorDecls.end(),
|
|
|
|
Record.begin(), Record.end());
|
2009-04-28 04:06:05 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case VTABLE_USES:
|
2010-08-06 02:21:25 +08:00
|
|
|
// Later tables overwrite earlier ones.
|
2010-07-06 23:37:04 +08:00
|
|
|
VTableUses.swap(Record);
|
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case DYNAMIC_CLASSES:
|
2010-08-06 02:21:25 +08:00
|
|
|
// Optimization for the first block.
|
|
|
|
if (DynamicClasses.empty())
|
|
|
|
DynamicClasses.swap(Record);
|
|
|
|
else
|
|
|
|
DynamicClasses.insert(DynamicClasses.end(),
|
|
|
|
Record.begin(), Record.end());
|
2010-07-06 23:37:04 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case PENDING_IMPLICIT_INSTANTIATIONS:
|
2010-08-05 17:48:16 +08:00
|
|
|
// Optimization for the first block.
|
2010-08-25 16:44:16 +08:00
|
|
|
if (PendingInstantiations.empty())
|
|
|
|
PendingInstantiations.swap(Record);
|
2010-08-05 17:48:16 +08:00
|
|
|
else
|
2010-08-25 16:44:16 +08:00
|
|
|
PendingInstantiations.insert(PendingInstantiations.end(),
|
|
|
|
Record.begin(), Record.end());
|
2010-08-05 17:48:16 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case SEMA_DECL_REFS:
|
2010-08-06 02:21:25 +08:00
|
|
|
// Later tables overwrite earlier ones.
|
2010-08-02 15:14:54 +08:00
|
|
|
SemaDeclRefs.swap(Record);
|
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case ORIGINAL_FILE_NAME:
|
2010-08-19 07:57:06 +08:00
|
|
|
// The primary AST will be the last to get here, so it will be the one
|
2010-07-21 05:20:32 +08:00
|
|
|
// that's used.
|
2009-11-11 13:29:04 +08:00
|
|
|
ActualOriginalFileName.assign(BlobStart, BlobLen);
|
|
|
|
OriginalFileName = ActualOriginalFileName;
|
2009-07-07 08:12:59 +08:00
|
|
|
MaybeAddSystemRootToFilename(OriginalFileName);
|
2009-05-12 09:31:05 +08:00
|
|
|
break;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case VERSION_CONTROL_BRANCH_REVISION: {
|
2010-02-13 07:31:14 +08:00
|
|
|
const std::string &CurBranch = getClangFullRepositoryVersion();
|
2010-08-19 07:57:06 +08:00
|
|
|
llvm::StringRef ASTBranch(BlobStart, BlobLen);
|
|
|
|
if (llvm::StringRef(CurBranch) != ASTBranch && !DisableValidation) {
|
|
|
|
Diag(diag::warn_pch_different_branch) << ASTBranch << CurBranch;
|
2009-10-06 05:07:28 +08:00
|
|
|
return IgnorePCH;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2010-07-22 04:07:32 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case MACRO_DEFINITION_OFFSETS:
|
2010-07-22 04:07:32 +08:00
|
|
|
F.MacroDefinitionOffsets = (const uint32_t *)BlobStart;
|
|
|
|
F.NumPreallocatedPreprocessingEntities = Record[0];
|
|
|
|
F.LocalNumMacroDefinitions = Record[1];
|
Implement serialization and lazy deserialization of the preprocessing
record (which includes all macro instantiations and definitions). As
with all lay deserialization, this introduces a new external source
(here, an external preprocessing record source) that loads all of the
preprocessed entities prior to iterating over the entities.
The preprocessing record is an optional part of the precompiled header
that is disabled by default (enabled with
-detailed-preprocessing-record). When the preprocessor given to the
PCH writer has a preprocessing record, that record is written into the
PCH file. When the PCH reader is given a PCH file that contains a
preprocessing record, it will be lazily loaded (which, effectively,
implicitly adds -detailed-preprocessing-record). This is the first
case where we have sections of the precompiled header that are
added/removed based on a compilation flag, which is
unfortunate. However, this data consumes ~550k in the PCH file for
Cocoa.h (out of ~9.9MB), and there is a non-trivial cost to gathering
this detailed preprocessing information, so it's too expensive to turn
on by default. In the future, we should investigate a better encoding
of this information.
llvm-svn: 99002
2010-03-20 05:51:54 +08:00
|
|
|
break;
|
2010-08-13 08:28:03 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case DECL_REPLACEMENTS: {
|
2010-08-13 08:28:03 +08:00
|
|
|
if (Record.size() % 2 != 0) {
|
2010-08-19 07:57:06 +08:00
|
|
|
Error("invalid DECL_REPLACEMENTS block in AST file");
|
2010-08-13 08:28:03 +08:00
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
for (unsigned I = 0, N = Record.size(); I != N; I += 2)
|
2010-08-19 07:57:32 +08:00
|
|
|
ReplacedDecls[static_cast<DeclID>(Record[I])] =
|
2010-08-13 08:28:03 +08:00
|
|
|
std::make_pair(&F, Record[I+1]);
|
|
|
|
break;
|
|
|
|
}
|
2010-08-25 06:50:19 +08:00
|
|
|
|
|
|
|
case ADDITIONAL_TEMPLATE_SPECIALIZATIONS: {
|
|
|
|
AdditionalTemplateSpecializations &ATS =
|
|
|
|
AdditionalTemplateSpecializationsPending[Record[0]];
|
|
|
|
ATS.insert(ATS.end(), Record.begin()+1, Record.end());
|
|
|
|
break;
|
|
|
|
}
|
2009-04-11 08:14:32 +08:00
|
|
|
}
|
2010-07-20 04:52:06 +08:00
|
|
|
First = false;
|
2009-04-10 06:27:44 +08:00
|
|
|
}
|
2010-08-19 07:57:06 +08:00
|
|
|
Error("premature end of bitstream in AST file");
|
2009-04-11 04:39:37 +08:00
|
|
|
return Failure;
|
2009-04-10 06:27:44 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:56 +08:00
|
|
|
ASTReader::ASTReadResult ASTReader::ReadAST(const std::string &FileName) {
|
|
|
|
switch(ReadASTCore(FileName)) {
|
2010-07-17 04:41:52 +08:00
|
|
|
case Failure: return Failure;
|
|
|
|
case IgnorePCH: return IgnorePCH;
|
|
|
|
case Success: break;
|
2009-04-11 07:10:45 +08:00
|
|
|
}
|
2009-04-10 06:27:44 +08:00
|
|
|
|
2010-07-17 08:12:06 +08:00
|
|
|
// Here comes stuff that we only do once the entire chain is loaded.
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-09-22 08:42:30 +08:00
|
|
|
// Allocate space for loaded slocentries, identifiers, decls and types.
|
2010-07-22 04:07:32 +08:00
|
|
|
unsigned TotalNumIdentifiers = 0, TotalNumTypes = 0, TotalNumDecls = 0,
|
2010-08-05 04:40:17 +08:00
|
|
|
TotalNumPreallocatedPreprocessingEntities = 0, TotalNumMacroDefs = 0,
|
|
|
|
TotalNumSelectors = 0;
|
2010-07-20 06:06:55 +08:00
|
|
|
for (unsigned I = 0, N = Chain.size(); I != N; ++I) {
|
2010-09-22 08:42:30 +08:00
|
|
|
TotalNumSLocEntries += Chain[I]->LocalNumSLocEntries;
|
2010-09-23 04:19:08 +08:00
|
|
|
NextSLocOffset += Chain[I]->LocalSLocSize;
|
2010-07-20 06:28:42 +08:00
|
|
|
TotalNumIdentifiers += Chain[I]->LocalNumIdentifiers;
|
2010-07-20 06:06:55 +08:00
|
|
|
TotalNumTypes += Chain[I]->LocalNumTypes;
|
|
|
|
TotalNumDecls += Chain[I]->LocalNumDecls;
|
2010-07-22 04:07:32 +08:00
|
|
|
TotalNumPreallocatedPreprocessingEntities +=
|
|
|
|
Chain[I]->NumPreallocatedPreprocessingEntities;
|
|
|
|
TotalNumMacroDefs += Chain[I]->LocalNumMacroDefinitions;
|
2010-08-05 04:40:17 +08:00
|
|
|
TotalNumSelectors += Chain[I]->LocalNumSelectors;
|
2010-07-20 06:06:55 +08:00
|
|
|
}
|
2010-09-23 04:19:08 +08:00
|
|
|
SourceMgr.PreallocateSLocEntries(this, TotalNumSLocEntries, NextSLocOffset);
|
2010-07-20 06:28:42 +08:00
|
|
|
IdentifiersLoaded.resize(TotalNumIdentifiers);
|
2010-07-20 06:06:55 +08:00
|
|
|
TypesLoaded.resize(TotalNumTypes);
|
|
|
|
DeclsLoaded.resize(TotalNumDecls);
|
2010-07-22 04:07:32 +08:00
|
|
|
MacroDefinitionsLoaded.resize(TotalNumMacroDefs);
|
|
|
|
if (PP) {
|
|
|
|
if (TotalNumIdentifiers > 0)
|
|
|
|
PP->getHeaderSearchInfo().SetExternalLookup(this);
|
|
|
|
if (TotalNumPreallocatedPreprocessingEntities > 0) {
|
|
|
|
if (!PP->getPreprocessingRecord())
|
|
|
|
PP->createPreprocessingRecord();
|
|
|
|
PP->getPreprocessingRecord()->SetExternalSource(*this,
|
|
|
|
TotalNumPreallocatedPreprocessingEntities);
|
|
|
|
}
|
|
|
|
}
|
2010-08-05 04:40:17 +08:00
|
|
|
SelectorsLoaded.resize(TotalNumSelectors);
|
2010-09-22 08:42:30 +08:00
|
|
|
// Preload SLocEntries.
|
|
|
|
for (unsigned I = 0, N = PreloadSLocEntries.size(); I != N; ++I) {
|
|
|
|
ASTReadResult Result = ReadSLocEntryRecord(PreloadSLocEntries[I]);
|
|
|
|
if (Result != Success)
|
|
|
|
return Result;
|
|
|
|
}
|
2010-07-20 06:06:55 +08:00
|
|
|
|
2010-07-17 04:41:52 +08:00
|
|
|
// Check the predefines buffers.
|
2010-07-27 08:27:13 +08:00
|
|
|
if (!DisableValidation && CheckPredefinesBuffers())
|
2009-04-29 04:33:11 +08:00
|
|
|
return IgnorePCH;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-06-19 08:03:23 +08:00
|
|
|
if (PP) {
|
2009-07-18 17:26:51 +08:00
|
|
|
// Initialization of keywords and pragmas occurs before the
|
2010-08-19 07:57:06 +08:00
|
|
|
// AST file is read, so there may be some identifiers that were
|
2009-06-19 08:03:23 +08:00
|
|
|
// loaded into the IdentifierTable before we intercepted the
|
|
|
|
// creation of identifiers. Iterate through the list of known
|
|
|
|
// identifiers and determine whether we have to establish
|
|
|
|
// preprocessor definitions or top-level identifier declaration
|
|
|
|
// chains for those identifiers.
|
|
|
|
//
|
|
|
|
// We copy the IdentifierInfo pointers to a small vector first,
|
|
|
|
// since de-serializing declarations or macro definitions can add
|
|
|
|
// new entries into the identifier table, invalidating the
|
|
|
|
// iterators.
|
|
|
|
llvm::SmallVector<IdentifierInfo *, 128> Identifiers;
|
|
|
|
for (IdentifierTable::iterator Id = PP->getIdentifierTable().begin(),
|
|
|
|
IdEnd = PP->getIdentifierTable().end();
|
|
|
|
Id != IdEnd; ++Id)
|
|
|
|
Identifiers.push_back(Id->second);
|
2010-07-22 04:07:32 +08:00
|
|
|
// We need to search the tables in all files.
|
|
|
|
for (unsigned J = 0, M = Chain.size(); J != M; ++J) {
|
2010-08-19 07:57:06 +08:00
|
|
|
ASTIdentifierLookupTable *IdTable
|
|
|
|
= (ASTIdentifierLookupTable *)Chain[J]->IdentifierLookupTable;
|
|
|
|
// Not all AST files necessarily have identifier tables, only the useful
|
2010-07-23 01:01:13 +08:00
|
|
|
// ones.
|
|
|
|
if (!IdTable)
|
|
|
|
continue;
|
2010-07-22 04:07:32 +08:00
|
|
|
for (unsigned I = 0, N = Identifiers.size(); I != N; ++I) {
|
|
|
|
IdentifierInfo *II = Identifiers[I];
|
|
|
|
// Look in the on-disk hash tables for an entry for this identifier
|
2010-08-19 07:57:06 +08:00
|
|
|
ASTIdentifierLookupTrait Info(*this, Chain[J]->Stream, II);
|
2010-07-22 04:07:32 +08:00
|
|
|
std::pair<const char*,unsigned> Key(II->getNameStart(),II->getLength());
|
2010-08-19 07:57:06 +08:00
|
|
|
ASTIdentifierLookupTable::iterator Pos = IdTable->find(Key, &Info);
|
2010-07-21 05:20:32 +08:00
|
|
|
if (Pos == IdTable->end())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// Dereferencing the iterator has the effect of populating the
|
|
|
|
// IdentifierInfo node with the various declarations it needs.
|
|
|
|
(void)*Pos;
|
|
|
|
}
|
2009-06-19 08:03:23 +08:00
|
|
|
}
|
Lazy deserialization of the declaration chains associated with
identifiers from a precompiled header.
This patch changes the primary name lookup method for entities within
a precompiled header. Previously, we would load all of the names of
declarations at translation unit scope into a large DenseMap (inside
the TranslationUnitDecl's DeclContext), and then perform a special
"last resort" lookup into this DeclContext when we knew there was a
PCH file (see Sema::LookupName). Now, when we see an identifier named
for the first time, we load all of the declarations with that name
that are visible from the translation unit into the IdentifierInfo's
chain of declarations. Thus, the explicit "look into the translation
unit's DeclContext" code is gone, and Sema effectively uses the same
IdentifierInfo-based name lookup mechanism whether we are using a PCH
file or not.
This approach should help PCH scale with the size of the input program
rather than the size of the PCH file. The "Hello, World!" application
with Carbon.h as a PCH file now loads 20% of the identifiers in the
PCH file rather than 85% of the identifiers.
90% of the 20% of identifiers loaded are actually loaded when we
deserialize the preprocessor state. The next step is to make the
preprocessor load macros lazily, which should drastically reduce the
number of types, declarations, and identifiers loaded for "Hello,
World".
llvm-svn: 69737
2009-04-22 06:25:48 +08:00
|
|
|
}
|
|
|
|
|
2009-06-19 08:03:23 +08:00
|
|
|
if (Context)
|
|
|
|
InitializeContext(*Context);
|
2009-04-15 05:18:50 +08:00
|
|
|
|
Lazy deserialization of the declaration chains associated with
identifiers from a precompiled header.
This patch changes the primary name lookup method for entities within
a precompiled header. Previously, we would load all of the names of
declarations at translation unit scope into a large DenseMap (inside
the TranslationUnitDecl's DeclContext), and then perform a special
"last resort" lookup into this DeclContext when we knew there was a
PCH file (see Sema::LookupName). Now, when we see an identifier named
for the first time, we load all of the declarations with that name
that are visible from the translation unit into the IdentifierInfo's
chain of declarations. Thus, the explicit "look into the translation
unit's DeclContext" code is gone, and Sema effectively uses the same
IdentifierInfo-based name lookup mechanism whether we are using a PCH
file or not.
This approach should help PCH scale with the size of the input program
rather than the size of the PCH file. The "Hello, World!" application
with Carbon.h as a PCH file now loads 20% of the identifiers in the
PCH file rather than 85% of the identifiers.
90% of the 20% of identifiers loaded are actually loaded when we
deserialize the preprocessor state. The next step is to make the
preprocessor load macros lazily, which should drastically reduce the
number of types, declarations, and identifiers loaded for "Hello,
World".
llvm-svn: 69737
2009-04-22 06:25:48 +08:00
|
|
|
return Success;
|
2009-04-15 05:18:50 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:56 +08:00
|
|
|
ASTReader::ASTReadResult ASTReader::ReadASTCore(llvm::StringRef FileName) {
|
2010-07-17 04:41:52 +08:00
|
|
|
Chain.push_back(new PerFileData());
|
|
|
|
PerFileData &F = *Chain.back();
|
|
|
|
|
2010-08-19 07:57:06 +08:00
|
|
|
// Set the AST file name.
|
2010-07-17 04:41:52 +08:00
|
|
|
F.FileName = FileName;
|
|
|
|
|
2010-08-19 07:57:06 +08:00
|
|
|
// Open the AST file.
|
2010-07-17 04:41:52 +08:00
|
|
|
//
|
|
|
|
// FIXME: This shouldn't be here, we should just take a raw_ostream.
|
|
|
|
std::string ErrStr;
|
|
|
|
F.Buffer.reset(llvm::MemoryBuffer::getFileOrSTDIN(FileName, &ErrStr));
|
|
|
|
if (!F.Buffer) {
|
|
|
|
Error(ErrStr.c_str());
|
|
|
|
return IgnorePCH;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Initialize the stream
|
|
|
|
F.StreamFile.init((const unsigned char *)F.Buffer->getBufferStart(),
|
|
|
|
(const unsigned char *)F.Buffer->getBufferEnd());
|
|
|
|
llvm::BitstreamCursor &Stream = F.Stream;
|
|
|
|
Stream.init(F.StreamFile);
|
2010-07-22 04:07:32 +08:00
|
|
|
F.SizeInBits = F.Buffer->getBufferSize() * 8;
|
2010-07-17 04:41:52 +08:00
|
|
|
|
|
|
|
// Sniff for the signature.
|
|
|
|
if (Stream.Read(8) != 'C' ||
|
|
|
|
Stream.Read(8) != 'P' ||
|
|
|
|
Stream.Read(8) != 'C' ||
|
|
|
|
Stream.Read(8) != 'H') {
|
|
|
|
Diag(diag::err_not_a_pch_file) << FileName;
|
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
|
2010-07-17 08:12:06 +08:00
|
|
|
while (!Stream.AtEndOfStream()) {
|
|
|
|
unsigned Code = Stream.ReadCode();
|
|
|
|
|
|
|
|
if (Code != llvm::bitc::ENTER_SUBBLOCK) {
|
2010-08-19 07:57:06 +08:00
|
|
|
Error("invalid record at top-level of AST file");
|
2010-07-17 08:12:06 +08:00
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned BlockID = Stream.ReadSubBlockID();
|
|
|
|
|
2010-08-19 07:57:06 +08:00
|
|
|
// We only know the AST subblock ID.
|
2010-07-17 08:12:06 +08:00
|
|
|
switch (BlockID) {
|
|
|
|
case llvm::bitc::BLOCKINFO_BLOCK_ID:
|
|
|
|
if (Stream.ReadBlockInfoBlock()) {
|
2010-08-19 07:57:06 +08:00
|
|
|
Error("malformed BlockInfoBlock in AST file");
|
2010-07-17 08:12:06 +08:00
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
break;
|
2010-08-19 07:57:32 +08:00
|
|
|
case AST_BLOCK_ID:
|
2010-08-19 07:56:56 +08:00
|
|
|
switch (ReadASTBlock(F)) {
|
2010-07-17 08:12:06 +08:00
|
|
|
case Success:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Failure:
|
|
|
|
return Failure;
|
|
|
|
|
|
|
|
case IgnorePCH:
|
|
|
|
// FIXME: We could consider reading through to the end of this
|
2010-08-19 07:57:06 +08:00
|
|
|
// AST block, skipping subblocks, to see if there are other
|
|
|
|
// AST blocks elsewhere.
|
2010-07-17 08:12:06 +08:00
|
|
|
|
|
|
|
// Clear out any preallocated source location entries, so that
|
|
|
|
// the source manager does not try to resolve them later.
|
|
|
|
SourceMgr.ClearPreallocatedSLocEntries();
|
|
|
|
|
|
|
|
// Remove the stat cache.
|
|
|
|
if (F.StatCache)
|
2010-08-19 07:57:06 +08:00
|
|
|
FileMgr.removeStatCache((ASTStatCache*)F.StatCache);
|
2010-07-17 08:12:06 +08:00
|
|
|
|
|
|
|
return IgnorePCH;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
if (Stream.SkipBlock()) {
|
2010-08-19 07:57:06 +08:00
|
|
|
Error("malformed block record in AST file");
|
2010-07-17 08:12:06 +08:00
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2010-07-17 04:41:52 +08:00
|
|
|
}
|
2010-07-17 08:12:06 +08:00
|
|
|
|
2010-07-17 04:41:52 +08:00
|
|
|
return Success;
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:43 +08:00
|
|
|
void ASTReader::setPreprocessor(Preprocessor &pp) {
|
Implement serialization and lazy deserialization of the preprocessing
record (which includes all macro instantiations and definitions). As
with all lay deserialization, this introduces a new external source
(here, an external preprocessing record source) that loads all of the
preprocessed entities prior to iterating over the entities.
The preprocessing record is an optional part of the precompiled header
that is disabled by default (enabled with
-detailed-preprocessing-record). When the preprocessor given to the
PCH writer has a preprocessing record, that record is written into the
PCH file. When the PCH reader is given a PCH file that contains a
preprocessing record, it will be lazily loaded (which, effectively,
implicitly adds -detailed-preprocessing-record). This is the first
case where we have sections of the precompiled header that are
added/removed based on a compilation flag, which is
unfortunate. However, this data consumes ~550k in the PCH file for
Cocoa.h (out of ~9.9MB), and there is a non-trivial cost to gathering
this detailed preprocessing information, so it's too expensive to turn
on by default. In the future, we should investigate a better encoding
of this information.
llvm-svn: 99002
2010-03-20 05:51:54 +08:00
|
|
|
PP = &pp;
|
2010-07-22 04:07:32 +08:00
|
|
|
|
|
|
|
unsigned TotalNum = 0;
|
|
|
|
for (unsigned I = 0, N = Chain.size(); I != N; ++I)
|
|
|
|
TotalNum += Chain[I]->NumPreallocatedPreprocessingEntities;
|
|
|
|
if (TotalNum) {
|
Implement serialization and lazy deserialization of the preprocessing
record (which includes all macro instantiations and definitions). As
with all lay deserialization, this introduces a new external source
(here, an external preprocessing record source) that loads all of the
preprocessed entities prior to iterating over the entities.
The preprocessing record is an optional part of the precompiled header
that is disabled by default (enabled with
-detailed-preprocessing-record). When the preprocessor given to the
PCH writer has a preprocessing record, that record is written into the
PCH file. When the PCH reader is given a PCH file that contains a
preprocessing record, it will be lazily loaded (which, effectively,
implicitly adds -detailed-preprocessing-record). This is the first
case where we have sections of the precompiled header that are
added/removed based on a compilation flag, which is
unfortunate. However, this data consumes ~550k in the PCH file for
Cocoa.h (out of ~9.9MB), and there is a non-trivial cost to gathering
this detailed preprocessing information, so it's too expensive to turn
on by default. In the future, we should investigate a better encoding
of this information.
llvm-svn: 99002
2010-03-20 05:51:54 +08:00
|
|
|
if (!PP->getPreprocessingRecord())
|
|
|
|
PP->createPreprocessingRecord();
|
2010-07-22 04:07:32 +08:00
|
|
|
PP->getPreprocessingRecord()->SetExternalSource(*this, TotalNum);
|
Implement serialization and lazy deserialization of the preprocessing
record (which includes all macro instantiations and definitions). As
with all lay deserialization, this introduces a new external source
(here, an external preprocessing record source) that loads all of the
preprocessed entities prior to iterating over the entities.
The preprocessing record is an optional part of the precompiled header
that is disabled by default (enabled with
-detailed-preprocessing-record). When the preprocessor given to the
PCH writer has a preprocessing record, that record is written into the
PCH file. When the PCH reader is given a PCH file that contains a
preprocessing record, it will be lazily loaded (which, effectively,
implicitly adds -detailed-preprocessing-record). This is the first
case where we have sections of the precompiled header that are
added/removed based on a compilation flag, which is
unfortunate. However, this data consumes ~550k in the PCH file for
Cocoa.h (out of ~9.9MB), and there is a non-trivial cost to gathering
this detailed preprocessing information, so it's too expensive to turn
on by default. In the future, we should investigate a better encoding
of this information.
llvm-svn: 99002
2010-03-20 05:51:54 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:43 +08:00
|
|
|
void ASTReader::InitializeContext(ASTContext &Ctx) {
|
2009-06-19 08:03:23 +08:00
|
|
|
Context = &Ctx;
|
|
|
|
assert(Context && "Passed null context!");
|
|
|
|
|
|
|
|
assert(PP && "Forgot to set Preprocessor ?");
|
|
|
|
PP->getIdentifierTable().setExternalIdentifierLookup(this);
|
|
|
|
PP->getHeaderSearchInfo().SetExternalLookup(this);
|
2010-01-05 03:18:44 +08:00
|
|
|
PP->setExternalSource(this);
|
2010-03-08 03:10:13 +08:00
|
|
|
|
2009-06-19 08:03:23 +08:00
|
|
|
// Load the translation unit declaration
|
2010-07-09 01:13:02 +08:00
|
|
|
GetTranslationUnitDecl();
|
2009-06-19 08:03:23 +08:00
|
|
|
|
|
|
|
// Load the special types.
|
|
|
|
Context->setBuiltinVaListType(
|
2010-08-19 07:57:32 +08:00
|
|
|
GetType(SpecialTypes[SPECIAL_TYPE_BUILTIN_VA_LIST]));
|
|
|
|
if (unsigned Id = SpecialTypes[SPECIAL_TYPE_OBJC_ID])
|
2009-06-19 08:03:23 +08:00
|
|
|
Context->setObjCIdType(GetType(Id));
|
2010-08-19 07:57:32 +08:00
|
|
|
if (unsigned Sel = SpecialTypes[SPECIAL_TYPE_OBJC_SELECTOR])
|
2009-06-19 08:03:23 +08:00
|
|
|
Context->setObjCSelType(GetType(Sel));
|
2010-08-19 07:57:32 +08:00
|
|
|
if (unsigned Proto = SpecialTypes[SPECIAL_TYPE_OBJC_PROTOCOL])
|
2009-06-19 08:03:23 +08:00
|
|
|
Context->setObjCProtoType(GetType(Proto));
|
2010-08-19 07:57:32 +08:00
|
|
|
if (unsigned Class = SpecialTypes[SPECIAL_TYPE_OBJC_CLASS])
|
2009-06-19 08:03:23 +08:00
|
|
|
Context->setObjCClassType(GetType(Class));
|
2009-07-11 07:34:53 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
if (unsigned String = SpecialTypes[SPECIAL_TYPE_CF_CONSTANT_STRING])
|
2009-06-19 08:03:23 +08:00
|
|
|
Context->setCFConstantStringType(GetType(String));
|
2009-09-09 23:08:12 +08:00
|
|
|
if (unsigned FastEnum
|
2010-08-19 07:57:32 +08:00
|
|
|
= SpecialTypes[SPECIAL_TYPE_OBJC_FAST_ENUMERATION_STATE])
|
2009-06-19 08:03:23 +08:00
|
|
|
Context->setObjCFastEnumerationStateType(GetType(FastEnum));
|
2010-08-19 07:57:32 +08:00
|
|
|
if (unsigned File = SpecialTypes[SPECIAL_TYPE_FILE]) {
|
2009-07-08 00:35:42 +08:00
|
|
|
QualType FileType = GetType(File);
|
2010-03-18 08:56:54 +08:00
|
|
|
if (FileType.isNull()) {
|
|
|
|
Error("FILE type is NULL");
|
|
|
|
return;
|
|
|
|
}
|
2009-09-22 07:43:11 +08:00
|
|
|
if (const TypedefType *Typedef = FileType->getAs<TypedefType>())
|
2009-07-08 00:35:42 +08:00
|
|
|
Context->setFILEDecl(Typedef->getDecl());
|
|
|
|
else {
|
2009-07-30 05:53:49 +08:00
|
|
|
const TagType *Tag = FileType->getAs<TagType>();
|
2010-03-18 08:56:54 +08:00
|
|
|
if (!Tag) {
|
2010-08-19 07:57:06 +08:00
|
|
|
Error("Invalid FILE type in AST file");
|
2010-03-18 08:56:54 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-07-08 00:35:42 +08:00
|
|
|
Context->setFILEDecl(Tag->getDecl());
|
|
|
|
}
|
|
|
|
}
|
2010-08-19 07:57:32 +08:00
|
|
|
if (unsigned Jmp_buf = SpecialTypes[SPECIAL_TYPE_jmp_buf]) {
|
2009-07-28 10:25:19 +08:00
|
|
|
QualType Jmp_bufType = GetType(Jmp_buf);
|
2010-03-18 08:56:54 +08:00
|
|
|
if (Jmp_bufType.isNull()) {
|
|
|
|
Error("jmp_bug type is NULL");
|
|
|
|
return;
|
|
|
|
}
|
2009-09-22 07:43:11 +08:00
|
|
|
if (const TypedefType *Typedef = Jmp_bufType->getAs<TypedefType>())
|
2009-07-28 10:25:19 +08:00
|
|
|
Context->setjmp_bufDecl(Typedef->getDecl());
|
|
|
|
else {
|
2009-07-30 05:53:49 +08:00
|
|
|
const TagType *Tag = Jmp_bufType->getAs<TagType>();
|
2010-03-18 08:56:54 +08:00
|
|
|
if (!Tag) {
|
2010-08-19 07:57:06 +08:00
|
|
|
Error("Invalid jmp_buf type in AST file");
|
2010-03-18 08:56:54 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-07-28 10:25:19 +08:00
|
|
|
Context->setjmp_bufDecl(Tag->getDecl());
|
|
|
|
}
|
|
|
|
}
|
2010-08-19 07:57:32 +08:00
|
|
|
if (unsigned Sigjmp_buf = SpecialTypes[SPECIAL_TYPE_sigjmp_buf]) {
|
2009-07-28 10:25:19 +08:00
|
|
|
QualType Sigjmp_bufType = GetType(Sigjmp_buf);
|
2010-03-18 08:56:54 +08:00
|
|
|
if (Sigjmp_bufType.isNull()) {
|
|
|
|
Error("sigjmp_buf type is NULL");
|
|
|
|
return;
|
|
|
|
}
|
2009-09-22 07:43:11 +08:00
|
|
|
if (const TypedefType *Typedef = Sigjmp_bufType->getAs<TypedefType>())
|
2009-07-28 10:25:19 +08:00
|
|
|
Context->setsigjmp_bufDecl(Typedef->getDecl());
|
|
|
|
else {
|
2009-07-30 05:53:49 +08:00
|
|
|
const TagType *Tag = Sigjmp_bufType->getAs<TagType>();
|
2010-08-19 07:57:06 +08:00
|
|
|
assert(Tag && "Invalid sigjmp_buf type in AST file");
|
2009-07-28 10:25:19 +08:00
|
|
|
Context->setsigjmp_bufDecl(Tag->getDecl());
|
|
|
|
}
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
if (unsigned ObjCIdRedef
|
2010-08-19 07:57:32 +08:00
|
|
|
= SpecialTypes[SPECIAL_TYPE_OBJC_ID_REDEFINITION])
|
2009-08-21 08:27:50 +08:00
|
|
|
Context->ObjCIdRedefinitionType = GetType(ObjCIdRedef);
|
2009-09-09 23:08:12 +08:00
|
|
|
if (unsigned ObjCClassRedef
|
2010-08-19 07:57:32 +08:00
|
|
|
= SpecialTypes[SPECIAL_TYPE_OBJC_CLASS_REDEFINITION])
|
2009-08-21 08:27:50 +08:00
|
|
|
Context->ObjCClassRedefinitionType = GetType(ObjCClassRedef);
|
2010-08-19 07:57:32 +08:00
|
|
|
if (unsigned String = SpecialTypes[SPECIAL_TYPE_BLOCK_DESCRIPTOR])
|
2009-10-20 10:12:22 +08:00
|
|
|
Context->setBlockDescriptorType(GetType(String));
|
2009-10-22 08:49:09 +08:00
|
|
|
if (unsigned String
|
2010-08-19 07:57:32 +08:00
|
|
|
= SpecialTypes[SPECIAL_TYPE_BLOCK_EXTENDED_DESCRIPTOR])
|
2009-10-22 08:49:09 +08:00
|
|
|
Context->setBlockDescriptorExtendedType(GetType(String));
|
2010-04-24 01:41:07 +08:00
|
|
|
if (unsigned ObjCSelRedef
|
2010-08-19 07:57:32 +08:00
|
|
|
= SpecialTypes[SPECIAL_TYPE_OBJC_SEL_REDEFINITION])
|
2010-04-24 01:41:07 +08:00
|
|
|
Context->ObjCSelRedefinitionType = GetType(ObjCSelRedef);
|
2010-08-19 07:57:32 +08:00
|
|
|
if (unsigned String = SpecialTypes[SPECIAL_TYPE_NS_CONSTANT_STRING])
|
2010-04-24 01:41:07 +08:00
|
|
|
Context->setNSConstantStringType(GetType(String));
|
2010-07-05 05:44:19 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
if (SpecialTypes[SPECIAL_TYPE_INT128_INSTALLED])
|
2010-07-05 05:44:19 +08:00
|
|
|
Context->setInt128Installed();
|
2009-06-19 08:03:23 +08:00
|
|
|
}
|
|
|
|
|
2009-05-12 09:31:05 +08:00
|
|
|
/// \brief Retrieve the name of the original source file name
|
2010-08-19 07:57:06 +08:00
|
|
|
/// directly from the AST file, without actually loading the AST
|
2009-05-12 09:31:05 +08:00
|
|
|
/// file.
|
2010-08-19 07:57:06 +08:00
|
|
|
std::string ASTReader::getOriginalSourceFile(const std::string &ASTFileName,
|
2009-12-03 17:13:06 +08:00
|
|
|
Diagnostic &Diags) {
|
2010-08-19 07:57:06 +08:00
|
|
|
// Open the AST file.
|
2009-05-12 09:31:05 +08:00
|
|
|
std::string ErrStr;
|
|
|
|
llvm::OwningPtr<llvm::MemoryBuffer> Buffer;
|
2010-08-19 07:57:06 +08:00
|
|
|
Buffer.reset(llvm::MemoryBuffer::getFile(ASTFileName.c_str(), &ErrStr));
|
2009-05-12 09:31:05 +08:00
|
|
|
if (!Buffer) {
|
2009-12-03 17:13:06 +08:00
|
|
|
Diags.Report(diag::err_fe_unable_to_read_pch_file) << ErrStr;
|
2009-05-12 09:31:05 +08:00
|
|
|
return std::string();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Initialize the stream
|
|
|
|
llvm::BitstreamReader StreamFile;
|
|
|
|
llvm::BitstreamCursor Stream;
|
2009-09-09 23:08:12 +08:00
|
|
|
StreamFile.init((const unsigned char *)Buffer->getBufferStart(),
|
2009-05-12 09:31:05 +08:00
|
|
|
(const unsigned char *)Buffer->getBufferEnd());
|
|
|
|
Stream.init(StreamFile);
|
|
|
|
|
|
|
|
// Sniff for the signature.
|
|
|
|
if (Stream.Read(8) != 'C' ||
|
|
|
|
Stream.Read(8) != 'P' ||
|
|
|
|
Stream.Read(8) != 'C' ||
|
|
|
|
Stream.Read(8) != 'H') {
|
2010-08-19 07:57:06 +08:00
|
|
|
Diags.Report(diag::err_fe_not_a_pch_file) << ASTFileName;
|
2009-05-12 09:31:05 +08:00
|
|
|
return std::string();
|
|
|
|
}
|
|
|
|
|
|
|
|
RecordData Record;
|
|
|
|
while (!Stream.AtEndOfStream()) {
|
|
|
|
unsigned Code = Stream.ReadCode();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-05-12 09:31:05 +08:00
|
|
|
if (Code == llvm::bitc::ENTER_SUBBLOCK) {
|
|
|
|
unsigned BlockID = Stream.ReadSubBlockID();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-08-19 07:57:06 +08:00
|
|
|
// We only know the AST subblock ID.
|
2009-05-12 09:31:05 +08:00
|
|
|
switch (BlockID) {
|
2010-08-19 07:57:32 +08:00
|
|
|
case AST_BLOCK_ID:
|
|
|
|
if (Stream.EnterSubBlock(AST_BLOCK_ID)) {
|
2010-08-19 07:57:06 +08:00
|
|
|
Diags.Report(diag::err_fe_pch_malformed_block) << ASTFileName;
|
2009-05-12 09:31:05 +08:00
|
|
|
return std::string();
|
|
|
|
}
|
|
|
|
break;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-05-12 09:31:05 +08:00
|
|
|
default:
|
|
|
|
if (Stream.SkipBlock()) {
|
2010-08-19 07:57:06 +08:00
|
|
|
Diags.Report(diag::err_fe_pch_malformed_block) << ASTFileName;
|
2009-05-12 09:31:05 +08:00
|
|
|
return std::string();
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Code == llvm::bitc::END_BLOCK) {
|
|
|
|
if (Stream.ReadBlockEnd()) {
|
2010-08-19 07:57:06 +08:00
|
|
|
Diags.Report(diag::err_fe_pch_error_at_end_block) << ASTFileName;
|
2009-05-12 09:31:05 +08:00
|
|
|
return std::string();
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Code == llvm::bitc::DEFINE_ABBREV) {
|
|
|
|
Stream.ReadAbbrevRecord();
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
Record.clear();
|
|
|
|
const char *BlobStart = 0;
|
|
|
|
unsigned BlobLen = 0;
|
2009-09-09 23:08:12 +08:00
|
|
|
if (Stream.ReadRecord(Code, Record, &BlobStart, &BlobLen)
|
2010-08-19 07:57:32 +08:00
|
|
|
== ORIGINAL_FILE_NAME)
|
2009-05-12 09:31:05 +08:00
|
|
|
return std::string(BlobStart, BlobLen);
|
2009-09-09 23:08:12 +08:00
|
|
|
}
|
2009-05-12 09:31:05 +08:00
|
|
|
|
|
|
|
return std::string();
|
|
|
|
}
|
|
|
|
|
2009-04-11 04:39:37 +08:00
|
|
|
/// \brief Parse the record that corresponds to a LangOptions data
|
|
|
|
/// structure.
|
|
|
|
///
|
2010-08-19 07:57:06 +08:00
|
|
|
/// This routine parses the language options from the AST file and then gives
|
|
|
|
/// them to the AST listener if one is set.
|
2009-04-11 04:39:37 +08:00
|
|
|
///
|
2010-08-19 07:57:06 +08:00
|
|
|
/// \returns true if the listener deems the file unacceptable, false otherwise.
|
2010-08-19 07:56:43 +08:00
|
|
|
bool ASTReader::ParseLanguageOptions(
|
2009-04-11 04:39:37 +08:00
|
|
|
const llvm::SmallVectorImpl<uint64_t> &Record) {
|
2009-06-19 08:03:23 +08:00
|
|
|
if (Listener) {
|
|
|
|
LangOptions LangOpts;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-06-19 08:03:23 +08:00
|
|
|
#define PARSE_LANGOPT(Option) \
|
|
|
|
LangOpts.Option = Record[Idx]; \
|
|
|
|
++Idx
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-06-19 08:03:23 +08:00
|
|
|
unsigned Idx = 0;
|
|
|
|
PARSE_LANGOPT(Trigraphs);
|
|
|
|
PARSE_LANGOPT(BCPLComment);
|
|
|
|
PARSE_LANGOPT(DollarIdents);
|
|
|
|
PARSE_LANGOPT(AsmPreprocessor);
|
|
|
|
PARSE_LANGOPT(GNUMode);
|
2010-04-18 04:17:31 +08:00
|
|
|
PARSE_LANGOPT(GNUKeywords);
|
2009-06-19 08:03:23 +08:00
|
|
|
PARSE_LANGOPT(ImplicitInt);
|
|
|
|
PARSE_LANGOPT(Digraphs);
|
|
|
|
PARSE_LANGOPT(HexFloats);
|
|
|
|
PARSE_LANGOPT(C99);
|
|
|
|
PARSE_LANGOPT(Microsoft);
|
|
|
|
PARSE_LANGOPT(CPlusPlus);
|
|
|
|
PARSE_LANGOPT(CPlusPlus0x);
|
|
|
|
PARSE_LANGOPT(CXXOperatorNames);
|
|
|
|
PARSE_LANGOPT(ObjC1);
|
|
|
|
PARSE_LANGOPT(ObjC2);
|
|
|
|
PARSE_LANGOPT(ObjCNonFragileABI);
|
2010-02-10 03:31:38 +08:00
|
|
|
PARSE_LANGOPT(ObjCNonFragileABI2);
|
2010-04-23 05:01:59 +08:00
|
|
|
PARSE_LANGOPT(NoConstantCFStrings);
|
2009-06-19 08:03:23 +08:00
|
|
|
PARSE_LANGOPT(PascalStrings);
|
|
|
|
PARSE_LANGOPT(WritableStrings);
|
|
|
|
PARSE_LANGOPT(LaxVectorConversions);
|
2009-06-26 07:01:11 +08:00
|
|
|
PARSE_LANGOPT(AltiVec);
|
2009-06-19 08:03:23 +08:00
|
|
|
PARSE_LANGOPT(Exceptions);
|
2010-02-11 02:48:44 +08:00
|
|
|
PARSE_LANGOPT(SjLjExceptions);
|
2009-06-19 08:03:23 +08:00
|
|
|
PARSE_LANGOPT(NeXTRuntime);
|
|
|
|
PARSE_LANGOPT(Freestanding);
|
|
|
|
PARSE_LANGOPT(NoBuiltin);
|
|
|
|
PARSE_LANGOPT(ThreadsafeStatics);
|
2009-09-03 22:36:33 +08:00
|
|
|
PARSE_LANGOPT(POSIXThreads);
|
2009-06-19 08:03:23 +08:00
|
|
|
PARSE_LANGOPT(Blocks);
|
|
|
|
PARSE_LANGOPT(EmitAllDecls);
|
|
|
|
PARSE_LANGOPT(MathErrno);
|
2010-06-27 05:25:03 +08:00
|
|
|
LangOpts.setSignedOverflowBehavior((LangOptions::SignedOverflowBehaviorTy)
|
|
|
|
Record[Idx++]);
|
2009-06-19 08:03:23 +08:00
|
|
|
PARSE_LANGOPT(HeinousExtensions);
|
|
|
|
PARSE_LANGOPT(Optimize);
|
|
|
|
PARSE_LANGOPT(OptimizeSize);
|
|
|
|
PARSE_LANGOPT(Static);
|
|
|
|
PARSE_LANGOPT(PICLevel);
|
|
|
|
PARSE_LANGOPT(GNUInline);
|
|
|
|
PARSE_LANGOPT(NoInline);
|
|
|
|
PARSE_LANGOPT(AccessControl);
|
|
|
|
PARSE_LANGOPT(CharIsSigned);
|
2009-11-06 04:14:16 +08:00
|
|
|
PARSE_LANGOPT(ShortWChar);
|
2010-06-27 05:25:03 +08:00
|
|
|
LangOpts.setGCMode((LangOptions::GCMode)Record[Idx++]);
|
|
|
|
LangOpts.setVisibilityMode((LangOptions::VisibilityMode)Record[Idx++]);
|
2009-09-21 12:16:19 +08:00
|
|
|
LangOpts.setStackProtectorMode((LangOptions::StackProtectorMode)
|
2010-06-27 05:25:03 +08:00
|
|
|
Record[Idx++]);
|
2009-06-19 08:03:23 +08:00
|
|
|
PARSE_LANGOPT(InstantiationDepth);
|
2009-06-26 07:01:11 +08:00
|
|
|
PARSE_LANGOPT(OpenCL);
|
2009-12-12 09:27:46 +08:00
|
|
|
PARSE_LANGOPT(CatchUndefined);
|
|
|
|
// FIXME: Missing ElideConstructors?!
|
2009-06-19 08:03:23 +08:00
|
|
|
#undef PARSE_LANGOPT
|
|
|
|
|
|
|
|
return Listener->ReadLanguageOptions(LangOpts);
|
2009-04-11 04:39:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:43 +08:00
|
|
|
void ASTReader::ReadPreprocessedEntities() {
|
Implement serialization and lazy deserialization of the preprocessing
record (which includes all macro instantiations and definitions). As
with all lay deserialization, this introduces a new external source
(here, an external preprocessing record source) that loads all of the
preprocessed entities prior to iterating over the entities.
The preprocessing record is an optional part of the precompiled header
that is disabled by default (enabled with
-detailed-preprocessing-record). When the preprocessor given to the
PCH writer has a preprocessing record, that record is written into the
PCH file. When the PCH reader is given a PCH file that contains a
preprocessing record, it will be lazily loaded (which, effectively,
implicitly adds -detailed-preprocessing-record). This is the first
case where we have sections of the precompiled header that are
added/removed based on a compilation flag, which is
unfortunate. However, this data consumes ~550k in the PCH file for
Cocoa.h (out of ~9.9MB), and there is a non-trivial cost to gathering
this detailed preprocessing information, so it's too expensive to turn
on by default. In the future, we should investigate a better encoding
of this information.
llvm-svn: 99002
2010-03-20 05:51:54 +08:00
|
|
|
ReadDefinedMacros();
|
|
|
|
}
|
|
|
|
|
2010-07-21 06:37:49 +08:00
|
|
|
/// \brief Get the correct cursor and offset for loading a type.
|
2010-08-19 07:56:43 +08:00
|
|
|
ASTReader::RecordLocation ASTReader::TypeCursorForIndex(unsigned Index) {
|
2010-07-21 06:37:49 +08:00
|
|
|
PerFileData *F = 0;
|
|
|
|
for (unsigned I = 0, N = Chain.size(); I != N; ++I) {
|
|
|
|
F = Chain[N - I - 1];
|
|
|
|
if (Index < F->LocalNumTypes)
|
|
|
|
break;
|
|
|
|
Index -= F->LocalNumTypes;
|
|
|
|
}
|
|
|
|
assert(F && F->LocalNumTypes > Index && "Broken chain");
|
2010-07-21 06:55:31 +08:00
|
|
|
return RecordLocation(&F->DeclsCursor, F->TypeOffsets[Index]);
|
2010-07-21 06:37:49 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief Read and return the type with the given index..
|
2009-04-10 06:27:44 +08:00
|
|
|
///
|
2010-07-21 06:37:49 +08:00
|
|
|
/// The index is the type ID, shifted and minus the number of predefs. This
|
|
|
|
/// routine actually reads the record corresponding to the type at the given
|
|
|
|
/// location. It is a helper routine for GetType, which deals with reading type
|
|
|
|
/// IDs.
|
2010-08-19 07:56:43 +08:00
|
|
|
QualType ASTReader::ReadTypeRecord(unsigned Index) {
|
2010-07-21 06:37:49 +08:00
|
|
|
RecordLocation Loc = TypeCursorForIndex(Index);
|
2010-07-21 06:55:31 +08:00
|
|
|
llvm::BitstreamCursor &DeclsCursor = *Loc.first;
|
2010-07-17 01:50:48 +08:00
|
|
|
|
2009-04-15 05:18:50 +08:00
|
|
|
// Keep track of where we are in the stream, then jump back there
|
|
|
|
// after reading this type.
|
2009-10-17 08:13:19 +08:00
|
|
|
SavedStreamPosition SavedPosition(DeclsCursor);
|
2009-04-15 05:18:50 +08:00
|
|
|
|
2010-06-29 06:28:35 +08:00
|
|
|
ReadingKindTracker ReadingKind(Read_Type, *this);
|
2010-08-12 02:52:41 +08:00
|
|
|
|
2009-07-07 02:54:52 +08:00
|
|
|
// Note that we are loading a type record.
|
2010-07-30 18:03:16 +08:00
|
|
|
Deserializing AType(this);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-07-21 06:37:49 +08:00
|
|
|
DeclsCursor.JumpToBit(Loc.second);
|
2009-04-10 06:27:44 +08:00
|
|
|
RecordData Record;
|
2009-10-17 08:13:19 +08:00
|
|
|
unsigned Code = DeclsCursor.ReadCode();
|
2010-08-19 07:57:32 +08:00
|
|
|
switch ((TypeCode)DeclsCursor.ReadRecord(Code, Record)) {
|
|
|
|
case TYPE_EXT_QUAL: {
|
2010-03-18 08:56:54 +08:00
|
|
|
if (Record.size() != 2) {
|
|
|
|
Error("Incorrect encoding of extended qualifier type");
|
|
|
|
return QualType();
|
|
|
|
}
|
2009-04-16 06:00:08 +08:00
|
|
|
QualType Base = GetType(Record[0]);
|
2009-09-25 03:53:00 +08:00
|
|
|
Qualifiers Quals = Qualifiers::fromOpaqueValue(Record[1]);
|
|
|
|
return Context->getQualifiedType(Base, Quals);
|
2009-04-16 06:00:08 +08:00
|
|
|
}
|
2009-04-14 04:46:52 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TYPE_COMPLEX: {
|
2010-03-18 08:56:54 +08:00
|
|
|
if (Record.size() != 1) {
|
|
|
|
Error("Incorrect encoding of complex type");
|
|
|
|
return QualType();
|
|
|
|
}
|
2009-04-10 06:27:44 +08:00
|
|
|
QualType ElemType = GetType(Record[0]);
|
2009-04-28 05:45:14 +08:00
|
|
|
return Context->getComplexType(ElemType);
|
2009-04-10 06:27:44 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TYPE_POINTER: {
|
2010-03-18 08:56:54 +08:00
|
|
|
if (Record.size() != 1) {
|
|
|
|
Error("Incorrect encoding of pointer type");
|
|
|
|
return QualType();
|
|
|
|
}
|
2009-04-10 06:27:44 +08:00
|
|
|
QualType PointeeType = GetType(Record[0]);
|
2009-04-28 05:45:14 +08:00
|
|
|
return Context->getPointerType(PointeeType);
|
2009-04-10 06:27:44 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TYPE_BLOCK_POINTER: {
|
2010-03-18 08:56:54 +08:00
|
|
|
if (Record.size() != 1) {
|
|
|
|
Error("Incorrect encoding of block pointer type");
|
|
|
|
return QualType();
|
|
|
|
}
|
2009-04-10 06:27:44 +08:00
|
|
|
QualType PointeeType = GetType(Record[0]);
|
2009-04-28 05:45:14 +08:00
|
|
|
return Context->getBlockPointerType(PointeeType);
|
2009-04-10 06:27:44 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TYPE_LVALUE_REFERENCE: {
|
2010-03-18 08:56:54 +08:00
|
|
|
if (Record.size() != 1) {
|
|
|
|
Error("Incorrect encoding of lvalue reference type");
|
|
|
|
return QualType();
|
|
|
|
}
|
2009-04-10 06:27:44 +08:00
|
|
|
QualType PointeeType = GetType(Record[0]);
|
2009-04-28 05:45:14 +08:00
|
|
|
return Context->getLValueReferenceType(PointeeType);
|
2009-04-10 06:27:44 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TYPE_RVALUE_REFERENCE: {
|
2010-03-18 08:56:54 +08:00
|
|
|
if (Record.size() != 1) {
|
|
|
|
Error("Incorrect encoding of rvalue reference type");
|
|
|
|
return QualType();
|
|
|
|
}
|
2009-04-10 06:27:44 +08:00
|
|
|
QualType PointeeType = GetType(Record[0]);
|
2009-04-28 05:45:14 +08:00
|
|
|
return Context->getRValueReferenceType(PointeeType);
|
2009-04-10 06:27:44 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TYPE_MEMBER_POINTER: {
|
2010-07-02 19:55:15 +08:00
|
|
|
if (Record.size() != 2) {
|
2010-03-18 08:56:54 +08:00
|
|
|
Error("Incorrect encoding of member pointer type");
|
|
|
|
return QualType();
|
|
|
|
}
|
2009-04-10 06:27:44 +08:00
|
|
|
QualType PointeeType = GetType(Record[0]);
|
|
|
|
QualType ClassType = GetType(Record[1]);
|
2009-04-28 05:45:14 +08:00
|
|
|
return Context->getMemberPointerType(PointeeType, ClassType.getTypePtr());
|
2009-04-10 06:27:44 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TYPE_CONSTANT_ARRAY: {
|
2009-04-14 04:46:52 +08:00
|
|
|
QualType ElementType = GetType(Record[0]);
|
|
|
|
ArrayType::ArraySizeModifier ASM = (ArrayType::ArraySizeModifier)Record[1];
|
|
|
|
unsigned IndexTypeQuals = Record[2];
|
|
|
|
unsigned Idx = 3;
|
|
|
|
llvm::APInt Size = ReadAPInt(Record, Idx);
|
2009-07-06 23:59:29 +08:00
|
|
|
return Context->getConstantArrayType(ElementType, Size,
|
|
|
|
ASM, IndexTypeQuals);
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TYPE_INCOMPLETE_ARRAY: {
|
2009-04-14 04:46:52 +08:00
|
|
|
QualType ElementType = GetType(Record[0]);
|
|
|
|
ArrayType::ArraySizeModifier ASM = (ArrayType::ArraySizeModifier)Record[1];
|
|
|
|
unsigned IndexTypeQuals = Record[2];
|
2009-04-28 05:45:14 +08:00
|
|
|
return Context->getIncompleteArrayType(ElementType, ASM, IndexTypeQuals);
|
2009-04-14 04:46:52 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TYPE_VARIABLE_ARRAY: {
|
2009-04-15 05:18:50 +08:00
|
|
|
QualType ElementType = GetType(Record[0]);
|
|
|
|
ArrayType::ArraySizeModifier ASM = (ArrayType::ArraySizeModifier)Record[1];
|
|
|
|
unsigned IndexTypeQuals = Record[2];
|
2009-07-06 23:59:29 +08:00
|
|
|
SourceLocation LBLoc = SourceLocation::getFromRawEncoding(Record[3]);
|
|
|
|
SourceLocation RBLoc = SourceLocation::getFromRawEncoding(Record[4]);
|
2010-07-23 06:43:28 +08:00
|
|
|
return Context->getVariableArrayType(ElementType, ReadExpr(DeclsCursor),
|
2009-07-06 23:59:29 +08:00
|
|
|
ASM, IndexTypeQuals,
|
|
|
|
SourceRange(LBLoc, RBLoc));
|
2009-04-14 04:46:52 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TYPE_VECTOR: {
|
2010-06-23 14:00:24 +08:00
|
|
|
if (Record.size() != 3) {
|
2010-08-19 07:57:06 +08:00
|
|
|
Error("incorrect encoding of vector type in AST file");
|
2009-04-14 04:46:52 +08:00
|
|
|
return QualType();
|
|
|
|
}
|
|
|
|
|
|
|
|
QualType ElementType = GetType(Record[0]);
|
|
|
|
unsigned NumElements = Record[1];
|
2010-06-23 14:00:24 +08:00
|
|
|
unsigned AltiVecSpec = Record[2];
|
|
|
|
return Context->getVectorType(ElementType, NumElements,
|
|
|
|
(VectorType::AltiVecSpecific)AltiVecSpec);
|
2009-04-14 04:46:52 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TYPE_EXT_VECTOR: {
|
2010-06-23 14:00:24 +08:00
|
|
|
if (Record.size() != 3) {
|
2010-08-19 07:57:06 +08:00
|
|
|
Error("incorrect encoding of extended vector type in AST file");
|
2009-04-14 04:46:52 +08:00
|
|
|
return QualType();
|
|
|
|
}
|
|
|
|
|
|
|
|
QualType ElementType = GetType(Record[0]);
|
|
|
|
unsigned NumElements = Record[1];
|
2009-04-28 05:45:14 +08:00
|
|
|
return Context->getExtVectorType(ElementType, NumElements);
|
2009-04-14 04:46:52 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TYPE_FUNCTION_NO_PROTO: {
|
2010-03-31 06:15:11 +08:00
|
|
|
if (Record.size() != 4) {
|
2009-04-29 05:53:25 +08:00
|
|
|
Error("incorrect encoding of no-proto function type");
|
2009-04-14 04:46:52 +08:00
|
|
|
return QualType();
|
|
|
|
}
|
|
|
|
QualType ResultType = GetType(Record[0]);
|
2010-03-31 06:15:11 +08:00
|
|
|
FunctionType::ExtInfo Info(Record[1], Record[2], (CallingConv)Record[3]);
|
2010-03-31 04:24:48 +08:00
|
|
|
return Context->getFunctionNoProtoType(ResultType, Info);
|
2009-04-14 04:46:52 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TYPE_FUNCTION_PROTO: {
|
2009-04-14 04:46:52 +08:00
|
|
|
QualType ResultType = GetType(Record[0]);
|
2009-12-23 02:11:50 +08:00
|
|
|
bool NoReturn = Record[1];
|
2010-03-31 06:15:11 +08:00
|
|
|
unsigned RegParm = Record[2];
|
|
|
|
CallingConv CallConv = (CallingConv)Record[3];
|
|
|
|
unsigned Idx = 4;
|
2009-04-14 04:46:52 +08:00
|
|
|
unsigned NumParams = Record[Idx++];
|
|
|
|
llvm::SmallVector<QualType, 16> ParamTypes;
|
|
|
|
for (unsigned I = 0; I != NumParams; ++I)
|
|
|
|
ParamTypes.push_back(GetType(Record[Idx++]));
|
|
|
|
bool isVariadic = Record[Idx++];
|
|
|
|
unsigned Quals = Record[Idx++];
|
2009-05-28 06:11:52 +08:00
|
|
|
bool hasExceptionSpec = Record[Idx++];
|
|
|
|
bool hasAnyExceptionSpec = Record[Idx++];
|
|
|
|
unsigned NumExceptions = Record[Idx++];
|
|
|
|
llvm::SmallVector<QualType, 2> Exceptions;
|
|
|
|
for (unsigned I = 0; I != NumExceptions; ++I)
|
|
|
|
Exceptions.push_back(GetType(Record[Idx++]));
|
2009-05-21 17:52:38 +08:00
|
|
|
return Context->getFunctionType(ResultType, ParamTypes.data(), NumParams,
|
2009-05-28 06:11:52 +08:00
|
|
|
isVariadic, Quals, hasExceptionSpec,
|
|
|
|
hasAnyExceptionSpec, NumExceptions,
|
2010-03-31 04:24:48 +08:00
|
|
|
Exceptions.data(),
|
2010-03-31 06:15:11 +08:00
|
|
|
FunctionType::ExtInfo(NoReturn, RegParm,
|
|
|
|
CallConv));
|
2009-04-14 04:46:52 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TYPE_UNRESOLVED_USING:
|
2009-12-05 06:46:56 +08:00
|
|
|
return Context->getTypeDeclType(
|
|
|
|
cast<UnresolvedUsingTypenameDecl>(GetDecl(Record[0])));
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TYPE_TYPEDEF: {
|
2010-07-02 19:55:11 +08:00
|
|
|
if (Record.size() != 2) {
|
2010-03-18 08:56:54 +08:00
|
|
|
Error("incorrect encoding of typedef type");
|
|
|
|
return QualType();
|
|
|
|
}
|
2010-07-02 19:55:11 +08:00
|
|
|
TypedefDecl *Decl = cast<TypedefDecl>(GetDecl(Record[0]));
|
|
|
|
QualType Canonical = GetType(Record[1]);
|
|
|
|
return Context->getTypedefType(Decl, Canonical);
|
|
|
|
}
|
2009-04-14 04:46:52 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TYPE_TYPEOF_EXPR:
|
2010-07-23 06:43:28 +08:00
|
|
|
return Context->getTypeOfExprType(ReadExpr(DeclsCursor));
|
2009-04-14 04:46:52 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TYPE_TYPEOF: {
|
2009-04-14 04:46:52 +08:00
|
|
|
if (Record.size() != 1) {
|
2010-08-19 07:57:06 +08:00
|
|
|
Error("incorrect encoding of typeof(type) in AST file");
|
2009-04-14 04:46:52 +08:00
|
|
|
return QualType();
|
|
|
|
}
|
|
|
|
QualType UnderlyingType = GetType(Record[0]);
|
2009-04-28 05:45:14 +08:00
|
|
|
return Context->getTypeOfType(UnderlyingType);
|
2009-04-14 04:46:52 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TYPE_DECLTYPE:
|
2010-07-23 06:43:28 +08:00
|
|
|
return Context->getDecltypeType(ReadExpr(DeclsCursor));
|
2009-06-25 03:06:50 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TYPE_RECORD: {
|
2010-07-08 21:09:53 +08:00
|
|
|
if (Record.size() != 2) {
|
2010-03-18 08:56:54 +08:00
|
|
|
Error("incorrect encoding of record type");
|
|
|
|
return QualType();
|
|
|
|
}
|
2010-07-08 21:09:53 +08:00
|
|
|
bool IsDependent = Record[0];
|
|
|
|
QualType T = Context->getRecordType(cast<RecordDecl>(GetDecl(Record[1])));
|
|
|
|
T->Dependent = IsDependent;
|
|
|
|
return T;
|
|
|
|
}
|
2009-04-14 04:46:52 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TYPE_ENUM: {
|
2010-07-08 21:09:53 +08:00
|
|
|
if (Record.size() != 2) {
|
2010-03-18 08:56:54 +08:00
|
|
|
Error("incorrect encoding of enum type");
|
|
|
|
return QualType();
|
|
|
|
}
|
2010-07-08 21:09:53 +08:00
|
|
|
bool IsDependent = Record[0];
|
|
|
|
QualType T = Context->getEnumType(cast<EnumDecl>(GetDecl(Record[1])));
|
|
|
|
T->Dependent = IsDependent;
|
|
|
|
return T;
|
|
|
|
}
|
2009-04-14 02:14:40 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TYPE_ELABORATED: {
|
2010-06-26 00:24:58 +08:00
|
|
|
unsigned Idx = 0;
|
|
|
|
ElaboratedTypeKeyword Keyword = (ElaboratedTypeKeyword)Record[Idx++];
|
|
|
|
NestedNameSpecifier *NNS = ReadNestedNameSpecifier(Record, Idx);
|
|
|
|
QualType NamedType = GetType(Record[Idx++]);
|
|
|
|
return Context->getElaboratedType(Keyword, NNS, NamedType);
|
2009-09-05 08:15:47 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TYPE_OBJC_INTERFACE: {
|
2009-04-22 14:45:28 +08:00
|
|
|
unsigned Idx = 0;
|
|
|
|
ObjCInterfaceDecl *ItfD = cast<ObjCInterfaceDecl>(GetDecl(Record[Idx++]));
|
2010-05-15 19:32:37 +08:00
|
|
|
return Context->getObjCInterfaceType(ItfD);
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TYPE_OBJC_OBJECT: {
|
2010-05-15 19:32:37 +08:00
|
|
|
unsigned Idx = 0;
|
|
|
|
QualType Base = GetType(Record[Idx++]);
|
2009-04-22 14:45:28 +08:00
|
|
|
unsigned NumProtos = Record[Idx++];
|
|
|
|
llvm::SmallVector<ObjCProtocolDecl*, 4> Protos;
|
|
|
|
for (unsigned I = 0; I != NumProtos; ++I)
|
|
|
|
Protos.push_back(cast<ObjCProtocolDecl>(GetDecl(Record[Idx++])));
|
2010-05-15 19:32:37 +08:00
|
|
|
return Context->getObjCObjectType(Base, Protos.data(), NumProtos);
|
2009-04-22 14:45:28 +08:00
|
|
|
}
|
2009-04-14 04:46:52 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TYPE_OBJC_OBJECT_POINTER: {
|
2009-04-22 14:40:03 +08:00
|
|
|
unsigned Idx = 0;
|
2010-05-15 19:32:37 +08:00
|
|
|
QualType Pointee = GetType(Record[Idx++]);
|
|
|
|
return Context->getObjCObjectPointerType(Pointee);
|
2009-04-22 14:40:03 +08:00
|
|
|
}
|
2009-09-30 03:42:55 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TYPE_SUBST_TEMPLATE_TYPE_PARM: {
|
2009-10-18 17:09:24 +08:00
|
|
|
unsigned Idx = 0;
|
|
|
|
QualType Parm = GetType(Record[Idx++]);
|
|
|
|
QualType Replacement = GetType(Record[Idx++]);
|
|
|
|
return
|
|
|
|
Context->getSubstTemplateTypeParmType(cast<TemplateTypeParmType>(Parm),
|
|
|
|
Replacement);
|
|
|
|
}
|
2010-03-10 11:28:59 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TYPE_INJECTED_CLASS_NAME: {
|
2010-03-10 11:28:59 +08:00
|
|
|
CXXRecordDecl *D = cast<CXXRecordDecl>(GetDecl(Record[0]));
|
|
|
|
QualType TST = GetType(Record[1]); // probably derivable
|
2010-07-02 19:55:20 +08:00
|
|
|
// FIXME: ASTContext::getInjectedClassNameType is not currently suitable
|
2010-08-19 07:57:06 +08:00
|
|
|
// for AST reading, too much interdependencies.
|
2010-07-02 19:55:20 +08:00
|
|
|
return
|
|
|
|
QualType(new (*Context, TypeAlignment) InjectedClassNameType(D, TST), 0);
|
2010-03-10 11:28:59 +08:00
|
|
|
}
|
2010-06-20 03:28:53 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TYPE_TEMPLATE_TYPE_PARM: {
|
2010-06-20 03:29:09 +08:00
|
|
|
unsigned Idx = 0;
|
|
|
|
unsigned Depth = Record[Idx++];
|
|
|
|
unsigned Index = Record[Idx++];
|
|
|
|
bool Pack = Record[Idx++];
|
|
|
|
IdentifierInfo *Name = GetIdentifierInfo(Record, Idx);
|
|
|
|
return Context->getTemplateTypeParmType(Depth, Index, Pack, Name);
|
|
|
|
}
|
2010-06-24 16:57:31 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TYPE_DEPENDENT_NAME: {
|
2010-06-24 16:57:31 +08:00
|
|
|
unsigned Idx = 0;
|
|
|
|
ElaboratedTypeKeyword Keyword = (ElaboratedTypeKeyword)Record[Idx++];
|
|
|
|
NestedNameSpecifier *NNS = ReadNestedNameSpecifier(Record, Idx);
|
|
|
|
const IdentifierInfo *Name = this->GetIdentifierInfo(Record, Idx);
|
2010-07-02 19:55:24 +08:00
|
|
|
QualType Canon = GetType(Record[Idx++]);
|
|
|
|
return Context->getDependentNameType(Keyword, NNS, Name, Canon);
|
2010-06-24 16:57:31 +08:00
|
|
|
}
|
2010-06-26 00:24:58 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TYPE_DEPENDENT_TEMPLATE_SPECIALIZATION: {
|
2010-06-26 00:24:58 +08:00
|
|
|
unsigned Idx = 0;
|
|
|
|
ElaboratedTypeKeyword Keyword = (ElaboratedTypeKeyword)Record[Idx++];
|
|
|
|
NestedNameSpecifier *NNS = ReadNestedNameSpecifier(Record, Idx);
|
|
|
|
const IdentifierInfo *Name = this->GetIdentifierInfo(Record, Idx);
|
|
|
|
unsigned NumArgs = Record[Idx++];
|
|
|
|
llvm::SmallVector<TemplateArgument, 8> Args;
|
|
|
|
Args.reserve(NumArgs);
|
|
|
|
while (NumArgs--)
|
2010-07-23 06:43:28 +08:00
|
|
|
Args.push_back(ReadTemplateArgument(DeclsCursor, Record, Idx));
|
2010-06-26 00:24:58 +08:00
|
|
|
return Context->getDependentTemplateSpecializationType(Keyword, NNS, Name,
|
|
|
|
Args.size(), Args.data());
|
|
|
|
}
|
2010-06-30 16:49:25 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TYPE_DEPENDENT_SIZED_ARRAY: {
|
2010-06-30 16:49:25 +08:00
|
|
|
unsigned Idx = 0;
|
|
|
|
|
|
|
|
// ArrayType
|
|
|
|
QualType ElementType = GetType(Record[Idx++]);
|
|
|
|
ArrayType::ArraySizeModifier ASM
|
|
|
|
= (ArrayType::ArraySizeModifier)Record[Idx++];
|
|
|
|
unsigned IndexTypeQuals = Record[Idx++];
|
|
|
|
|
|
|
|
// DependentSizedArrayType
|
2010-07-23 06:43:28 +08:00
|
|
|
Expr *NumElts = ReadExpr(DeclsCursor);
|
2010-06-30 16:49:25 +08:00
|
|
|
SourceRange Brackets = ReadSourceRange(Record, Idx);
|
|
|
|
|
|
|
|
return Context->getDependentSizedArrayType(ElementType, NumElts, ASM,
|
|
|
|
IndexTypeQuals, Brackets);
|
|
|
|
}
|
2010-06-20 03:28:53 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TYPE_TEMPLATE_SPECIALIZATION: {
|
2010-06-20 03:29:09 +08:00
|
|
|
unsigned Idx = 0;
|
2010-07-08 21:09:53 +08:00
|
|
|
bool IsDependent = Record[Idx++];
|
2010-06-20 03:29:09 +08:00
|
|
|
TemplateName Name = ReadTemplateName(Record, Idx);
|
|
|
|
llvm::SmallVector<TemplateArgument, 8> Args;
|
2010-07-23 06:43:28 +08:00
|
|
|
ReadTemplateArgumentList(Args, DeclsCursor, Record, Idx);
|
2010-06-23 21:48:30 +08:00
|
|
|
QualType Canon = GetType(Record[Idx++]);
|
2010-07-08 21:09:53 +08:00
|
|
|
QualType T;
|
2010-07-02 19:55:11 +08:00
|
|
|
if (Canon.isNull())
|
2010-07-08 21:09:53 +08:00
|
|
|
T = Context->getCanonicalTemplateSpecializationType(Name, Args.data(),
|
|
|
|
Args.size());
|
2010-07-02 19:55:11 +08:00
|
|
|
else
|
2010-07-08 21:09:53 +08:00
|
|
|
T = Context->getTemplateSpecializationType(Name, Args.data(),
|
|
|
|
Args.size(), Canon);
|
|
|
|
T->Dependent = IsDependent;
|
|
|
|
return T;
|
2010-06-20 03:29:09 +08:00
|
|
|
}
|
2009-04-10 06:27:44 +08:00
|
|
|
}
|
|
|
|
// Suppress a GCC warning
|
|
|
|
return QualType();
|
|
|
|
}
|
|
|
|
|
2009-10-17 05:56:05 +08:00
|
|
|
namespace {
|
|
|
|
|
|
|
|
class TypeLocReader : public TypeLocVisitor<TypeLocReader> {
|
2010-08-19 07:56:43 +08:00
|
|
|
ASTReader &Reader;
|
2010-07-23 06:43:28 +08:00
|
|
|
llvm::BitstreamCursor &DeclsCursor;
|
2010-08-19 07:56:43 +08:00
|
|
|
const ASTReader::RecordData &Record;
|
2009-10-17 05:56:05 +08:00
|
|
|
unsigned &Idx;
|
|
|
|
|
|
|
|
public:
|
2010-08-19 07:56:43 +08:00
|
|
|
TypeLocReader(ASTReader &Reader, llvm::BitstreamCursor &Cursor,
|
|
|
|
const ASTReader::RecordData &Record, unsigned &Idx)
|
2010-07-23 06:43:28 +08:00
|
|
|
: Reader(Reader), DeclsCursor(Cursor), Record(Record), Idx(Idx) { }
|
2009-10-17 05:56:05 +08:00
|
|
|
|
2009-10-18 09:05:36 +08:00
|
|
|
// We want compile-time assurance that we've enumerated all of
|
|
|
|
// these, so unfortunately we have to declare them first, then
|
|
|
|
// define them out-of-line.
|
|
|
|
#define ABSTRACT_TYPELOC(CLASS, PARENT)
|
2009-10-17 05:56:05 +08:00
|
|
|
#define TYPELOC(CLASS, PARENT) \
|
2009-10-18 09:05:36 +08:00
|
|
|
void Visit##CLASS##TypeLoc(CLASS##TypeLoc TyLoc);
|
2009-10-17 05:56:05 +08:00
|
|
|
#include "clang/AST/TypeLocNodes.def"
|
|
|
|
|
2009-10-18 09:05:36 +08:00
|
|
|
void VisitFunctionTypeLoc(FunctionTypeLoc);
|
|
|
|
void VisitArrayTypeLoc(ArrayTypeLoc);
|
2009-10-17 05:56:05 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2009-10-18 09:05:36 +08:00
|
|
|
void TypeLocReader::VisitQualifiedTypeLoc(QualifiedTypeLoc TL) {
|
2009-10-17 05:56:05 +08:00
|
|
|
// nothing to do
|
|
|
|
}
|
2009-10-18 09:05:36 +08:00
|
|
|
void TypeLocReader::VisitBuiltinTypeLoc(BuiltinTypeLoc TL) {
|
2010-01-19 02:04:31 +08:00
|
|
|
TL.setBuiltinLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
|
|
|
if (TL.needsExtraLocalData()) {
|
|
|
|
TL.setWrittenTypeSpec(static_cast<DeclSpec::TST>(Record[Idx++]));
|
|
|
|
TL.setWrittenSignSpec(static_cast<DeclSpec::TSS>(Record[Idx++]));
|
|
|
|
TL.setWrittenWidthSpec(static_cast<DeclSpec::TSW>(Record[Idx++]));
|
|
|
|
TL.setModeAttr(Record[Idx++]);
|
|
|
|
}
|
2009-10-17 05:56:05 +08:00
|
|
|
}
|
2009-10-18 09:05:36 +08:00
|
|
|
void TypeLocReader::VisitComplexTypeLoc(ComplexTypeLoc TL) {
|
|
|
|
TL.setNameLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
2009-10-17 05:56:05 +08:00
|
|
|
}
|
2009-10-18 09:05:36 +08:00
|
|
|
void TypeLocReader::VisitPointerTypeLoc(PointerTypeLoc TL) {
|
|
|
|
TL.setStarLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
2009-10-17 05:56:05 +08:00
|
|
|
}
|
2009-10-18 09:05:36 +08:00
|
|
|
void TypeLocReader::VisitBlockPointerTypeLoc(BlockPointerTypeLoc TL) {
|
|
|
|
TL.setCaretLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
2009-10-17 05:56:05 +08:00
|
|
|
}
|
2009-10-18 09:05:36 +08:00
|
|
|
void TypeLocReader::VisitLValueReferenceTypeLoc(LValueReferenceTypeLoc TL) {
|
|
|
|
TL.setAmpLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
2009-10-17 05:56:05 +08:00
|
|
|
}
|
2009-10-18 09:05:36 +08:00
|
|
|
void TypeLocReader::VisitRValueReferenceTypeLoc(RValueReferenceTypeLoc TL) {
|
|
|
|
TL.setAmpAmpLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
2009-10-17 05:56:05 +08:00
|
|
|
}
|
2009-10-18 09:05:36 +08:00
|
|
|
void TypeLocReader::VisitMemberPointerTypeLoc(MemberPointerTypeLoc TL) {
|
|
|
|
TL.setStarLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
2009-10-17 05:56:05 +08:00
|
|
|
}
|
2009-10-18 09:05:36 +08:00
|
|
|
void TypeLocReader::VisitArrayTypeLoc(ArrayTypeLoc TL) {
|
|
|
|
TL.setLBracketLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
|
|
|
TL.setRBracketLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
2009-10-17 05:56:05 +08:00
|
|
|
if (Record[Idx++])
|
2010-07-23 06:43:28 +08:00
|
|
|
TL.setSizeExpr(Reader.ReadExpr(DeclsCursor));
|
2009-10-17 08:13:19 +08:00
|
|
|
else
|
2009-10-18 09:05:36 +08:00
|
|
|
TL.setSizeExpr(0);
|
|
|
|
}
|
|
|
|
void TypeLocReader::VisitConstantArrayTypeLoc(ConstantArrayTypeLoc TL) {
|
|
|
|
VisitArrayTypeLoc(TL);
|
|
|
|
}
|
|
|
|
void TypeLocReader::VisitIncompleteArrayTypeLoc(IncompleteArrayTypeLoc TL) {
|
|
|
|
VisitArrayTypeLoc(TL);
|
|
|
|
}
|
|
|
|
void TypeLocReader::VisitVariableArrayTypeLoc(VariableArrayTypeLoc TL) {
|
|
|
|
VisitArrayTypeLoc(TL);
|
|
|
|
}
|
|
|
|
void TypeLocReader::VisitDependentSizedArrayTypeLoc(
|
|
|
|
DependentSizedArrayTypeLoc TL) {
|
|
|
|
VisitArrayTypeLoc(TL);
|
|
|
|
}
|
|
|
|
void TypeLocReader::VisitDependentSizedExtVectorTypeLoc(
|
|
|
|
DependentSizedExtVectorTypeLoc TL) {
|
|
|
|
TL.setNameLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
|
|
|
}
|
|
|
|
void TypeLocReader::VisitVectorTypeLoc(VectorTypeLoc TL) {
|
|
|
|
TL.setNameLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
|
|
|
}
|
|
|
|
void TypeLocReader::VisitExtVectorTypeLoc(ExtVectorTypeLoc TL) {
|
|
|
|
TL.setNameLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
|
|
|
}
|
|
|
|
void TypeLocReader::VisitFunctionTypeLoc(FunctionTypeLoc TL) {
|
|
|
|
TL.setLParenLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
|
|
|
TL.setRParenLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
|
|
|
for (unsigned i = 0, e = TL.getNumArgs(); i != e; ++i) {
|
2009-10-23 09:28:53 +08:00
|
|
|
TL.setArg(i, cast_or_null<ParmVarDecl>(Reader.GetDecl(Record[Idx++])));
|
2009-10-18 09:05:36 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
void TypeLocReader::VisitFunctionProtoTypeLoc(FunctionProtoTypeLoc TL) {
|
|
|
|
VisitFunctionTypeLoc(TL);
|
|
|
|
}
|
|
|
|
void TypeLocReader::VisitFunctionNoProtoTypeLoc(FunctionNoProtoTypeLoc TL) {
|
|
|
|
VisitFunctionTypeLoc(TL);
|
|
|
|
}
|
2009-12-05 06:46:56 +08:00
|
|
|
void TypeLocReader::VisitUnresolvedUsingTypeLoc(UnresolvedUsingTypeLoc TL) {
|
|
|
|
TL.setNameLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
|
|
|
}
|
2009-10-18 09:05:36 +08:00
|
|
|
void TypeLocReader::VisitTypedefTypeLoc(TypedefTypeLoc TL) {
|
|
|
|
TL.setNameLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
|
|
|
}
|
|
|
|
void TypeLocReader::VisitTypeOfExprTypeLoc(TypeOfExprTypeLoc TL) {
|
2010-01-14 04:03:27 +08:00
|
|
|
TL.setTypeofLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
|
|
|
TL.setLParenLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
|
|
|
TL.setRParenLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
2009-10-18 09:05:36 +08:00
|
|
|
}
|
|
|
|
void TypeLocReader::VisitTypeOfTypeLoc(TypeOfTypeLoc TL) {
|
2010-01-14 04:03:27 +08:00
|
|
|
TL.setTypeofLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
|
|
|
TL.setLParenLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
|
|
|
TL.setRParenLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
2010-07-23 06:43:28 +08:00
|
|
|
TL.setUnderlyingTInfo(Reader.GetTypeSourceInfo(DeclsCursor, Record, Idx));
|
2009-10-18 09:05:36 +08:00
|
|
|
}
|
|
|
|
void TypeLocReader::VisitDecltypeTypeLoc(DecltypeTypeLoc TL) {
|
|
|
|
TL.setNameLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
|
|
|
}
|
|
|
|
void TypeLocReader::VisitRecordTypeLoc(RecordTypeLoc TL) {
|
|
|
|
TL.setNameLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
|
|
|
}
|
|
|
|
void TypeLocReader::VisitEnumTypeLoc(EnumTypeLoc TL) {
|
|
|
|
TL.setNameLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
|
|
|
}
|
|
|
|
void TypeLocReader::VisitTemplateTypeParmTypeLoc(TemplateTypeParmTypeLoc TL) {
|
|
|
|
TL.setNameLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
|
|
|
}
|
2009-10-18 17:09:24 +08:00
|
|
|
void TypeLocReader::VisitSubstTemplateTypeParmTypeLoc(
|
|
|
|
SubstTemplateTypeParmTypeLoc TL) {
|
|
|
|
TL.setNameLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
|
|
|
}
|
2009-10-18 09:05:36 +08:00
|
|
|
void TypeLocReader::VisitTemplateSpecializationTypeLoc(
|
|
|
|
TemplateSpecializationTypeLoc TL) {
|
2009-10-29 16:12:44 +08:00
|
|
|
TL.setTemplateNameLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
|
|
|
TL.setLAngleLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
|
|
|
TL.setRAngleLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
|
|
|
for (unsigned i = 0, e = TL.getNumArgs(); i != e; ++i)
|
|
|
|
TL.setArgLocInfo(i,
|
|
|
|
Reader.GetTemplateArgumentLocInfo(TL.getTypePtr()->getArg(i).getKind(),
|
2010-07-23 06:43:28 +08:00
|
|
|
DeclsCursor, Record, Idx));
|
2009-10-18 09:05:36 +08:00
|
|
|
}
|
2010-05-12 05:36:43 +08:00
|
|
|
void TypeLocReader::VisitElaboratedTypeLoc(ElaboratedTypeLoc TL) {
|
2010-05-20 05:37:53 +08:00
|
|
|
TL.setKeywordLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
|
|
|
TL.setQualifierRange(Reader.ReadSourceRange(Record, Idx));
|
2009-10-18 09:05:36 +08:00
|
|
|
}
|
2010-03-10 11:28:59 +08:00
|
|
|
void TypeLocReader::VisitInjectedClassNameTypeLoc(InjectedClassNameTypeLoc TL) {
|
|
|
|
TL.setNameLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
|
|
|
}
|
2010-04-01 01:34:00 +08:00
|
|
|
void TypeLocReader::VisitDependentNameTypeLoc(DependentNameTypeLoc TL) {
|
2010-05-20 05:37:53 +08:00
|
|
|
TL.setKeywordLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
|
|
|
TL.setQualifierRange(Reader.ReadSourceRange(Record, Idx));
|
2009-10-18 09:05:36 +08:00
|
|
|
TL.setNameLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
|
|
|
}
|
2010-06-11 08:33:02 +08:00
|
|
|
void TypeLocReader::VisitDependentTemplateSpecializationTypeLoc(
|
|
|
|
DependentTemplateSpecializationTypeLoc TL) {
|
|
|
|
TL.setKeywordLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
|
|
|
TL.setQualifierRange(Reader.ReadSourceRange(Record, Idx));
|
|
|
|
TL.setNameLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
|
|
|
TL.setLAngleLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
|
|
|
TL.setRAngleLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
|
|
|
for (unsigned I = 0, E = TL.getNumArgs(); I != E; ++I)
|
|
|
|
TL.setArgLocInfo(I,
|
|
|
|
Reader.GetTemplateArgumentLocInfo(TL.getTypePtr()->getArg(I).getKind(),
|
2010-07-23 06:43:28 +08:00
|
|
|
DeclsCursor, Record, Idx));
|
2010-06-11 08:33:02 +08:00
|
|
|
}
|
2009-10-18 09:05:36 +08:00
|
|
|
void TypeLocReader::VisitObjCInterfaceTypeLoc(ObjCInterfaceTypeLoc TL) {
|
|
|
|
TL.setNameLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
2010-05-15 19:32:37 +08:00
|
|
|
}
|
|
|
|
void TypeLocReader::VisitObjCObjectTypeLoc(ObjCObjectTypeLoc TL) {
|
|
|
|
TL.setHasBaseTypeAsWritten(Record[Idx++]);
|
2009-10-23 06:37:11 +08:00
|
|
|
TL.setLAngleLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
|
|
|
TL.setRAngleLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
|
|
|
for (unsigned i = 0, e = TL.getNumProtocols(); i != e; ++i)
|
|
|
|
TL.setProtocolLoc(i, SourceLocation::getFromRawEncoding(Record[Idx++]));
|
2009-10-18 09:05:36 +08:00
|
|
|
}
|
|
|
|
void TypeLocReader::VisitObjCObjectPointerTypeLoc(ObjCObjectPointerTypeLoc TL) {
|
|
|
|
TL.setStarLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
2009-10-17 05:56:05 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:43 +08:00
|
|
|
TypeSourceInfo *ASTReader::GetTypeSourceInfo(llvm::BitstreamCursor &DeclsCursor,
|
2010-07-23 06:43:28 +08:00
|
|
|
const RecordData &Record,
|
2009-10-17 05:56:05 +08:00
|
|
|
unsigned &Idx) {
|
|
|
|
QualType InfoTy = GetType(Record[Idx++]);
|
|
|
|
if (InfoTy.isNull())
|
|
|
|
return 0;
|
|
|
|
|
2009-12-07 10:54:59 +08:00
|
|
|
TypeSourceInfo *TInfo = getContext()->CreateTypeSourceInfo(InfoTy);
|
2010-07-23 06:43:28 +08:00
|
|
|
TypeLocReader TLR(*this, DeclsCursor, Record, Idx);
|
2009-12-07 10:54:59 +08:00
|
|
|
for (TypeLoc TL = TInfo->getTypeLoc(); !TL.isNull(); TL = TL.getNextTypeLoc())
|
2009-10-17 05:56:05 +08:00
|
|
|
TLR.Visit(TL);
|
2009-12-07 10:54:59 +08:00
|
|
|
return TInfo;
|
2009-10-17 05:56:05 +08:00
|
|
|
}
|
2009-04-10 06:27:44 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
QualType ASTReader::GetType(TypeID ID) {
|
2009-09-25 03:53:00 +08:00
|
|
|
unsigned FastQuals = ID & Qualifiers::FastMask;
|
|
|
|
unsigned Index = ID >> Qualifiers::FastWidth;
|
2009-04-10 06:27:44 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
if (Index < NUM_PREDEF_TYPE_IDS) {
|
2009-04-10 06:27:44 +08:00
|
|
|
QualType T;
|
2010-08-19 07:57:32 +08:00
|
|
|
switch ((PredefinedTypeIDs)Index) {
|
|
|
|
case PREDEF_TYPE_NULL_ID: return QualType();
|
|
|
|
case PREDEF_TYPE_VOID_ID: T = Context->VoidTy; break;
|
|
|
|
case PREDEF_TYPE_BOOL_ID: T = Context->BoolTy; break;
|
2009-04-10 06:27:44 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case PREDEF_TYPE_CHAR_U_ID:
|
|
|
|
case PREDEF_TYPE_CHAR_S_ID:
|
2009-04-10 06:27:44 +08:00
|
|
|
// FIXME: Check that the signedness of CharTy is correct!
|
2009-04-28 05:45:14 +08:00
|
|
|
T = Context->CharTy;
|
2009-04-10 06:27:44 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case PREDEF_TYPE_UCHAR_ID: T = Context->UnsignedCharTy; break;
|
|
|
|
case PREDEF_TYPE_USHORT_ID: T = Context->UnsignedShortTy; break;
|
|
|
|
case PREDEF_TYPE_UINT_ID: T = Context->UnsignedIntTy; break;
|
|
|
|
case PREDEF_TYPE_ULONG_ID: T = Context->UnsignedLongTy; break;
|
|
|
|
case PREDEF_TYPE_ULONGLONG_ID: T = Context->UnsignedLongLongTy; break;
|
|
|
|
case PREDEF_TYPE_UINT128_ID: T = Context->UnsignedInt128Ty; break;
|
|
|
|
case PREDEF_TYPE_SCHAR_ID: T = Context->SignedCharTy; break;
|
|
|
|
case PREDEF_TYPE_WCHAR_ID: T = Context->WCharTy; break;
|
|
|
|
case PREDEF_TYPE_SHORT_ID: T = Context->ShortTy; break;
|
|
|
|
case PREDEF_TYPE_INT_ID: T = Context->IntTy; break;
|
|
|
|
case PREDEF_TYPE_LONG_ID: T = Context->LongTy; break;
|
|
|
|
case PREDEF_TYPE_LONGLONG_ID: T = Context->LongLongTy; break;
|
|
|
|
case PREDEF_TYPE_INT128_ID: T = Context->Int128Ty; break;
|
|
|
|
case PREDEF_TYPE_FLOAT_ID: T = Context->FloatTy; break;
|
|
|
|
case PREDEF_TYPE_DOUBLE_ID: T = Context->DoubleTy; break;
|
|
|
|
case PREDEF_TYPE_LONGDOUBLE_ID: T = Context->LongDoubleTy; break;
|
|
|
|
case PREDEF_TYPE_OVERLOAD_ID: T = Context->OverloadTy; break;
|
|
|
|
case PREDEF_TYPE_DEPENDENT_ID: T = Context->DependentTy; break;
|
|
|
|
case PREDEF_TYPE_NULLPTR_ID: T = Context->NullPtrTy; break;
|
|
|
|
case PREDEF_TYPE_CHAR16_ID: T = Context->Char16Ty; break;
|
|
|
|
case PREDEF_TYPE_CHAR32_ID: T = Context->Char32Ty; break;
|
|
|
|
case PREDEF_TYPE_OBJC_ID: T = Context->ObjCBuiltinIdTy; break;
|
|
|
|
case PREDEF_TYPE_OBJC_CLASS: T = Context->ObjCBuiltinClassTy; break;
|
|
|
|
case PREDEF_TYPE_OBJC_SEL: T = Context->ObjCBuiltinSelTy; break;
|
2009-04-10 06:27:44 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
assert(!T.isNull() && "Unknown predefined type");
|
2009-09-25 03:53:00 +08:00
|
|
|
return T.withFastQualifiers(FastQuals);
|
2009-04-10 06:27:44 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
Index -= NUM_PREDEF_TYPE_IDS;
|
2010-07-21 06:37:49 +08:00
|
|
|
assert(Index < TypesLoaded.size() && "Type index out-of-range");
|
2010-07-15 04:26:45 +08:00
|
|
|
if (TypesLoaded[Index].isNull()) {
|
2010-07-21 06:37:49 +08:00
|
|
|
TypesLoaded[Index] = ReadTypeRecord(Index);
|
2010-08-19 07:57:06 +08:00
|
|
|
TypesLoaded[Index]->setFromAST();
|
2010-08-21 00:04:27 +08:00
|
|
|
TypeIdxs[TypesLoaded[Index]] = TypeIdx::fromTypeID(ID);
|
2010-07-15 07:45:08 +08:00
|
|
|
if (DeserializationListener)
|
2010-08-21 00:03:59 +08:00
|
|
|
DeserializationListener->TypeRead(TypeIdx::fromTypeID(ID),
|
2010-07-17 00:36:56 +08:00
|
|
|
TypesLoaded[Index]);
|
2010-07-15 04:26:45 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-09-25 03:53:00 +08:00
|
|
|
return TypesLoaded[Index].withFastQualifiers(FastQuals);
|
2009-04-10 06:27:44 +08:00
|
|
|
}
|
|
|
|
|
2010-08-21 00:04:27 +08:00
|
|
|
TypeID ASTReader::GetTypeID(QualType T) const {
|
|
|
|
return MakeTypeID(T,
|
|
|
|
std::bind1st(std::mem_fun(&ASTReader::GetTypeIdx), this));
|
|
|
|
}
|
|
|
|
|
|
|
|
TypeIdx ASTReader::GetTypeIdx(QualType T) const {
|
|
|
|
if (T.isNull())
|
|
|
|
return TypeIdx();
|
|
|
|
assert(!T.getLocalFastQualifiers());
|
|
|
|
|
|
|
|
TypeIdxMap::const_iterator I = TypeIdxs.find(T);
|
|
|
|
// GetTypeIdx is mostly used for computing the hash of DeclarationNames and
|
|
|
|
// comparing keys of ASTDeclContextNameLookupTable.
|
|
|
|
// If the type didn't come from the AST file use a specially marked index
|
|
|
|
// so that any hash/key comparison fail since no such index is stored
|
|
|
|
// in a AST file.
|
|
|
|
if (I == TypeIdxs.end())
|
|
|
|
return TypeIdx(-1);
|
|
|
|
return I->second;
|
|
|
|
}
|
|
|
|
|
2009-10-29 16:12:44 +08:00
|
|
|
TemplateArgumentLocInfo
|
2010-08-19 07:56:43 +08:00
|
|
|
ASTReader::GetTemplateArgumentLocInfo(TemplateArgument::ArgKind Kind,
|
2010-07-23 06:43:28 +08:00
|
|
|
llvm::BitstreamCursor &DeclsCursor,
|
2009-10-29 16:12:44 +08:00
|
|
|
const RecordData &Record,
|
2010-06-29 06:28:35 +08:00
|
|
|
unsigned &Index) {
|
2009-10-29 16:12:44 +08:00
|
|
|
switch (Kind) {
|
|
|
|
case TemplateArgument::Expression:
|
2010-07-23 06:43:28 +08:00
|
|
|
return ReadExpr(DeclsCursor);
|
2009-10-29 16:12:44 +08:00
|
|
|
case TemplateArgument::Type:
|
2010-07-23 06:43:28 +08:00
|
|
|
return GetTypeSourceInfo(DeclsCursor, Record, Index);
|
2009-11-11 09:00:40 +08:00
|
|
|
case TemplateArgument::Template: {
|
2010-06-28 17:31:42 +08:00
|
|
|
SourceRange QualifierRange = ReadSourceRange(Record, Index);
|
|
|
|
SourceLocation TemplateNameLoc = ReadSourceLocation(Record, Index);
|
|
|
|
return TemplateArgumentLocInfo(QualifierRange, TemplateNameLoc);
|
2009-11-11 09:00:40 +08:00
|
|
|
}
|
2009-10-29 16:12:44 +08:00
|
|
|
case TemplateArgument::Null:
|
|
|
|
case TemplateArgument::Integral:
|
|
|
|
case TemplateArgument::Declaration:
|
|
|
|
case TemplateArgument::Pack:
|
|
|
|
return TemplateArgumentLocInfo();
|
|
|
|
}
|
2009-12-12 13:05:38 +08:00
|
|
|
llvm_unreachable("unexpected template argument loc");
|
2009-10-29 16:12:44 +08:00
|
|
|
return TemplateArgumentLocInfo();
|
|
|
|
}
|
|
|
|
|
2010-06-22 17:54:59 +08:00
|
|
|
TemplateArgumentLoc
|
2010-08-19 07:56:43 +08:00
|
|
|
ASTReader::ReadTemplateArgumentLoc(llvm::BitstreamCursor &DeclsCursor,
|
2010-07-23 06:43:28 +08:00
|
|
|
const RecordData &Record, unsigned &Index) {
|
|
|
|
TemplateArgument Arg = ReadTemplateArgument(DeclsCursor, Record, Index);
|
2010-06-28 17:31:42 +08:00
|
|
|
|
|
|
|
if (Arg.getKind() == TemplateArgument::Expression) {
|
|
|
|
if (Record[Index++]) // bool InfoHasSameExpr.
|
|
|
|
return TemplateArgumentLoc(Arg, TemplateArgumentLocInfo(Arg.getAsExpr()));
|
|
|
|
}
|
2010-06-22 17:54:59 +08:00
|
|
|
return TemplateArgumentLoc(Arg, GetTemplateArgumentLocInfo(Arg.getKind(),
|
2010-07-23 06:43:28 +08:00
|
|
|
DeclsCursor,
|
2010-06-29 06:28:35 +08:00
|
|
|
Record, Index));
|
2010-06-22 17:54:59 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:43 +08:00
|
|
|
Decl *ASTReader::GetExternalDecl(uint32_t ID) {
|
2010-06-01 17:23:16 +08:00
|
|
|
return GetDecl(ID);
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:43 +08:00
|
|
|
TranslationUnitDecl *ASTReader::GetTranslationUnitDecl() {
|
2010-07-15 07:45:08 +08:00
|
|
|
if (!DeclsLoaded[0]) {
|
2010-08-24 08:50:04 +08:00
|
|
|
ReadDeclRecord(0, 1);
|
2010-07-15 07:45:08 +08:00
|
|
|
if (DeserializationListener)
|
2010-07-17 00:36:56 +08:00
|
|
|
DeserializationListener->DeclRead(1, DeclsLoaded[0]);
|
2010-07-15 07:45:08 +08:00
|
|
|
}
|
2010-07-09 01:13:02 +08:00
|
|
|
|
|
|
|
return cast<TranslationUnitDecl>(DeclsLoaded[0]);
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
Decl *ASTReader::GetDecl(DeclID ID) {
|
2009-04-10 06:27:44 +08:00
|
|
|
if (ID == 0)
|
|
|
|
return 0;
|
|
|
|
|
2009-04-26 02:35:21 +08:00
|
|
|
if (ID > DeclsLoaded.size()) {
|
2010-08-19 07:57:06 +08:00
|
|
|
Error("declaration ID out-of-range for AST file");
|
2009-04-26 02:35:21 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-04-10 06:27:44 +08:00
|
|
|
unsigned Index = ID - 1;
|
2010-07-15 07:45:08 +08:00
|
|
|
if (!DeclsLoaded[Index]) {
|
2010-08-04 01:30:10 +08:00
|
|
|
ReadDeclRecord(Index, ID);
|
2010-07-15 07:45:08 +08:00
|
|
|
if (DeserializationListener)
|
|
|
|
DeserializationListener->DeclRead(ID, DeclsLoaded[Index]);
|
|
|
|
}
|
2009-04-10 06:27:44 +08:00
|
|
|
|
2009-04-26 02:35:21 +08:00
|
|
|
return DeclsLoaded[Index];
|
2009-04-10 06:27:44 +08:00
|
|
|
}
|
|
|
|
|
2009-04-27 13:46:25 +08:00
|
|
|
/// \brief Resolve the offset of a statement into a statement.
|
|
|
|
///
|
|
|
|
/// This operation will read a new statement from the external
|
|
|
|
/// source each time it is called, and is meant to be used via a
|
|
|
|
/// LazyOffsetPtr (which is used by Decls for the body of functions, etc).
|
2010-08-19 07:56:43 +08:00
|
|
|
Stmt *ASTReader::GetExternalDeclStmt(uint64_t Offset) {
|
2010-07-23 01:01:13 +08:00
|
|
|
// Offset here is a global offset across the entire chain.
|
|
|
|
for (unsigned I = 0, N = Chain.size(); I != N; ++I) {
|
|
|
|
PerFileData &F = *Chain[N - I - 1];
|
|
|
|
if (Offset < F.SizeInBits) {
|
|
|
|
// Since we know that this statement is part of a decl, make sure to use
|
|
|
|
// the decl cursor to read it.
|
|
|
|
F.DeclsCursor.JumpToBit(Offset);
|
|
|
|
return ReadStmtFromStream(F.DeclsCursor);
|
|
|
|
}
|
|
|
|
Offset -= F.SizeInBits;
|
|
|
|
}
|
|
|
|
llvm_unreachable("Broken chain");
|
2009-04-18 08:07:54 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:43 +08:00
|
|
|
bool ASTReader::FindExternalLexicalDecls(const DeclContext *DC,
|
2010-06-01 17:23:16 +08:00
|
|
|
llvm::SmallVectorImpl<Decl*> &Decls) {
|
2009-09-09 23:08:12 +08:00
|
|
|
assert(DC->hasExternalLexicalStorage() &&
|
2009-04-10 06:27:44 +08:00
|
|
|
"DeclContext has no lexical decls in storage");
|
2010-03-18 08:56:54 +08:00
|
|
|
|
2010-07-23 01:01:13 +08:00
|
|
|
// There might be lexical decls in multiple parts of the chain, for the TU
|
|
|
|
// at least.
|
2010-09-28 10:24:44 +08:00
|
|
|
// DeclContextOffsets might reallocate as we load additional decls below,
|
|
|
|
// so make a copy of the vector.
|
|
|
|
DeclContextInfos Infos = DeclContextOffsets[DC];
|
2010-07-23 01:01:13 +08:00
|
|
|
for (DeclContextInfos::iterator I = Infos.begin(), E = Infos.end();
|
|
|
|
I != E; ++I) {
|
2010-07-27 08:17:23 +08:00
|
|
|
// IDs can be 0 if this context doesn't contain declarations.
|
|
|
|
if (!I->LexicalDecls)
|
2010-07-23 01:01:13 +08:00
|
|
|
continue;
|
2009-04-15 05:18:50 +08:00
|
|
|
|
2010-07-23 01:01:13 +08:00
|
|
|
// Load all of the declaration IDs
|
2010-08-19 07:57:32 +08:00
|
|
|
for (const DeclID *ID = I->LexicalDecls,
|
2010-07-27 08:17:23 +08:00
|
|
|
*IDE = ID + I->NumLexicalDecls;
|
2010-09-28 10:24:44 +08:00
|
|
|
ID != IDE; ++ID) {
|
|
|
|
Decl *D = GetDecl(*ID);
|
|
|
|
assert(D && "Null decl in lexical decls");
|
|
|
|
Decls.push_back(D);
|
|
|
|
}
|
2010-03-18 08:56:54 +08:00
|
|
|
}
|
2009-04-10 06:27:44 +08:00
|
|
|
|
2009-04-23 06:34:57 +08:00
|
|
|
++NumLexicalDeclContextsRead;
|
2009-04-10 06:27:44 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2010-06-01 17:23:16 +08:00
|
|
|
DeclContext::lookup_result
|
2010-08-19 07:56:43 +08:00
|
|
|
ASTReader::FindExternalVisibleDeclsByName(const DeclContext *DC,
|
2010-06-01 17:23:16 +08:00
|
|
|
DeclarationName Name) {
|
2009-09-09 23:08:12 +08:00
|
|
|
assert(DC->hasExternalVisibleStorage() &&
|
2009-04-10 06:27:44 +08:00
|
|
|
"DeclContext has no visible decls in storage");
|
2010-08-21 00:04:35 +08:00
|
|
|
if (!Name)
|
|
|
|
return DeclContext::lookup_result(DeclContext::lookup_iterator(0),
|
|
|
|
DeclContext::lookup_iterator(0));
|
2009-04-10 06:27:44 +08:00
|
|
|
|
2010-08-21 00:04:35 +08:00
|
|
|
llvm::SmallVector<NamedDecl *, 64> Decls;
|
2010-08-24 08:49:55 +08:00
|
|
|
// There might be visible decls in multiple parts of the chain, for the TU
|
2010-08-24 08:50:16 +08:00
|
|
|
// and namespaces. For any given name, the last available results replace
|
|
|
|
// all earlier ones. For this reason, we walk in reverse.
|
2010-07-23 01:01:13 +08:00
|
|
|
DeclContextInfos &Infos = DeclContextOffsets[DC];
|
2010-08-24 08:50:16 +08:00
|
|
|
for (DeclContextInfos::reverse_iterator I = Infos.rbegin(), E = Infos.rend();
|
2010-07-23 01:01:13 +08:00
|
|
|
I != E; ++I) {
|
2010-08-21 00:04:35 +08:00
|
|
|
if (!I->NameLookupTableData)
|
2010-07-23 01:01:13 +08:00
|
|
|
continue;
|
|
|
|
|
2010-08-21 00:04:35 +08:00
|
|
|
ASTDeclContextNameLookupTable *LookupTable =
|
|
|
|
(ASTDeclContextNameLookupTable*)I->NameLookupTableData;
|
|
|
|
ASTDeclContextNameLookupTable::iterator Pos = LookupTable->find(Name);
|
|
|
|
if (Pos == LookupTable->end())
|
2010-07-23 01:01:13 +08:00
|
|
|
continue;
|
2009-04-10 06:27:44 +08:00
|
|
|
|
2010-08-21 00:04:35 +08:00
|
|
|
ASTDeclContextNameLookupTrait::data_type Data = *Pos;
|
|
|
|
for (; Data.first != Data.second; ++Data.first)
|
|
|
|
Decls.push_back(cast<NamedDecl>(GetDecl(*Data.first)));
|
2010-08-24 08:50:16 +08:00
|
|
|
break;
|
2009-04-10 06:27:44 +08:00
|
|
|
}
|
|
|
|
|
2009-04-23 06:34:57 +08:00
|
|
|
++NumVisibleDeclContextsRead;
|
2010-06-01 17:23:16 +08:00
|
|
|
|
2010-08-21 00:04:35 +08:00
|
|
|
SetExternalVisibleDeclsForName(DC, Name, Decls);
|
2010-06-01 17:23:16 +08:00
|
|
|
return const_cast<DeclContext*>(DC)->lookup(Name);
|
2009-04-10 06:27:44 +08:00
|
|
|
}
|
|
|
|
|
2010-08-21 07:35:55 +08:00
|
|
|
void ASTReader::MaterializeVisibleDecls(const DeclContext *DC) {
|
|
|
|
assert(DC->hasExternalVisibleStorage() &&
|
|
|
|
"DeclContext has no visible decls in storage");
|
|
|
|
|
|
|
|
llvm::SmallVector<NamedDecl *, 64> Decls;
|
|
|
|
// There might be visible decls in multiple parts of the chain, for the TU
|
|
|
|
// and namespaces.
|
|
|
|
DeclContextInfos &Infos = DeclContextOffsets[DC];
|
|
|
|
for (DeclContextInfos::iterator I = Infos.begin(), E = Infos.end();
|
|
|
|
I != E; ++I) {
|
|
|
|
if (!I->NameLookupTableData)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
ASTDeclContextNameLookupTable *LookupTable =
|
|
|
|
(ASTDeclContextNameLookupTable*)I->NameLookupTableData;
|
|
|
|
for (ASTDeclContextNameLookupTable::item_iterator
|
|
|
|
ItemI = LookupTable->item_begin(),
|
|
|
|
ItemEnd = LookupTable->item_end() ; ItemI != ItemEnd; ++ItemI) {
|
|
|
|
ASTDeclContextNameLookupTable::item_iterator::value_type Val
|
|
|
|
= *ItemI;
|
|
|
|
ASTDeclContextNameLookupTrait::data_type Data = Val.second;
|
|
|
|
Decls.clear();
|
|
|
|
for (; Data.first != Data.second; ++Data.first)
|
|
|
|
Decls.push_back(cast<NamedDecl>(GetDecl(*Data.first)));
|
|
|
|
MaterializeVisibleDeclsForName(DC, Val.first, Decls);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:43 +08:00
|
|
|
void ASTReader::PassInterestingDeclsToConsumer() {
|
2010-07-07 23:46:26 +08:00
|
|
|
assert(Consumer);
|
|
|
|
while (!InterestingDecls.empty()) {
|
|
|
|
DeclGroupRef DG(InterestingDecls.front());
|
|
|
|
InterestingDecls.pop_front();
|
2010-08-12 02:52:41 +08:00
|
|
|
Consumer->HandleInterestingDecl(DG);
|
2010-07-07 23:46:26 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:43 +08:00
|
|
|
void ASTReader::StartTranslationUnit(ASTConsumer *Consumer) {
|
2009-04-23 03:09:20 +08:00
|
|
|
this->Consumer = Consumer;
|
|
|
|
|
2009-04-14 08:24:19 +08:00
|
|
|
if (!Consumer)
|
|
|
|
return;
|
|
|
|
|
|
|
|
for (unsigned I = 0, N = ExternalDefinitions.size(); I != N; ++I) {
|
2010-07-07 23:46:26 +08:00
|
|
|
// Force deserialization of this decl, which will cause it to be queued for
|
|
|
|
// passing to the consumer.
|
2009-09-17 11:06:44 +08:00
|
|
|
GetDecl(ExternalDefinitions[I]);
|
2009-04-14 08:24:19 +08:00
|
|
|
}
|
2009-04-25 08:41:30 +08:00
|
|
|
|
2010-07-07 23:46:26 +08:00
|
|
|
PassInterestingDeclsToConsumer();
|
2009-04-14 08:24:19 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:43 +08:00
|
|
|
void ASTReader::PrintStats() {
|
2010-08-19 07:57:06 +08:00
|
|
|
std::fprintf(stderr, "*** AST File Statistics:\n");
|
2009-04-10 06:27:44 +08:00
|
|
|
|
2009-09-09 23:08:12 +08:00
|
|
|
unsigned NumTypesLoaded
|
2009-04-26 03:10:14 +08:00
|
|
|
= TypesLoaded.size() - std::count(TypesLoaded.begin(), TypesLoaded.end(),
|
2009-09-25 03:53:00 +08:00
|
|
|
QualType());
|
2009-04-26 03:10:14 +08:00
|
|
|
unsigned NumDeclsLoaded
|
|
|
|
= DeclsLoaded.size() - std::count(DeclsLoaded.begin(), DeclsLoaded.end(),
|
|
|
|
(Decl *)0);
|
|
|
|
unsigned NumIdentifiersLoaded
|
|
|
|
= IdentifiersLoaded.size() - std::count(IdentifiersLoaded.begin(),
|
|
|
|
IdentifiersLoaded.end(),
|
|
|
|
(IdentifierInfo *)0);
|
2009-09-09 23:08:12 +08:00
|
|
|
unsigned NumSelectorsLoaded
|
2009-04-26 03:10:14 +08:00
|
|
|
= SelectorsLoaded.size() - std::count(SelectorsLoaded.begin(),
|
|
|
|
SelectorsLoaded.end(),
|
|
|
|
Selector());
|
2009-04-26 01:48:32 +08:00
|
|
|
|
2009-04-28 02:38:38 +08:00
|
|
|
std::fprintf(stderr, " %u stat cache hits\n", NumStatHits);
|
|
|
|
std::fprintf(stderr, " %u stat cache misses\n", NumStatMisses);
|
2009-04-27 14:38:32 +08:00
|
|
|
if (TotalNumSLocEntries)
|
|
|
|
std::fprintf(stderr, " %u/%u source location entries read (%f%%)\n",
|
|
|
|
NumSLocEntriesRead, TotalNumSLocEntries,
|
|
|
|
((float)NumSLocEntriesRead/TotalNumSLocEntries * 100));
|
2009-04-26 02:35:21 +08:00
|
|
|
if (!TypesLoaded.empty())
|
2009-04-26 01:48:32 +08:00
|
|
|
std::fprintf(stderr, " %u/%u types read (%f%%)\n",
|
2009-04-26 02:35:21 +08:00
|
|
|
NumTypesLoaded, (unsigned)TypesLoaded.size(),
|
|
|
|
((float)NumTypesLoaded/TypesLoaded.size() * 100));
|
|
|
|
if (!DeclsLoaded.empty())
|
2009-04-26 01:48:32 +08:00
|
|
|
std::fprintf(stderr, " %u/%u declarations read (%f%%)\n",
|
2009-04-26 02:35:21 +08:00
|
|
|
NumDeclsLoaded, (unsigned)DeclsLoaded.size(),
|
|
|
|
((float)NumDeclsLoaded/DeclsLoaded.size() * 100));
|
2009-04-26 03:10:14 +08:00
|
|
|
if (!IdentifiersLoaded.empty())
|
2009-04-26 01:48:32 +08:00
|
|
|
std::fprintf(stderr, " %u/%u identifiers read (%f%%)\n",
|
2009-04-26 03:10:14 +08:00
|
|
|
NumIdentifiersLoaded, (unsigned)IdentifiersLoaded.size(),
|
|
|
|
((float)NumIdentifiersLoaded/IdentifiersLoaded.size() * 100));
|
2010-08-05 04:40:17 +08:00
|
|
|
if (!SelectorsLoaded.empty())
|
2009-04-26 01:48:32 +08:00
|
|
|
std::fprintf(stderr, " %u/%u selectors read (%f%%)\n",
|
2010-08-05 04:40:17 +08:00
|
|
|
NumSelectorsLoaded, (unsigned)SelectorsLoaded.size(),
|
|
|
|
((float)NumSelectorsLoaded/SelectorsLoaded.size() * 100));
|
2009-04-26 01:48:32 +08:00
|
|
|
if (TotalNumStatements)
|
|
|
|
std::fprintf(stderr, " %u/%u statements read (%f%%)\n",
|
|
|
|
NumStatementsRead, TotalNumStatements,
|
|
|
|
((float)NumStatementsRead/TotalNumStatements * 100));
|
|
|
|
if (TotalNumMacros)
|
|
|
|
std::fprintf(stderr, " %u/%u macros read (%f%%)\n",
|
|
|
|
NumMacrosRead, TotalNumMacros,
|
|
|
|
((float)NumMacrosRead/TotalNumMacros * 100));
|
|
|
|
if (TotalLexicalDeclContexts)
|
|
|
|
std::fprintf(stderr, " %u/%u lexical declcontexts read (%f%%)\n",
|
|
|
|
NumLexicalDeclContextsRead, TotalLexicalDeclContexts,
|
|
|
|
((float)NumLexicalDeclContextsRead/TotalLexicalDeclContexts
|
|
|
|
* 100));
|
|
|
|
if (TotalVisibleDeclContexts)
|
|
|
|
std::fprintf(stderr, " %u/%u visible declcontexts read (%f%%)\n",
|
|
|
|
NumVisibleDeclContextsRead, TotalVisibleDeclContexts,
|
|
|
|
((float)NumVisibleDeclContextsRead/TotalVisibleDeclContexts
|
|
|
|
* 100));
|
2010-08-05 05:22:45 +08:00
|
|
|
if (TotalNumMethodPoolEntries) {
|
2009-04-26 01:48:32 +08:00
|
|
|
std::fprintf(stderr, " %u/%u method pool entries read (%f%%)\n",
|
2010-08-05 05:22:45 +08:00
|
|
|
NumMethodPoolEntriesRead, TotalNumMethodPoolEntries,
|
|
|
|
((float)NumMethodPoolEntriesRead/TotalNumMethodPoolEntries
|
2009-04-26 01:48:32 +08:00
|
|
|
* 100));
|
2010-08-05 05:22:45 +08:00
|
|
|
std::fprintf(stderr, " %u method pool misses\n", NumMethodPoolMisses);
|
2009-04-26 01:48:32 +08:00
|
|
|
}
|
2009-04-10 06:27:44 +08:00
|
|
|
std::fprintf(stderr, "\n");
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:43 +08:00
|
|
|
void ASTReader::InitializeSema(Sema &S) {
|
Lazy deserialization of the declaration chains associated with
identifiers from a precompiled header.
This patch changes the primary name lookup method for entities within
a precompiled header. Previously, we would load all of the names of
declarations at translation unit scope into a large DenseMap (inside
the TranslationUnitDecl's DeclContext), and then perform a special
"last resort" lookup into this DeclContext when we knew there was a
PCH file (see Sema::LookupName). Now, when we see an identifier named
for the first time, we load all of the declarations with that name
that are visible from the translation unit into the IdentifierInfo's
chain of declarations. Thus, the explicit "look into the translation
unit's DeclContext" code is gone, and Sema effectively uses the same
IdentifierInfo-based name lookup mechanism whether we are using a PCH
file or not.
This approach should help PCH scale with the size of the input program
rather than the size of the PCH file. The "Hello, World!" application
with Carbon.h as a PCH file now loads 20% of the identifiers in the
PCH file rather than 85% of the identifiers.
90% of the 20% of identifiers loaded are actually loaded when we
deserialize the preprocessor state. The next step is to make the
preprocessor load macros lazily, which should drastically reduce the
number of types, declarations, and identifiers loaded for "Hello,
World".
llvm-svn: 69737
2009-04-22 06:25:48 +08:00
|
|
|
SemaObj = &S;
|
2009-04-25 05:10:55 +08:00
|
|
|
S.ExternalSource = this;
|
|
|
|
|
2009-04-23 05:15:06 +08:00
|
|
|
// Makes sure any declarations that were deserialized "too early"
|
|
|
|
// still get added to the identifier's declaration chains.
|
2010-09-25 07:29:12 +08:00
|
|
|
for (unsigned I = 0, N = PreloadedDecls.size(); I != N; ++I) {
|
|
|
|
if (SemaObj->TUScope)
|
2010-08-21 17:40:31 +08:00
|
|
|
SemaObj->TUScope->AddDecl(PreloadedDecls[I]);
|
2010-09-25 07:29:12 +08:00
|
|
|
|
|
|
|
SemaObj->IdResolver.AddDecl(PreloadedDecls[I]);
|
2009-04-23 05:15:06 +08:00
|
|
|
}
|
|
|
|
PreloadedDecls.clear();
|
2009-04-23 06:02:47 +08:00
|
|
|
|
|
|
|
// If there were any tentative definitions, deserialize them and add
|
2010-02-01 06:27:38 +08:00
|
|
|
// them to Sema's list of tentative definitions.
|
2009-04-23 06:02:47 +08:00
|
|
|
for (unsigned I = 0, N = TentativeDefinitions.size(); I != N; ++I) {
|
|
|
|
VarDecl *Var = cast<VarDecl>(GetDecl(TentativeDefinitions[I]));
|
2010-02-01 06:27:38 +08:00
|
|
|
SemaObj->TentativeDefinitions.push_back(Var);
|
2009-04-23 06:02:47 +08:00
|
|
|
}
|
2010-03-08 03:10:13 +08:00
|
|
|
|
2010-08-14 02:42:17 +08:00
|
|
|
// If there were any unused file scoped decls, deserialize them and add to
|
|
|
|
// Sema's list of unused file scoped decls.
|
|
|
|
for (unsigned I = 0, N = UnusedFileScopedDecls.size(); I != N; ++I) {
|
|
|
|
DeclaratorDecl *D = cast<DeclaratorDecl>(GetDecl(UnusedFileScopedDecls[I]));
|
|
|
|
SemaObj->UnusedFileScopedDecls.push_back(D);
|
2010-02-12 08:07:30 +08:00
|
|
|
}
|
2009-04-23 06:18:58 +08:00
|
|
|
|
2010-08-05 17:48:08 +08:00
|
|
|
// If there were any weak undeclared identifiers, deserialize them and add to
|
|
|
|
// Sema's list of weak undeclared identifiers.
|
|
|
|
if (!WeakUndeclaredIdentifiers.empty()) {
|
|
|
|
unsigned Idx = 0;
|
|
|
|
for (unsigned I = 0, N = WeakUndeclaredIdentifiers[Idx++]; I != N; ++I) {
|
|
|
|
IdentifierInfo *WeakId = GetIdentifierInfo(WeakUndeclaredIdentifiers,Idx);
|
|
|
|
IdentifierInfo *AliasId=GetIdentifierInfo(WeakUndeclaredIdentifiers,Idx);
|
|
|
|
SourceLocation Loc = ReadSourceLocation(WeakUndeclaredIdentifiers, Idx);
|
|
|
|
bool Used = WeakUndeclaredIdentifiers[Idx++];
|
|
|
|
Sema::WeakInfo WI(AliasId, Loc);
|
|
|
|
WI.setUsed(Used);
|
|
|
|
SemaObj->WeakUndeclaredIdentifiers.insert(std::make_pair(WeakId, WI));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-04-23 06:18:58 +08:00
|
|
|
// If there were any locally-scoped external declarations,
|
|
|
|
// deserialize them and add them to Sema's table of locally-scoped
|
|
|
|
// external declarations.
|
|
|
|
for (unsigned I = 0, N = LocallyScopedExternalDecls.size(); I != N; ++I) {
|
|
|
|
NamedDecl *D = cast<NamedDecl>(GetDecl(LocallyScopedExternalDecls[I]));
|
|
|
|
SemaObj->LocallyScopedExternalDecls[D->getDeclName()] = D;
|
|
|
|
}
|
2009-04-28 04:06:05 +08:00
|
|
|
|
|
|
|
// If there were any ext_vector type declarations, deserialize them
|
|
|
|
// and add them to Sema's vector of such declarations.
|
|
|
|
for (unsigned I = 0, N = ExtVectorDecls.size(); I != N; ++I)
|
|
|
|
SemaObj->ExtVectorDecls.push_back(
|
|
|
|
cast<TypedefDecl>(GetDecl(ExtVectorDecls[I])));
|
2010-07-06 23:37:04 +08:00
|
|
|
|
|
|
|
// FIXME: Do VTable uses and dynamic classes deserialize too much ?
|
|
|
|
// Can we cut them down before writing them ?
|
|
|
|
|
|
|
|
// If there were any VTable uses, deserialize the information and add it
|
|
|
|
// to Sema's vector and map of VTable uses.
|
2010-08-04 01:29:52 +08:00
|
|
|
if (!VTableUses.empty()) {
|
|
|
|
unsigned Idx = 0;
|
|
|
|
for (unsigned I = 0, N = VTableUses[Idx++]; I != N; ++I) {
|
|
|
|
CXXRecordDecl *Class = cast<CXXRecordDecl>(GetDecl(VTableUses[Idx++]));
|
|
|
|
SourceLocation Loc = ReadSourceLocation(VTableUses, Idx);
|
|
|
|
bool DefinitionRequired = VTableUses[Idx++];
|
|
|
|
SemaObj->VTableUses.push_back(std::make_pair(Class, Loc));
|
|
|
|
SemaObj->VTablesUsed[Class] = DefinitionRequired;
|
|
|
|
}
|
2010-07-06 23:37:04 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// If there were any dynamic classes declarations, deserialize them
|
|
|
|
// and add them to Sema's vector of such declarations.
|
|
|
|
for (unsigned I = 0, N = DynamicClasses.size(); I != N; ++I)
|
|
|
|
SemaObj->DynamicClasses.push_back(
|
|
|
|
cast<CXXRecordDecl>(GetDecl(DynamicClasses[I])));
|
2010-07-24 03:11:11 +08:00
|
|
|
|
2010-08-05 17:48:16 +08:00
|
|
|
// If there were any pending implicit instantiations, deserialize them
|
|
|
|
// and add them to Sema's queue of such instantiations.
|
2010-08-25 16:44:16 +08:00
|
|
|
assert(PendingInstantiations.size() % 2 == 0 && "Expected pairs of entries");
|
|
|
|
for (unsigned Idx = 0, N = PendingInstantiations.size(); Idx < N;) {
|
|
|
|
ValueDecl *D=cast<ValueDecl>(GetDecl(PendingInstantiations[Idx++]));
|
|
|
|
SourceLocation Loc = ReadSourceLocation(PendingInstantiations, Idx);
|
|
|
|
SemaObj->PendingInstantiations.push_back(std::make_pair(D, Loc));
|
2010-08-05 17:48:16 +08:00
|
|
|
}
|
|
|
|
|
2010-08-02 15:14:54 +08:00
|
|
|
// Load the offsets of the declarations that Sema references.
|
|
|
|
// They will be lazily deserialized when needed.
|
|
|
|
if (!SemaDeclRefs.empty()) {
|
|
|
|
assert(SemaDeclRefs.size() == 2 && "More decl refs than expected!");
|
|
|
|
SemaObj->StdNamespace = SemaDeclRefs[0];
|
|
|
|
SemaObj->StdBadAlloc = SemaDeclRefs[1];
|
|
|
|
}
|
|
|
|
|
2010-07-24 03:11:11 +08:00
|
|
|
// If there are @selector references added them to its pool. This is for
|
|
|
|
// implementation of -Wselector.
|
2010-08-05 04:40:17 +08:00
|
|
|
if (!ReferencedSelectorsData.empty()) {
|
|
|
|
unsigned int DataSize = ReferencedSelectorsData.size()-1;
|
2010-07-24 03:11:11 +08:00
|
|
|
unsigned I = 0;
|
|
|
|
while (I < DataSize) {
|
2010-08-05 04:40:17 +08:00
|
|
|
Selector Sel = DecodeSelector(ReferencedSelectorsData[I++]);
|
2010-07-24 03:11:11 +08:00
|
|
|
SourceLocation SelLoc =
|
2010-08-05 04:40:17 +08:00
|
|
|
SourceLocation::getFromRawEncoding(ReferencedSelectorsData[I++]);
|
2010-07-24 03:11:11 +08:00
|
|
|
SemaObj->ReferencedSelectors.insert(std::make_pair(Sel, SelLoc));
|
|
|
|
}
|
|
|
|
}
|
Lazy deserialization of the declaration chains associated with
identifiers from a precompiled header.
This patch changes the primary name lookup method for entities within
a precompiled header. Previously, we would load all of the names of
declarations at translation unit scope into a large DenseMap (inside
the TranslationUnitDecl's DeclContext), and then perform a special
"last resort" lookup into this DeclContext when we knew there was a
PCH file (see Sema::LookupName). Now, when we see an identifier named
for the first time, we load all of the declarations with that name
that are visible from the translation unit into the IdentifierInfo's
chain of declarations. Thus, the explicit "look into the translation
unit's DeclContext" code is gone, and Sema effectively uses the same
IdentifierInfo-based name lookup mechanism whether we are using a PCH
file or not.
This approach should help PCH scale with the size of the input program
rather than the size of the PCH file. The "Hello, World!" application
with Carbon.h as a PCH file now loads 20% of the identifiers in the
PCH file rather than 85% of the identifiers.
90% of the 20% of identifiers loaded are actually loaded when we
deserialize the preprocessor state. The next step is to make the
preprocessor load macros lazily, which should drastically reduce the
number of types, declarations, and identifiers loaded for "Hello,
World".
llvm-svn: 69737
2009-04-22 06:25:48 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:43 +08:00
|
|
|
IdentifierInfo* ASTReader::get(const char *NameStart, const char *NameEnd) {
|
2010-08-03 02:30:12 +08:00
|
|
|
// Try to find this name within our on-disk hash tables. We start with the
|
|
|
|
// most recent one, since that one contains the most up-to-date info.
|
2010-07-22 06:31:37 +08:00
|
|
|
for (unsigned I = 0, N = Chain.size(); I != N; ++I) {
|
2010-08-19 07:57:06 +08:00
|
|
|
ASTIdentifierLookupTable *IdTable
|
|
|
|
= (ASTIdentifierLookupTable *)Chain[I]->IdentifierLookupTable;
|
2010-07-23 01:01:13 +08:00
|
|
|
if (!IdTable)
|
|
|
|
continue;
|
2010-07-22 06:31:37 +08:00
|
|
|
std::pair<const char*, unsigned> Key(NameStart, NameEnd - NameStart);
|
2010-08-19 07:57:06 +08:00
|
|
|
ASTIdentifierLookupTable::iterator Pos = IdTable->find(Key);
|
2010-07-22 06:31:37 +08:00
|
|
|
if (Pos == IdTable->end())
|
|
|
|
continue;
|
Lazy deserialization of the declaration chains associated with
identifiers from a precompiled header.
This patch changes the primary name lookup method for entities within
a precompiled header. Previously, we would load all of the names of
declarations at translation unit scope into a large DenseMap (inside
the TranslationUnitDecl's DeclContext), and then perform a special
"last resort" lookup into this DeclContext when we knew there was a
PCH file (see Sema::LookupName). Now, when we see an identifier named
for the first time, we load all of the declarations with that name
that are visible from the translation unit into the IdentifierInfo's
chain of declarations. Thus, the explicit "look into the translation
unit's DeclContext" code is gone, and Sema effectively uses the same
IdentifierInfo-based name lookup mechanism whether we are using a PCH
file or not.
This approach should help PCH scale with the size of the input program
rather than the size of the PCH file. The "Hello, World!" application
with Carbon.h as a PCH file now loads 20% of the identifiers in the
PCH file rather than 85% of the identifiers.
90% of the 20% of identifiers loaded are actually loaded when we
deserialize the preprocessor state. The next step is to make the
preprocessor load macros lazily, which should drastically reduce the
number of types, declarations, and identifiers loaded for "Hello,
World".
llvm-svn: 69737
2009-04-22 06:25:48 +08:00
|
|
|
|
2010-07-22 06:31:37 +08:00
|
|
|
// Dereferencing the iterator has the effect of building the
|
|
|
|
// IdentifierInfo node and populating it with the various
|
|
|
|
// declarations it needs.
|
2010-08-03 02:30:12 +08:00
|
|
|
return *Pos;
|
2010-07-22 06:31:37 +08:00
|
|
|
}
|
2010-08-03 02:30:12 +08:00
|
|
|
return 0;
|
Lazy deserialization of the declaration chains associated with
identifiers from a precompiled header.
This patch changes the primary name lookup method for entities within
a precompiled header. Previously, we would load all of the names of
declarations at translation unit scope into a large DenseMap (inside
the TranslationUnitDecl's DeclContext), and then perform a special
"last resort" lookup into this DeclContext when we knew there was a
PCH file (see Sema::LookupName). Now, when we see an identifier named
for the first time, we load all of the declarations with that name
that are visible from the translation unit into the IdentifierInfo's
chain of declarations. Thus, the explicit "look into the translation
unit's DeclContext" code is gone, and Sema effectively uses the same
IdentifierInfo-based name lookup mechanism whether we are using a PCH
file or not.
This approach should help PCH scale with the size of the input program
rather than the size of the PCH file. The "Hello, World!" application
with Carbon.h as a PCH file now loads 20% of the identifiers in the
PCH file rather than 85% of the identifiers.
90% of the 20% of identifiers loaded are actually loaded when we
deserialize the preprocessor state. The next step is to make the
preprocessor load macros lazily, which should drastically reduce the
number of types, declarations, and identifiers loaded for "Hello,
World".
llvm-svn: 69737
2009-04-22 06:25:48 +08:00
|
|
|
}
|
|
|
|
|
2009-09-09 23:08:12 +08:00
|
|
|
std::pair<ObjCMethodList, ObjCMethodList>
|
2010-08-19 07:56:43 +08:00
|
|
|
ASTReader::ReadMethodPool(Selector Sel) {
|
2010-08-05 04:40:17 +08:00
|
|
|
// Find this selector in a hash table. We want to find the most recent entry.
|
|
|
|
for (unsigned I = 0, N = Chain.size(); I != N; ++I) {
|
|
|
|
PerFileData &F = *Chain[I];
|
|
|
|
if (!F.SelectorLookupTable)
|
|
|
|
continue;
|
2009-04-25 05:10:55 +08:00
|
|
|
|
2010-08-19 07:57:06 +08:00
|
|
|
ASTSelectorLookupTable *PoolTable
|
|
|
|
= (ASTSelectorLookupTable*)F.SelectorLookupTable;
|
|
|
|
ASTSelectorLookupTable::iterator Pos = PoolTable->find(Sel);
|
2010-08-05 04:40:17 +08:00
|
|
|
if (Pos != PoolTable->end()) {
|
|
|
|
++NumSelectorsRead;
|
2010-08-05 05:22:45 +08:00
|
|
|
// FIXME: Not quite happy with the statistics here. We probably should
|
|
|
|
// disable this tracking when called via LoadSelector.
|
|
|
|
// Also, should entries without methods count as misses?
|
|
|
|
++NumMethodPoolEntriesRead;
|
2010-08-19 07:57:06 +08:00
|
|
|
ASTSelectorLookupTrait::data_type Data = *Pos;
|
2010-08-05 04:40:17 +08:00
|
|
|
if (DeserializationListener)
|
|
|
|
DeserializationListener->SelectorRead(Data.ID, Sel);
|
|
|
|
return std::make_pair(Data.Instance, Data.Factory);
|
|
|
|
}
|
2009-04-26 01:48:32 +08:00
|
|
|
}
|
2009-04-25 05:10:55 +08:00
|
|
|
|
2010-08-05 05:22:45 +08:00
|
|
|
++NumMethodPoolMisses;
|
2010-08-05 04:40:17 +08:00
|
|
|
return std::pair<ObjCMethodList, ObjCMethodList>();
|
2009-04-25 05:10:55 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:43 +08:00
|
|
|
void ASTReader::LoadSelector(Selector Sel) {
|
2010-08-05 02:21:41 +08:00
|
|
|
// It would be complicated to avoid reading the methods anyway. So don't.
|
|
|
|
ReadMethodPool(Sel);
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:43 +08:00
|
|
|
void ASTReader::SetIdentifierInfo(unsigned ID, IdentifierInfo *II) {
|
Lazy deserialization of the declaration chains associated with
identifiers from a precompiled header.
This patch changes the primary name lookup method for entities within
a precompiled header. Previously, we would load all of the names of
declarations at translation unit scope into a large DenseMap (inside
the TranslationUnitDecl's DeclContext), and then perform a special
"last resort" lookup into this DeclContext when we knew there was a
PCH file (see Sema::LookupName). Now, when we see an identifier named
for the first time, we load all of the declarations with that name
that are visible from the translation unit into the IdentifierInfo's
chain of declarations. Thus, the explicit "look into the translation
unit's DeclContext" code is gone, and Sema effectively uses the same
IdentifierInfo-based name lookup mechanism whether we are using a PCH
file or not.
This approach should help PCH scale with the size of the input program
rather than the size of the PCH file. The "Hello, World!" application
with Carbon.h as a PCH file now loads 20% of the identifiers in the
PCH file rather than 85% of the identifiers.
90% of the 20% of identifiers loaded are actually loaded when we
deserialize the preprocessor state. The next step is to make the
preprocessor load macros lazily, which should drastically reduce the
number of types, declarations, and identifiers loaded for "Hello,
World".
llvm-svn: 69737
2009-04-22 06:25:48 +08:00
|
|
|
assert(ID && "Non-zero identifier ID required");
|
2009-04-29 05:53:25 +08:00
|
|
|
assert(ID <= IdentifiersLoaded.size() && "identifier ID out of range");
|
2009-04-26 03:10:14 +08:00
|
|
|
IdentifiersLoaded[ID - 1] = II;
|
2010-07-24 07:49:55 +08:00
|
|
|
if (DeserializationListener)
|
|
|
|
DeserializationListener->IdentifierRead(ID, II);
|
Lazy deserialization of the declaration chains associated with
identifiers from a precompiled header.
This patch changes the primary name lookup method for entities within
a precompiled header. Previously, we would load all of the names of
declarations at translation unit scope into a large DenseMap (inside
the TranslationUnitDecl's DeclContext), and then perform a special
"last resort" lookup into this DeclContext when we knew there was a
PCH file (see Sema::LookupName). Now, when we see an identifier named
for the first time, we load all of the declarations with that name
that are visible from the translation unit into the IdentifierInfo's
chain of declarations. Thus, the explicit "look into the translation
unit's DeclContext" code is gone, and Sema effectively uses the same
IdentifierInfo-based name lookup mechanism whether we are using a PCH
file or not.
This approach should help PCH scale with the size of the input program
rather than the size of the PCH file. The "Hello, World!" application
with Carbon.h as a PCH file now loads 20% of the identifiers in the
PCH file rather than 85% of the identifiers.
90% of the 20% of identifiers loaded are actually loaded when we
deserialize the preprocessor state. The next step is to make the
preprocessor load macros lazily, which should drastically reduce the
number of types, declarations, and identifiers loaded for "Hello,
World".
llvm-svn: 69737
2009-04-22 06:25:48 +08:00
|
|
|
}
|
|
|
|
|
2009-07-07 02:54:52 +08:00
|
|
|
/// \brief Set the globally-visible declarations associated with the given
|
|
|
|
/// identifier.
|
|
|
|
///
|
2010-08-19 07:57:06 +08:00
|
|
|
/// If the AST reader is currently in a state where the given declaration IDs
|
2009-09-09 23:08:12 +08:00
|
|
|
/// cannot safely be resolved, they are queued until it is safe to resolve
|
2009-07-07 02:54:52 +08:00
|
|
|
/// them.
|
|
|
|
///
|
|
|
|
/// \param II an IdentifierInfo that refers to one or more globally-visible
|
|
|
|
/// declarations.
|
|
|
|
///
|
|
|
|
/// \param DeclIDs the set of declaration IDs with the name @p II that are
|
|
|
|
/// visible at global scope.
|
|
|
|
///
|
|
|
|
/// \param Nonrecursive should be true to indicate that the caller knows that
|
|
|
|
/// this call is non-recursive, and therefore the globally-visible declarations
|
|
|
|
/// will not be placed onto the pending queue.
|
2009-09-09 23:08:12 +08:00
|
|
|
void
|
2010-08-19 07:56:43 +08:00
|
|
|
ASTReader::SetGloballyVisibleDecls(IdentifierInfo *II,
|
2009-07-07 02:54:52 +08:00
|
|
|
const llvm::SmallVectorImpl<uint32_t> &DeclIDs,
|
|
|
|
bool Nonrecursive) {
|
2010-07-30 18:03:16 +08:00
|
|
|
if (NumCurrentElementsDeserializing && !Nonrecursive) {
|
2009-07-07 02:54:52 +08:00
|
|
|
PendingIdentifierInfos.push_back(PendingIdentifierInfo());
|
|
|
|
PendingIdentifierInfo &PII = PendingIdentifierInfos.back();
|
|
|
|
PII.II = II;
|
2010-09-07 07:43:28 +08:00
|
|
|
PII.DeclIDs.append(DeclIDs.begin(), DeclIDs.end());
|
2009-07-07 02:54:52 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-07-07 02:54:52 +08:00
|
|
|
for (unsigned I = 0, N = DeclIDs.size(); I != N; ++I) {
|
|
|
|
NamedDecl *D = cast<NamedDecl>(GetDecl(DeclIDs[I]));
|
|
|
|
if (SemaObj) {
|
2010-08-13 11:15:25 +08:00
|
|
|
if (SemaObj->TUScope) {
|
|
|
|
// Introduce this declaration into the translation-unit scope
|
|
|
|
// and add it to the declaration chain for this identifier, so
|
|
|
|
// that (unqualified) name lookup will find it.
|
2010-08-21 17:40:31 +08:00
|
|
|
SemaObj->TUScope->AddDecl(D);
|
2010-08-13 11:15:25 +08:00
|
|
|
}
|
2010-09-25 07:29:12 +08:00
|
|
|
SemaObj->IdResolver.AddDeclToIdentifierChain(II, D);
|
2009-07-07 02:54:52 +08:00
|
|
|
} else {
|
|
|
|
// Queue this declaration so that it will be added to the
|
|
|
|
// translation unit scope and identifier's declaration chain
|
|
|
|
// once a Sema object is known.
|
|
|
|
PreloadedDecls.push_back(D);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:43 +08:00
|
|
|
IdentifierInfo *ASTReader::DecodeIdentifierInfo(unsigned ID) {
|
2009-04-11 08:14:32 +08:00
|
|
|
if (ID == 0)
|
|
|
|
return 0;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-07-21 08:46:22 +08:00
|
|
|
if (IdentifiersLoaded.empty()) {
|
2010-08-19 07:57:06 +08:00
|
|
|
Error("no identifier table in AST file");
|
2009-04-11 08:14:32 +08:00
|
|
|
return 0;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-06-19 08:03:23 +08:00
|
|
|
assert(PP && "Forgot to set Preprocessor ?");
|
2010-07-21 08:46:22 +08:00
|
|
|
ID -= 1;
|
|
|
|
if (!IdentifiersLoaded[ID]) {
|
|
|
|
unsigned Index = ID;
|
|
|
|
const char *Str = 0;
|
|
|
|
for (unsigned I = 0, N = Chain.size(); I != N; ++I) {
|
|
|
|
PerFileData *F = Chain[N - I - 1];
|
|
|
|
if (Index < F->LocalNumIdentifiers) {
|
|
|
|
uint32_t Offset = F->IdentifierOffsets[Index];
|
|
|
|
Str = F->IdentifierTableData + Offset;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
Index -= F->LocalNumIdentifiers;
|
|
|
|
}
|
|
|
|
assert(Str && "Broken Chain");
|
2009-04-26 05:04:17 +08:00
|
|
|
|
2010-08-19 07:57:06 +08:00
|
|
|
// All of the strings in the AST file are preceded by a 16-bit length.
|
|
|
|
// Extract that 16-bit length to avoid having to execute strlen().
|
2009-10-23 12:45:31 +08:00
|
|
|
// NOTE: 'StrLenPtr' is an 'unsigned char*' so that we load bytes as
|
|
|
|
// unsigned integers. This is important to avoid integer overflow when
|
|
|
|
// we cast them to 'unsigned'.
|
2009-10-23 11:57:22 +08:00
|
|
|
const unsigned char *StrLenPtr = (const unsigned char*) Str - 2;
|
2009-04-29 04:01:51 +08:00
|
|
|
unsigned StrLen = (((unsigned) StrLenPtr[0])
|
|
|
|
| (((unsigned) StrLenPtr[1]) << 8)) - 1;
|
2010-07-21 08:46:22 +08:00
|
|
|
IdentifiersLoaded[ID]
|
2010-03-12 18:32:27 +08:00
|
|
|
= &PP->getIdentifierTable().get(Str, StrLen);
|
2010-07-24 07:49:55 +08:00
|
|
|
if (DeserializationListener)
|
|
|
|
DeserializationListener->IdentifierRead(ID + 1, IdentifiersLoaded[ID]);
|
2009-04-11 08:14:32 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-07-21 08:46:22 +08:00
|
|
|
return IdentifiersLoaded[ID];
|
2009-04-10 06:27:44 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:43 +08:00
|
|
|
void ASTReader::ReadSLocEntry(unsigned ID) {
|
2009-04-27 14:38:32 +08:00
|
|
|
ReadSLocEntryRecord(ID);
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:43 +08:00
|
|
|
Selector ASTReader::DecodeSelector(unsigned ID) {
|
2009-04-23 18:39:46 +08:00
|
|
|
if (ID == 0)
|
|
|
|
return Selector();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-08-05 04:40:17 +08:00
|
|
|
if (ID > SelectorsLoaded.size()) {
|
2010-08-19 07:57:06 +08:00
|
|
|
Error("selector ID out of range in AST file");
|
2009-04-23 18:39:46 +08:00
|
|
|
return Selector();
|
|
|
|
}
|
2009-04-26 01:48:32 +08:00
|
|
|
|
2010-08-05 04:40:17 +08:00
|
|
|
if (SelectorsLoaded[ID - 1].getAsOpaquePtr() == 0) {
|
2009-04-26 01:48:32 +08:00
|
|
|
// Load this selector from the selector table.
|
2010-08-05 04:40:17 +08:00
|
|
|
unsigned Idx = ID - 1;
|
|
|
|
for (unsigned I = 0, N = Chain.size(); I != N; ++I) {
|
|
|
|
PerFileData &F = *Chain[N - I - 1];
|
|
|
|
if (Idx < F.LocalNumSelectors) {
|
2010-08-19 07:57:06 +08:00
|
|
|
ASTSelectorLookupTrait Trait(*this);
|
2010-08-05 04:40:17 +08:00
|
|
|
SelectorsLoaded[ID - 1] =
|
|
|
|
Trait.ReadKey(F.SelectorLookupTableData + F.SelectorOffsets[Idx], 0);
|
|
|
|
if (DeserializationListener)
|
|
|
|
DeserializationListener->SelectorRead(ID, SelectorsLoaded[ID - 1]);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
Idx -= F.LocalNumSelectors;
|
|
|
|
}
|
2009-04-26 01:48:32 +08:00
|
|
|
}
|
|
|
|
|
2010-08-05 04:40:17 +08:00
|
|
|
return SelectorsLoaded[ID - 1];
|
2009-04-23 18:39:46 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:43 +08:00
|
|
|
Selector ASTReader::GetExternalSelector(uint32_t ID) {
|
2010-04-07 01:30:22 +08:00
|
|
|
return DecodeSelector(ID);
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:43 +08:00
|
|
|
uint32_t ASTReader::GetNumExternalSelectors() {
|
2010-08-05 04:40:17 +08:00
|
|
|
// ID 0 (the null selector) is considered an external selector.
|
|
|
|
return getTotalNumSelectors() + 1;
|
2010-04-07 01:30:22 +08:00
|
|
|
}
|
|
|
|
|
2009-09-09 23:08:12 +08:00
|
|
|
DeclarationName
|
2010-08-19 07:56:43 +08:00
|
|
|
ASTReader::ReadDeclarationName(const RecordData &Record, unsigned &Idx) {
|
2009-04-10 06:27:44 +08:00
|
|
|
DeclarationName::NameKind Kind = (DeclarationName::NameKind)Record[Idx++];
|
|
|
|
switch (Kind) {
|
|
|
|
case DeclarationName::Identifier:
|
|
|
|
return DeclarationName(GetIdentifierInfo(Record, Idx));
|
|
|
|
|
|
|
|
case DeclarationName::ObjCZeroArgSelector:
|
|
|
|
case DeclarationName::ObjCOneArgSelector:
|
|
|
|
case DeclarationName::ObjCMultiArgSelector:
|
2009-04-23 23:15:40 +08:00
|
|
|
return DeclarationName(GetSelector(Record, Idx));
|
2009-04-10 06:27:44 +08:00
|
|
|
|
|
|
|
case DeclarationName::CXXConstructorName:
|
2009-04-28 05:45:14 +08:00
|
|
|
return Context->DeclarationNames.getCXXConstructorName(
|
2009-08-05 13:36:45 +08:00
|
|
|
Context->getCanonicalType(GetType(Record[Idx++])));
|
2009-04-10 06:27:44 +08:00
|
|
|
|
|
|
|
case DeclarationName::CXXDestructorName:
|
2009-04-28 05:45:14 +08:00
|
|
|
return Context->DeclarationNames.getCXXDestructorName(
|
2009-08-05 13:36:45 +08:00
|
|
|
Context->getCanonicalType(GetType(Record[Idx++])));
|
2009-04-10 06:27:44 +08:00
|
|
|
|
|
|
|
case DeclarationName::CXXConversionFunctionName:
|
2009-04-28 05:45:14 +08:00
|
|
|
return Context->DeclarationNames.getCXXConversionFunctionName(
|
2009-08-05 13:36:45 +08:00
|
|
|
Context->getCanonicalType(GetType(Record[Idx++])));
|
2009-04-10 06:27:44 +08:00
|
|
|
|
|
|
|
case DeclarationName::CXXOperatorName:
|
2009-04-28 05:45:14 +08:00
|
|
|
return Context->DeclarationNames.getCXXOperatorName(
|
2009-04-10 06:27:44 +08:00
|
|
|
(OverloadedOperatorKind)Record[Idx++]);
|
|
|
|
|
2009-11-29 15:34:05 +08:00
|
|
|
case DeclarationName::CXXLiteralOperatorName:
|
|
|
|
return Context->DeclarationNames.getCXXLiteralOperatorName(
|
|
|
|
GetIdentifierInfo(Record, Idx));
|
|
|
|
|
2009-04-10 06:27:44 +08:00
|
|
|
case DeclarationName::CXXUsingDirective:
|
|
|
|
return DeclarationName::getUsingDirectiveName();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Required to silence GCC warning
|
|
|
|
return DeclarationName();
|
|
|
|
}
|
2009-04-11 04:39:37 +08:00
|
|
|
|
2010-06-20 03:29:09 +08:00
|
|
|
TemplateName
|
2010-08-19 07:56:43 +08:00
|
|
|
ASTReader::ReadTemplateName(const RecordData &Record, unsigned &Idx) {
|
2010-06-20 03:29:09 +08:00
|
|
|
TemplateName::NameKind Kind = (TemplateName::NameKind)Record[Idx++];
|
|
|
|
switch (Kind) {
|
|
|
|
case TemplateName::Template:
|
|
|
|
return TemplateName(cast_or_null<TemplateDecl>(GetDecl(Record[Idx++])));
|
|
|
|
|
|
|
|
case TemplateName::OverloadedTemplate: {
|
|
|
|
unsigned size = Record[Idx++];
|
|
|
|
UnresolvedSet<8> Decls;
|
|
|
|
while (size--)
|
|
|
|
Decls.addDecl(cast<NamedDecl>(GetDecl(Record[Idx++])));
|
|
|
|
|
|
|
|
return Context->getOverloadedTemplateName(Decls.begin(), Decls.end());
|
|
|
|
}
|
|
|
|
|
|
|
|
case TemplateName::QualifiedTemplate: {
|
|
|
|
NestedNameSpecifier *NNS = ReadNestedNameSpecifier(Record, Idx);
|
|
|
|
bool hasTemplKeyword = Record[Idx++];
|
|
|
|
TemplateDecl *Template = cast<TemplateDecl>(GetDecl(Record[Idx++]));
|
|
|
|
return Context->getQualifiedTemplateName(NNS, hasTemplKeyword, Template);
|
|
|
|
}
|
|
|
|
|
|
|
|
case TemplateName::DependentTemplate: {
|
|
|
|
NestedNameSpecifier *NNS = ReadNestedNameSpecifier(Record, Idx);
|
|
|
|
if (Record[Idx++]) // isIdentifier
|
|
|
|
return Context->getDependentTemplateName(NNS,
|
|
|
|
GetIdentifierInfo(Record, Idx));
|
|
|
|
return Context->getDependentTemplateName(NNS,
|
2010-06-28 17:31:42 +08:00
|
|
|
(OverloadedOperatorKind)Record[Idx++]);
|
2010-06-20 03:29:09 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
assert(0 && "Unhandled template name kind!");
|
|
|
|
return TemplateName();
|
|
|
|
}
|
|
|
|
|
|
|
|
TemplateArgument
|
2010-08-19 07:56:43 +08:00
|
|
|
ASTReader::ReadTemplateArgument(llvm::BitstreamCursor &DeclsCursor,
|
2010-07-23 06:43:28 +08:00
|
|
|
const RecordData &Record, unsigned &Idx) {
|
2010-06-20 03:29:09 +08:00
|
|
|
switch ((TemplateArgument::ArgKind)Record[Idx++]) {
|
|
|
|
case TemplateArgument::Null:
|
|
|
|
return TemplateArgument();
|
|
|
|
case TemplateArgument::Type:
|
|
|
|
return TemplateArgument(GetType(Record[Idx++]));
|
|
|
|
case TemplateArgument::Declaration:
|
|
|
|
return TemplateArgument(GetDecl(Record[Idx++]));
|
2010-06-28 17:31:34 +08:00
|
|
|
case TemplateArgument::Integral: {
|
|
|
|
llvm::APSInt Value = ReadAPSInt(Record, Idx);
|
|
|
|
QualType T = GetType(Record[Idx++]);
|
|
|
|
return TemplateArgument(Value, T);
|
|
|
|
}
|
2010-06-20 03:29:09 +08:00
|
|
|
case TemplateArgument::Template:
|
|
|
|
return TemplateArgument(ReadTemplateName(Record, Idx));
|
|
|
|
case TemplateArgument::Expression:
|
2010-07-23 06:43:28 +08:00
|
|
|
return TemplateArgument(ReadExpr(DeclsCursor));
|
2010-06-20 03:29:09 +08:00
|
|
|
case TemplateArgument::Pack: {
|
|
|
|
unsigned NumArgs = Record[Idx++];
|
|
|
|
llvm::SmallVector<TemplateArgument, 8> Args;
|
|
|
|
Args.reserve(NumArgs);
|
|
|
|
while (NumArgs--)
|
2010-07-23 06:43:28 +08:00
|
|
|
Args.push_back(ReadTemplateArgument(DeclsCursor, Record, Idx));
|
2010-06-20 03:29:09 +08:00
|
|
|
TemplateArgument TemplArg;
|
|
|
|
TemplArg.setArgumentPack(Args.data(), Args.size(), /*CopyArgs=*/true);
|
|
|
|
return TemplArg;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
assert(0 && "Unhandled template argument kind!");
|
|
|
|
return TemplateArgument();
|
|
|
|
}
|
|
|
|
|
2010-06-23 21:48:30 +08:00
|
|
|
TemplateParameterList *
|
2010-08-19 07:56:43 +08:00
|
|
|
ASTReader::ReadTemplateParameterList(const RecordData &Record, unsigned &Idx) {
|
2010-06-23 21:48:30 +08:00
|
|
|
SourceLocation TemplateLoc = ReadSourceLocation(Record, Idx);
|
|
|
|
SourceLocation LAngleLoc = ReadSourceLocation(Record, Idx);
|
|
|
|
SourceLocation RAngleLoc = ReadSourceLocation(Record, Idx);
|
|
|
|
|
|
|
|
unsigned NumParams = Record[Idx++];
|
|
|
|
llvm::SmallVector<NamedDecl *, 16> Params;
|
|
|
|
Params.reserve(NumParams);
|
|
|
|
while (NumParams--)
|
|
|
|
Params.push_back(cast<NamedDecl>(GetDecl(Record[Idx++])));
|
|
|
|
|
|
|
|
TemplateParameterList* TemplateParams =
|
|
|
|
TemplateParameterList::Create(*Context, TemplateLoc, LAngleLoc,
|
|
|
|
Params.data(), Params.size(), RAngleLoc);
|
|
|
|
return TemplateParams;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2010-08-19 07:56:43 +08:00
|
|
|
ASTReader::
|
2010-06-23 21:48:30 +08:00
|
|
|
ReadTemplateArgumentList(llvm::SmallVector<TemplateArgument, 8> &TemplArgs,
|
2010-07-23 06:43:28 +08:00
|
|
|
llvm::BitstreamCursor &DeclsCursor,
|
2010-06-23 21:48:30 +08:00
|
|
|
const RecordData &Record, unsigned &Idx) {
|
|
|
|
unsigned NumTemplateArgs = Record[Idx++];
|
|
|
|
TemplArgs.reserve(NumTemplateArgs);
|
|
|
|
while (NumTemplateArgs--)
|
2010-07-23 06:43:28 +08:00
|
|
|
TemplArgs.push_back(ReadTemplateArgument(DeclsCursor, Record, Idx));
|
2010-06-23 21:48:30 +08:00
|
|
|
}
|
|
|
|
|
2010-07-02 19:55:32 +08:00
|
|
|
/// \brief Read a UnresolvedSet structure.
|
2010-08-19 07:56:43 +08:00
|
|
|
void ASTReader::ReadUnresolvedSet(UnresolvedSetImpl &Set,
|
2010-07-02 19:55:32 +08:00
|
|
|
const RecordData &Record, unsigned &Idx) {
|
|
|
|
unsigned NumDecls = Record[Idx++];
|
|
|
|
while (NumDecls--) {
|
|
|
|
NamedDecl *D = cast<NamedDecl>(GetDecl(Record[Idx++]));
|
|
|
|
AccessSpecifier AS = (AccessSpecifier)Record[Idx++];
|
|
|
|
Set.addDecl(D, AS);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-07-03 07:30:27 +08:00
|
|
|
CXXBaseSpecifier
|
2010-08-19 07:56:43 +08:00
|
|
|
ASTReader::ReadCXXBaseSpecifier(llvm::BitstreamCursor &DeclsCursor,
|
2010-07-27 00:56:01 +08:00
|
|
|
const RecordData &Record, unsigned &Idx) {
|
2010-07-03 07:30:27 +08:00
|
|
|
bool isVirtual = static_cast<bool>(Record[Idx++]);
|
|
|
|
bool isBaseOfClass = static_cast<bool>(Record[Idx++]);
|
|
|
|
AccessSpecifier AS = static_cast<AccessSpecifier>(Record[Idx++]);
|
2010-07-27 00:56:01 +08:00
|
|
|
TypeSourceInfo *TInfo = GetTypeSourceInfo(DeclsCursor, Record, Idx);
|
2010-07-03 07:30:27 +08:00
|
|
|
SourceRange Range = ReadSourceRange(Record, Idx);
|
2010-07-27 00:56:01 +08:00
|
|
|
return CXXBaseSpecifier(Range, isVirtual, isBaseOfClass, AS, TInfo);
|
2010-07-03 07:30:27 +08:00
|
|
|
}
|
|
|
|
|
2010-08-09 18:54:12 +08:00
|
|
|
std::pair<CXXBaseOrMemberInitializer **, unsigned>
|
2010-08-19 07:56:43 +08:00
|
|
|
ASTReader::ReadCXXBaseOrMemberInitializers(llvm::BitstreamCursor &Cursor,
|
2010-08-09 18:54:12 +08:00
|
|
|
const RecordData &Record,
|
|
|
|
unsigned &Idx) {
|
|
|
|
CXXBaseOrMemberInitializer **BaseOrMemberInitializers = 0;
|
|
|
|
unsigned NumInitializers = Record[Idx++];
|
|
|
|
if (NumInitializers) {
|
|
|
|
ASTContext &C = *getContext();
|
|
|
|
|
|
|
|
BaseOrMemberInitializers
|
|
|
|
= new (C) CXXBaseOrMemberInitializer*[NumInitializers];
|
|
|
|
for (unsigned i=0; i != NumInitializers; ++i) {
|
|
|
|
TypeSourceInfo *BaseClassInfo = 0;
|
|
|
|
bool IsBaseVirtual = false;
|
|
|
|
FieldDecl *Member = 0;
|
|
|
|
|
|
|
|
bool IsBaseInitializer = Record[Idx++];
|
|
|
|
if (IsBaseInitializer) {
|
|
|
|
BaseClassInfo = GetTypeSourceInfo(Cursor, Record, Idx);
|
|
|
|
IsBaseVirtual = Record[Idx++];
|
|
|
|
} else {
|
|
|
|
Member = cast<FieldDecl>(GetDecl(Record[Idx++]));
|
|
|
|
}
|
|
|
|
SourceLocation MemberLoc = ReadSourceLocation(Record, Idx);
|
|
|
|
Expr *Init = ReadExpr(Cursor);
|
|
|
|
FieldDecl *AnonUnionMember
|
|
|
|
= cast_or_null<FieldDecl>(GetDecl(Record[Idx++]));
|
|
|
|
SourceLocation LParenLoc = ReadSourceLocation(Record, Idx);
|
|
|
|
SourceLocation RParenLoc = ReadSourceLocation(Record, Idx);
|
|
|
|
bool IsWritten = Record[Idx++];
|
|
|
|
unsigned SourceOrderOrNumArrayIndices;
|
|
|
|
llvm::SmallVector<VarDecl *, 8> Indices;
|
|
|
|
if (IsWritten) {
|
|
|
|
SourceOrderOrNumArrayIndices = Record[Idx++];
|
|
|
|
} else {
|
|
|
|
SourceOrderOrNumArrayIndices = Record[Idx++];
|
|
|
|
Indices.reserve(SourceOrderOrNumArrayIndices);
|
|
|
|
for (unsigned i=0; i != SourceOrderOrNumArrayIndices; ++i)
|
|
|
|
Indices.push_back(cast<VarDecl>(GetDecl(Record[Idx++])));
|
|
|
|
}
|
|
|
|
|
|
|
|
CXXBaseOrMemberInitializer *BOMInit;
|
|
|
|
if (IsBaseInitializer) {
|
|
|
|
BOMInit = new (C) CXXBaseOrMemberInitializer(C, BaseClassInfo,
|
|
|
|
IsBaseVirtual, LParenLoc,
|
|
|
|
Init, RParenLoc);
|
|
|
|
} else if (IsWritten) {
|
|
|
|
BOMInit = new (C) CXXBaseOrMemberInitializer(C, Member, MemberLoc,
|
|
|
|
LParenLoc, Init, RParenLoc);
|
|
|
|
} else {
|
|
|
|
BOMInit = CXXBaseOrMemberInitializer::Create(C, Member, MemberLoc,
|
|
|
|
LParenLoc, Init, RParenLoc,
|
|
|
|
Indices.data(),
|
|
|
|
Indices.size());
|
|
|
|
}
|
|
|
|
|
2010-09-07 03:04:27 +08:00
|
|
|
if (IsWritten)
|
|
|
|
BOMInit->setSourceOrder(SourceOrderOrNumArrayIndices);
|
2010-08-09 18:54:12 +08:00
|
|
|
BOMInit->setAnonUnionMember(AnonUnionMember);
|
|
|
|
BaseOrMemberInitializers[i] = BOMInit;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return std::make_pair(BaseOrMemberInitializers, NumInitializers);
|
|
|
|
}
|
|
|
|
|
2010-05-08 05:43:38 +08:00
|
|
|
NestedNameSpecifier *
|
2010-08-19 07:56:43 +08:00
|
|
|
ASTReader::ReadNestedNameSpecifier(const RecordData &Record, unsigned &Idx) {
|
2010-05-08 05:43:38 +08:00
|
|
|
unsigned N = Record[Idx++];
|
|
|
|
NestedNameSpecifier *NNS = 0, *Prev = 0;
|
|
|
|
for (unsigned I = 0; I != N; ++I) {
|
|
|
|
NestedNameSpecifier::SpecifierKind Kind
|
|
|
|
= (NestedNameSpecifier::SpecifierKind)Record[Idx++];
|
|
|
|
switch (Kind) {
|
|
|
|
case NestedNameSpecifier::Identifier: {
|
|
|
|
IdentifierInfo *II = GetIdentifierInfo(Record, Idx);
|
|
|
|
NNS = NestedNameSpecifier::Create(*Context, Prev, II);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case NestedNameSpecifier::Namespace: {
|
|
|
|
NamespaceDecl *NS = cast<NamespaceDecl>(GetDecl(Record[Idx++]));
|
|
|
|
NNS = NestedNameSpecifier::Create(*Context, Prev, NS);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case NestedNameSpecifier::TypeSpec:
|
|
|
|
case NestedNameSpecifier::TypeSpecWithTemplate: {
|
|
|
|
Type *T = GetType(Record[Idx++]).getTypePtr();
|
|
|
|
bool Template = Record[Idx++];
|
|
|
|
NNS = NestedNameSpecifier::Create(*Context, Prev, Template, T);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case NestedNameSpecifier::Global: {
|
|
|
|
NNS = NestedNameSpecifier::GlobalSpecifier(*Context);
|
|
|
|
// No associated value, and there can't be a prefix.
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2010-07-07 23:46:30 +08:00
|
|
|
Prev = NNS;
|
2010-05-08 05:43:38 +08:00
|
|
|
}
|
|
|
|
return NNS;
|
|
|
|
}
|
|
|
|
|
|
|
|
SourceRange
|
2010-08-19 07:56:43 +08:00
|
|
|
ASTReader::ReadSourceRange(const RecordData &Record, unsigned &Idx) {
|
2010-06-02 23:47:10 +08:00
|
|
|
SourceLocation beg = SourceLocation::getFromRawEncoding(Record[Idx++]);
|
|
|
|
SourceLocation end = SourceLocation::getFromRawEncoding(Record[Idx++]);
|
|
|
|
return SourceRange(beg, end);
|
2010-05-08 05:43:38 +08:00
|
|
|
}
|
|
|
|
|
2009-04-14 02:14:40 +08:00
|
|
|
/// \brief Read an integral value
|
2010-08-19 07:56:43 +08:00
|
|
|
llvm::APInt ASTReader::ReadAPInt(const RecordData &Record, unsigned &Idx) {
|
2009-04-14 02:14:40 +08:00
|
|
|
unsigned BitWidth = Record[Idx++];
|
|
|
|
unsigned NumWords = llvm::APInt::getNumWords(BitWidth);
|
|
|
|
llvm::APInt Result(BitWidth, NumWords, &Record[Idx]);
|
|
|
|
Idx += NumWords;
|
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief Read a signed integral value
|
2010-08-19 07:56:43 +08:00
|
|
|
llvm::APSInt ASTReader::ReadAPSInt(const RecordData &Record, unsigned &Idx) {
|
2009-04-14 02:14:40 +08:00
|
|
|
bool isUnsigned = Record[Idx++];
|
|
|
|
return llvm::APSInt(ReadAPInt(Record, Idx), isUnsigned);
|
|
|
|
}
|
|
|
|
|
2009-04-15 05:55:33 +08:00
|
|
|
/// \brief Read a floating-point value
|
2010-08-19 07:56:43 +08:00
|
|
|
llvm::APFloat ASTReader::ReadAPFloat(const RecordData &Record, unsigned &Idx) {
|
2009-04-15 05:55:33 +08:00
|
|
|
return llvm::APFloat(ReadAPInt(Record, Idx));
|
|
|
|
}
|
|
|
|
|
2009-04-16 05:30:51 +08:00
|
|
|
// \brief Read a string
|
2010-08-19 07:56:43 +08:00
|
|
|
std::string ASTReader::ReadString(const RecordData &Record, unsigned &Idx) {
|
2009-04-16 05:30:51 +08:00
|
|
|
unsigned Len = Record[Idx++];
|
2009-05-21 17:52:38 +08:00
|
|
|
std::string Result(Record.data() + Idx, Record.data() + Idx + Len);
|
2009-04-16 05:30:51 +08:00
|
|
|
Idx += Len;
|
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:43 +08:00
|
|
|
CXXTemporary *ASTReader::ReadCXXTemporary(const RecordData &Record,
|
2010-05-10 08:25:06 +08:00
|
|
|
unsigned &Idx) {
|
|
|
|
CXXDestructorDecl *Decl = cast<CXXDestructorDecl>(GetDecl(Record[Idx++]));
|
|
|
|
return CXXTemporary::Create(*Context, Decl);
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:43 +08:00
|
|
|
DiagnosticBuilder ASTReader::Diag(unsigned DiagID) {
|
2009-04-11 07:10:45 +08:00
|
|
|
return Diag(SourceLocation(), DiagID);
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:43 +08:00
|
|
|
DiagnosticBuilder ASTReader::Diag(SourceLocation Loc, unsigned DiagID) {
|
2009-06-19 08:03:23 +08:00
|
|
|
return Diags.Report(FullSourceLoc(Loc, SourceMgr), DiagID);
|
2009-04-11 04:39:37 +08:00
|
|
|
}
|
2009-04-17 08:04:06 +08:00
|
|
|
|
Lazy deserialization of the declaration chains associated with
identifiers from a precompiled header.
This patch changes the primary name lookup method for entities within
a precompiled header. Previously, we would load all of the names of
declarations at translation unit scope into a large DenseMap (inside
the TranslationUnitDecl's DeclContext), and then perform a special
"last resort" lookup into this DeclContext when we knew there was a
PCH file (see Sema::LookupName). Now, when we see an identifier named
for the first time, we load all of the declarations with that name
that are visible from the translation unit into the IdentifierInfo's
chain of declarations. Thus, the explicit "look into the translation
unit's DeclContext" code is gone, and Sema effectively uses the same
IdentifierInfo-based name lookup mechanism whether we are using a PCH
file or not.
This approach should help PCH scale with the size of the input program
rather than the size of the PCH file. The "Hello, World!" application
with Carbon.h as a PCH file now loads 20% of the identifiers in the
PCH file rather than 85% of the identifiers.
90% of the 20% of identifiers loaded are actually loaded when we
deserialize the preprocessor state. The next step is to make the
preprocessor load macros lazily, which should drastically reduce the
number of types, declarations, and identifiers loaded for "Hello,
World".
llvm-svn: 69737
2009-04-22 06:25:48 +08:00
|
|
|
/// \brief Retrieve the identifier table associated with the
|
|
|
|
/// preprocessor.
|
2010-08-19 07:56:43 +08:00
|
|
|
IdentifierTable &ASTReader::getIdentifierTable() {
|
2009-06-19 08:03:23 +08:00
|
|
|
assert(PP && "Forgot to set Preprocessor ?");
|
|
|
|
return PP->getIdentifierTable();
|
Lazy deserialization of the declaration chains associated with
identifiers from a precompiled header.
This patch changes the primary name lookup method for entities within
a precompiled header. Previously, we would load all of the names of
declarations at translation unit scope into a large DenseMap (inside
the TranslationUnitDecl's DeclContext), and then perform a special
"last resort" lookup into this DeclContext when we knew there was a
PCH file (see Sema::LookupName). Now, when we see an identifier named
for the first time, we load all of the declarations with that name
that are visible from the translation unit into the IdentifierInfo's
chain of declarations. Thus, the explicit "look into the translation
unit's DeclContext" code is gone, and Sema effectively uses the same
IdentifierInfo-based name lookup mechanism whether we are using a PCH
file or not.
This approach should help PCH scale with the size of the input program
rather than the size of the PCH file. The "Hello, World!" application
with Carbon.h as a PCH file now loads 20% of the identifiers in the
PCH file rather than 85% of the identifiers.
90% of the 20% of identifiers loaded are actually loaded when we
deserialize the preprocessor state. The next step is to make the
preprocessor load macros lazily, which should drastically reduce the
number of types, declarations, and identifiers loaded for "Hello,
World".
llvm-svn: 69737
2009-04-22 06:25:48 +08:00
|
|
|
}
|
|
|
|
|
2009-04-17 08:04:06 +08:00
|
|
|
/// \brief Record that the given ID maps to the given switch-case
|
|
|
|
/// statement.
|
2010-08-19 07:56:43 +08:00
|
|
|
void ASTReader::RecordSwitchCaseID(SwitchCase *SC, unsigned ID) {
|
2009-04-17 08:04:06 +08:00
|
|
|
assert(SwitchCaseStmts[ID] == 0 && "Already have a SwitchCase with this ID");
|
|
|
|
SwitchCaseStmts[ID] = SC;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief Retrieve the switch-case statement with the given ID.
|
2010-08-19 07:56:43 +08:00
|
|
|
SwitchCase *ASTReader::getSwitchCaseWithID(unsigned ID) {
|
2009-04-17 08:04:06 +08:00
|
|
|
assert(SwitchCaseStmts[ID] != 0 && "No SwitchCase with this ID");
|
|
|
|
return SwitchCaseStmts[ID];
|
|
|
|
}
|
2009-04-18 02:18:49 +08:00
|
|
|
|
|
|
|
/// \brief Record that the given label statement has been
|
|
|
|
/// deserialized and has the given ID.
|
2010-08-19 07:56:43 +08:00
|
|
|
void ASTReader::RecordLabelStmt(LabelStmt *S, unsigned ID) {
|
2009-09-09 23:08:12 +08:00
|
|
|
assert(LabelStmts.find(ID) == LabelStmts.end() &&
|
2009-04-18 02:18:49 +08:00
|
|
|
"Deserialized label twice");
|
|
|
|
LabelStmts[ID] = S;
|
|
|
|
|
|
|
|
// If we've already seen any goto statements that point to this
|
|
|
|
// label, resolve them now.
|
|
|
|
typedef std::multimap<unsigned, GotoStmt *>::iterator GotoIter;
|
|
|
|
std::pair<GotoIter, GotoIter> Gotos = UnresolvedGotoStmts.equal_range(ID);
|
|
|
|
for (GotoIter Goto = Gotos.first; Goto != Gotos.second; ++Goto)
|
|
|
|
Goto->second->setLabel(S);
|
|
|
|
UnresolvedGotoStmts.erase(Gotos.first, Gotos.second);
|
2009-04-18 02:58:21 +08:00
|
|
|
|
|
|
|
// If we've already seen any address-label statements that point to
|
|
|
|
// this label, resolve them now.
|
|
|
|
typedef std::multimap<unsigned, AddrLabelExpr *>::iterator AddrLabelIter;
|
2009-09-09 23:08:12 +08:00
|
|
|
std::pair<AddrLabelIter, AddrLabelIter> AddrLabels
|
2009-04-18 02:58:21 +08:00
|
|
|
= UnresolvedAddrLabelExprs.equal_range(ID);
|
2009-09-09 23:08:12 +08:00
|
|
|
for (AddrLabelIter AddrLabel = AddrLabels.first;
|
2009-04-18 02:58:21 +08:00
|
|
|
AddrLabel != AddrLabels.second; ++AddrLabel)
|
|
|
|
AddrLabel->second->setLabel(S);
|
|
|
|
UnresolvedAddrLabelExprs.erase(AddrLabels.first, AddrLabels.second);
|
2009-04-18 02:18:49 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief Set the label of the given statement to the label
|
|
|
|
/// identified by ID.
|
|
|
|
///
|
|
|
|
/// Depending on the order in which the label and other statements
|
|
|
|
/// referencing that label occur, this operation may complete
|
|
|
|
/// immediately (updating the statement) or it may queue the
|
|
|
|
/// statement to be back-patched later.
|
2010-08-19 07:56:43 +08:00
|
|
|
void ASTReader::SetLabelOf(GotoStmt *S, unsigned ID) {
|
2009-04-18 02:18:49 +08:00
|
|
|
std::map<unsigned, LabelStmt *>::iterator Label = LabelStmts.find(ID);
|
|
|
|
if (Label != LabelStmts.end()) {
|
|
|
|
// We've already seen this label, so set the label of the goto and
|
|
|
|
// we're done.
|
|
|
|
S->setLabel(Label->second);
|
|
|
|
} else {
|
|
|
|
// We haven't seen this label yet, so add this goto to the set of
|
|
|
|
// unresolved goto statements.
|
|
|
|
UnresolvedGotoStmts.insert(std::make_pair(ID, S));
|
|
|
|
}
|
|
|
|
}
|
2009-04-18 02:58:21 +08:00
|
|
|
|
|
|
|
/// \brief Set the label of the given expression to the label
|
|
|
|
/// identified by ID.
|
|
|
|
///
|
|
|
|
/// Depending on the order in which the label and other statements
|
|
|
|
/// referencing that label occur, this operation may complete
|
|
|
|
/// immediately (updating the statement) or it may queue the
|
|
|
|
/// statement to be back-patched later.
|
2010-08-19 07:56:43 +08:00
|
|
|
void ASTReader::SetLabelOf(AddrLabelExpr *S, unsigned ID) {
|
2009-04-18 02:58:21 +08:00
|
|
|
std::map<unsigned, LabelStmt *>::iterator Label = LabelStmts.find(ID);
|
|
|
|
if (Label != LabelStmts.end()) {
|
|
|
|
// We've already seen this label, so set the label of the
|
|
|
|
// label-address expression and we're done.
|
|
|
|
S->setLabel(Label->second);
|
|
|
|
} else {
|
|
|
|
// We haven't seen this label yet, so add this label-address
|
|
|
|
// expression to the set of unresolved label-address expressions.
|
|
|
|
UnresolvedAddrLabelExprs.insert(std::make_pair(ID, S));
|
|
|
|
}
|
|
|
|
}
|
2009-07-07 02:54:52 +08:00
|
|
|
|
2010-08-19 07:56:43 +08:00
|
|
|
void ASTReader::FinishedDeserializing() {
|
2010-07-30 18:03:16 +08:00
|
|
|
assert(NumCurrentElementsDeserializing &&
|
|
|
|
"FinishedDeserializing not paired with StartedDeserializing");
|
|
|
|
if (NumCurrentElementsDeserializing == 1) {
|
2009-07-07 02:54:52 +08:00
|
|
|
// If any identifiers with corresponding top-level declarations have
|
|
|
|
// been loaded, load those declarations now.
|
2010-07-30 18:03:16 +08:00
|
|
|
while (!PendingIdentifierInfos.empty()) {
|
|
|
|
SetGloballyVisibleDecls(PendingIdentifierInfos.front().II,
|
|
|
|
PendingIdentifierInfos.front().DeclIDs, true);
|
|
|
|
PendingIdentifierInfos.pop_front();
|
2009-07-07 02:54:52 +08:00
|
|
|
}
|
2010-07-07 23:46:26 +08:00
|
|
|
|
|
|
|
// We are not in recursive loading, so it's safe to pass the "interesting"
|
|
|
|
// decls to the consumer.
|
2010-07-30 18:03:16 +08:00
|
|
|
if (Consumer)
|
|
|
|
PassInterestingDeclsToConsumer();
|
2009-07-07 02:54:52 +08:00
|
|
|
}
|
2010-07-30 18:03:16 +08:00
|
|
|
--NumCurrentElementsDeserializing;
|
2009-07-07 02:54:52 +08:00
|
|
|
}
|
2010-08-19 08:28:17 +08:00
|
|
|
|
2010-08-24 08:50:04 +08:00
|
|
|
ASTReader::ASTReader(Preprocessor &PP, ASTContext *Context,
|
|
|
|
const char *isysroot, bool DisableValidation)
|
|
|
|
: Listener(new PCHValidator(PP, *this)), DeserializationListener(0),
|
|
|
|
SourceMgr(PP.getSourceManager()), FileMgr(PP.getFileManager()),
|
|
|
|
Diags(PP.getDiagnostics()), SemaObj(0), PP(&PP), Context(Context),
|
|
|
|
Consumer(0), isysroot(isysroot), DisableValidation(DisableValidation),
|
|
|
|
NumStatHits(0), NumStatMisses(0), NumSLocEntriesRead(0),
|
2010-09-23 04:19:08 +08:00
|
|
|
TotalNumSLocEntries(0), NextSLocOffset(0), NumStatementsRead(0),
|
|
|
|
TotalNumStatements(0), NumMacrosRead(0), TotalNumMacros(0),
|
|
|
|
NumSelectorsRead(0), NumMethodPoolEntriesRead(0), NumMethodPoolMisses(0),
|
2010-08-24 08:50:04 +08:00
|
|
|
TotalNumMethodPoolEntries(0), NumLexicalDeclContextsRead(0),
|
|
|
|
TotalLexicalDeclContexts(0), NumVisibleDeclContextsRead(0),
|
|
|
|
TotalVisibleDeclContexts(0), NumCurrentElementsDeserializing(0) {
|
|
|
|
RelocatablePCH = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
ASTReader::ASTReader(SourceManager &SourceMgr, FileManager &FileMgr,
|
|
|
|
Diagnostic &Diags, const char *isysroot,
|
|
|
|
bool DisableValidation)
|
|
|
|
: DeserializationListener(0), SourceMgr(SourceMgr), FileMgr(FileMgr),
|
|
|
|
Diags(Diags), SemaObj(0), PP(0), Context(0), Consumer(0),
|
|
|
|
isysroot(isysroot), DisableValidation(DisableValidation), NumStatHits(0),
|
|
|
|
NumStatMisses(0), NumSLocEntriesRead(0), TotalNumSLocEntries(0),
|
2010-09-23 04:19:08 +08:00
|
|
|
NextSLocOffset(0), NumStatementsRead(0), TotalNumStatements(0),
|
|
|
|
NumMacrosRead(0), TotalNumMacros(0), NumSelectorsRead(0),
|
|
|
|
NumMethodPoolEntriesRead(0), NumMethodPoolMisses(0),
|
|
|
|
TotalNumMethodPoolEntries(0), NumLexicalDeclContextsRead(0),
|
|
|
|
TotalLexicalDeclContexts(0), NumVisibleDeclContextsRead(0),
|
|
|
|
TotalVisibleDeclContexts(0), NumCurrentElementsDeserializing(0) {
|
2010-08-24 08:50:04 +08:00
|
|
|
RelocatablePCH = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
ASTReader::~ASTReader() {
|
|
|
|
for (unsigned i = 0, e = Chain.size(); i != e; ++i)
|
|
|
|
delete Chain[e - i - 1];
|
|
|
|
// Delete all visible decl lookup tables
|
|
|
|
for (DeclContextOffsetsMap::iterator I = DeclContextOffsets.begin(),
|
|
|
|
E = DeclContextOffsets.end();
|
|
|
|
I != E; ++I) {
|
|
|
|
for (DeclContextInfos::iterator J = I->second.begin(), F = I->second.end();
|
|
|
|
J != F; ++J) {
|
|
|
|
if (J->NameLookupTableData)
|
|
|
|
delete static_cast<ASTDeclContextNameLookupTable*>(
|
|
|
|
J->NameLookupTableData);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (DeclContextVisibleUpdatesPending::iterator
|
|
|
|
I = PendingVisibleUpdates.begin(),
|
|
|
|
E = PendingVisibleUpdates.end();
|
|
|
|
I != E; ++I) {
|
|
|
|
for (DeclContextVisibleUpdates::iterator J = I->second.begin(),
|
|
|
|
F = I->second.end();
|
|
|
|
J != F; ++J)
|
|
|
|
delete static_cast<ASTDeclContextNameLookupTable*>(*J);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-19 08:28:17 +08:00
|
|
|
ASTReader::PerFileData::PerFileData()
|
2010-09-23 04:19:08 +08:00
|
|
|
: SizeInBits(0), LocalNumSLocEntries(0), SLocOffsets(0), LocalSLocSize(0),
|
2010-09-22 08:42:27 +08:00
|
|
|
LocalNumIdentifiers(0), IdentifierOffsets(0), IdentifierTableData(0),
|
|
|
|
IdentifierLookupTable(0), LocalNumMacroDefinitions(0),
|
|
|
|
MacroDefinitionOffsets(0), LocalNumSelectors(0), SelectorOffsets(0),
|
|
|
|
SelectorLookupTableData(0), SelectorLookupTable(0), LocalNumDecls(0),
|
|
|
|
DeclOffsets(0), LocalNumTypes(0), TypeOffsets(0), StatCache(0),
|
|
|
|
NumPreallocatedPreprocessingEntities(0)
|
2010-08-19 08:28:17 +08:00
|
|
|
{}
|
|
|
|
|
|
|
|
ASTReader::PerFileData::~PerFileData() {
|
|
|
|
delete static_cast<ASTIdentifierLookupTable *>(IdentifierLookupTable);
|
|
|
|
delete static_cast<ASTSelectorLookupTable *>(SelectorLookupTable);
|
|
|
|
}
|
|
|
|
|