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"
|
2011-08-26 04:47:51 +08:00
|
|
|
#include "clang/Serialization/ModuleManager.h"
|
2011-12-09 08:02:23 +08:00
|
|
|
#include "clang/Serialization/SerializationDiagnostic.h"
|
2010-08-21 00:03:52 +08:00
|
|
|
#include "ASTCommon.h"
|
2011-08-26 04:47:51 +08:00
|
|
|
#include "ASTReaderInternals.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"
|
2011-03-01 07:58:31 +08:00
|
|
|
#include "clang/AST/NestedNameSpecifier.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"
|
2010-11-24 03:19:34 +08:00
|
|
|
#include "clang/Basic/FileSystemStatCache.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"
|
Implement a new 'availability' attribute, that allows one to specify
which versions of an OS provide a certain facility. For example,
void foo()
__attribute__((availability(macosx,introduced=10.2,deprecated=10.4,obsoleted=10.6)));
says that the function "foo" was introduced in 10.2, deprecated in
10.4, and completely obsoleted in 10.6. This attribute ties in with
the deployment targets (e.g., -mmacosx-version-min=10.1 specifies that
we want to deploy back to Mac OS X 10.1). There are several concrete
behaviors that this attribute enables, as illustrated with the
function foo() above:
- If we choose a deployment target >= Mac OS X 10.4, uses of "foo"
will result in a deprecation warning, as if we had placed
attribute((deprecated)) on it (but with a better diagnostic)
- If we choose a deployment target >= Mac OS X 10.6, uses of "foo"
will result in an "unavailable" warning (in C)/error (in C++), as
if we had placed attribute((unavailable)) on it
- If we choose a deployment target prior to 10.2, foo() is
weak-imported (if it is a kind of entity that can be weak
imported), as if we had placed the weak_import attribute on it.
Naturally, there can be multiple availability attributes on a
declaration, for different platforms; only the current platform
matters when checking availability attributes.
The only platforms this attribute currently works for are "ios" and
"macosx", since we already have -mxxxx-version-min flags for them and we
have experience there with macro tricks translating down to the
deprecated/unavailable/weak_import attributes. The end goal is to open
this up to other platforms, and even extension to other "platforms"
that are really libraries (say, through a #pragma clang
define_system), but that hasn't yet been designed and we may want to
shake out more issues with this narrower problem first.
Addresses <rdar://problem/6690412>.
As a drive-by bug-fix, if an entity is both deprecated and
unavailable, we only emit the "unavailable" diagnostic.
llvm-svn: 128127
2011-03-23 08:50:03 +08:00
|
|
|
#include "clang/Basic/VersionTuple.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"
|
Implement two related optimizations that make de-serialization of
AST/PCH files more lazy:
- Don't preload all of the file source-location entries when reading
the AST file. Instead, load them lazily, when needed.
- Only look up header-search information (whether a header was already
#import'd, how many times it's been included, etc.) when it's needed
by the preprocessor, rather than pre-populating it.
Previously, we would pre-load all of the file source-location entries,
which also populated the header-search information structure. This was
a relatively minor performance issue, since we would end up stat()'ing
all of the headers stored within a AST/PCH file when the AST/PCH file
was loaded. In the normal PCH use case, the stat()s were cached, so
the cost--of preloading ~860 source-location entries in the Cocoa.h
case---was relatively low.
However, the recent optimization that replaced stat+open with
open+fstat turned this into a major problem, since the preloading of
source-location entries would now end up opening those files. Worse,
those files wouldn't be closed until the file manager was destroyed,
so just opening a Cocoa.h PCH file would hold on to ~860 file
descriptors, and it was easy to blow through the process's limit on
the number of open file descriptors.
By eliminating the preloading of these files, we neither open nor stat
the headers stored in the PCH/AST file until they're actually needed
for something. Concretely, we went from
*** HeaderSearch Stats:
835 files tracked.
364 #import/#pragma once files.
823 included exactly once.
6 max times a file is included.
3 #include/#include_next/#import.
0 #includes skipped due to the multi-include optimization.
1 framework lookups.
0 subframework lookups.
*** Source Manager Stats:
835 files mapped, 3 mem buffers mapped.
37460 SLocEntry's allocated, 11215575B of Sloc address space used.
62 bytes of files mapped, 0 files with line #'s computed.
with a trivial program that uses a chained PCH including a Cocoa PCH
to
*** HeaderSearch Stats:
4 files tracked.
1 #import/#pragma once files.
3 included exactly once.
2 max times a file is included.
3 #include/#include_next/#import.
0 #includes skipped due to the multi-include optimization.
1 framework lookups.
0 subframework lookups.
*** Source Manager Stats:
3 files mapped, 3 mem buffers mapped.
37460 SLocEntry's allocated, 11215575B of Sloc address space used.
62 bytes of files mapped, 0 files with line #'s computed.
for the same program.
llvm-svn: 125286
2011-02-11 01:09:37 +08:00
|
|
|
#include "llvm/Support/FileSystem.h"
|
2010-11-30 02:12:39 +08:00
|
|
|
#include "llvm/Support/Path.h"
|
2010-12-10 01:36:38 +08:00
|
|
|
#include "llvm/Support/system_error.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>
|
Implement two related optimizations that make de-serialization of
AST/PCH files more lazy:
- Don't preload all of the file source-location entries when reading
the AST file. Instead, load them lazily, when needed.
- Only look up header-search information (whether a header was already
#import'd, how many times it's been included, etc.) when it's needed
by the preprocessor, rather than pre-populating it.
Previously, we would pre-load all of the file source-location entries,
which also populated the header-search information structure. This was
a relatively minor performance issue, since we would end up stat()'ing
all of the headers stored within a AST/PCH file when the AST/PCH file
was loaded. In the normal PCH use case, the stat()s were cached, so
the cost--of preloading ~860 source-location entries in the Cocoa.h
case---was relatively low.
However, the recent optimization that replaced stat+open with
open+fstat turned this into a major problem, since the preloading of
source-location entries would now end up opening those files. Worse,
those files wouldn't be closed until the file manager was destroyed,
so just opening a Cocoa.h PCH file would hold on to ~860 file
descriptors, and it was easy to blow through the process's limit on
the number of open file descriptors.
By eliminating the preloading of these files, we neither open nor stat
the headers stored in the PCH/AST file until they're actually needed
for something. Concretely, we went from
*** HeaderSearch Stats:
835 files tracked.
364 #import/#pragma once files.
823 included exactly once.
6 max times a file is included.
3 #include/#include_next/#import.
0 #includes skipped due to the multi-include optimization.
1 framework lookups.
0 subframework lookups.
*** Source Manager Stats:
835 files mapped, 3 mem buffers mapped.
37460 SLocEntry's allocated, 11215575B of Sloc address space used.
62 bytes of files mapped, 0 files with line #'s computed.
with a trivial program that uses a chained PCH including a Cocoa PCH
to
*** HeaderSearch Stats:
4 files tracked.
1 #import/#pragma once files.
3 included exactly once.
2 max times a file is included.
3 #include/#include_next/#import.
0 #includes skipped due to the multi-include optimization.
1 framework lookups.
0 subframework lookups.
*** Source Manager Stats:
3 files mapped, 3 mem buffers mapped.
37460 SLocEntry's allocated, 11215575B of Sloc address space used.
62 bytes of files mapped, 0 files with line #'s computed.
for the same program.
llvm-svn: 125286
2011-02-11 01:09:37 +08:00
|
|
|
|
2009-04-10 06:27:44 +08:00
|
|
|
using namespace clang;
|
2010-08-19 07:57:32 +08:00
|
|
|
using namespace clang::serialization;
|
2011-08-26 04:47:51 +08:00
|
|
|
using namespace clang::serialization::reader;
|
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();
|
2011-09-14 02:26:39 +08:00
|
|
|
|
|
|
|
#define LANGOPT(Name, Bits, Default, Description) \
|
|
|
|
if (PPLangOpts.Name != LangOpts.Name) { \
|
|
|
|
Reader.Diag(diag::err_pch_langopt_mismatch) \
|
|
|
|
<< Description << LangOpts.Name << PPLangOpts.Name; \
|
|
|
|
return true; \
|
|
|
|
}
|
|
|
|
|
|
|
|
#define VALUE_LANGOPT(Name, Bits, Default, Description) \
|
|
|
|
if (PPLangOpts.Name != LangOpts.Name) { \
|
|
|
|
Reader.Diag(diag::err_pch_langopt_value_mismatch) \
|
|
|
|
<< Description; \
|
|
|
|
return true; \
|
|
|
|
}
|
|
|
|
|
|
|
|
#define ENUM_LANGOPT(Name, Type, Bits, Default, Description) \
|
|
|
|
if (PPLangOpts.get##Name() != LangOpts.get##Name()) { \
|
|
|
|
Reader.Diag(diag::err_pch_langopt_value_mismatch) \
|
|
|
|
<< Description; \
|
|
|
|
return true; \
|
2009-06-19 08:03:23 +08:00
|
|
|
}
|
|
|
|
|
2011-09-14 02:26:39 +08:00
|
|
|
#define BENIGN_LANGOPT(Name, Bits, Default, Description)
|
|
|
|
#define BENIGN_ENUM_LANGOPT(Name, Type, Bits, Default, Description)
|
|
|
|
#include "clang/Basic/LangOptions.def"
|
|
|
|
|
2009-06-19 08:03:23 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2011-07-23 18:55:15 +08:00
|
|
|
bool PCHValidator::ReadTargetTriple(StringRef Triple) {
|
2009-11-11 08:52:11 +08:00
|
|
|
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-11-26 02:29:30 +08:00
|
|
|
namespace {
|
|
|
|
struct EmptyStringRef {
|
2011-07-23 18:55:15 +08:00
|
|
|
bool operator ()(StringRef r) const { return r.empty(); }
|
2010-11-26 02:29:30 +08:00
|
|
|
};
|
|
|
|
struct EmptyBlock {
|
|
|
|
bool operator ()(const PCHPredefinesBlock &r) const {return r.Data.empty();}
|
|
|
|
};
|
|
|
|
}
|
2010-07-15 01:49:11 +08:00
|
|
|
|
2011-07-23 18:55:15 +08:00
|
|
|
static bool EqualConcatenations(SmallVector<StringRef, 2> L,
|
2010-07-15 01:49:11 +08:00
|
|
|
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.
|
2011-07-23 18:55:15 +08:00
|
|
|
StringRef LR = L[0], RR = R[0].Data;
|
2010-07-15 01:49:11 +08:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-07-23 18:55:15 +08:00
|
|
|
static std::pair<FileID, StringRef::size_type>
|
|
|
|
FindMacro(const PCHPredefinesBlocks &Buffers, StringRef MacroDef) {
|
|
|
|
std::pair<FileID, StringRef::size_type> Res;
|
2010-07-15 01:49:11 +08:00
|
|
|
for (unsigned I = 0, N = Buffers.size(); I != N; ++I) {
|
|
|
|
Res.second = Buffers[I].Data.find(MacroDef);
|
2011-07-23 18:55:15 +08:00
|
|
|
if (Res.second != StringRef::npos) {
|
2010-07-15 01:49:11 +08:00
|
|
|
Res.first = Buffers[I].BufferID;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return Res;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool PCHValidator::ReadPredefinesBuffer(const PCHPredefinesBlocks &Buffers,
|
2011-07-23 18:55:15 +08:00
|
|
|
StringRef OriginalFileName,
|
2011-02-24 05:16:44 +08:00
|
|
|
std::string &SuggestedPredefines,
|
|
|
|
FileManager &FileMgr) {
|
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 \"";
|
2011-12-09 09:33:57 +08:00
|
|
|
PCHInclude += HeaderSearch::NormalizeDashIncludePath(OriginalFileName,
|
|
|
|
FileMgr);
|
2009-11-11 13:29:04 +08:00
|
|
|
PCHInclude += "\"\n";
|
2011-07-23 18:55:15 +08:00
|
|
|
std::pair<StringRef,StringRef> Split =
|
|
|
|
StringRef(PP.getPredefines()).split(PCHInclude.str());
|
|
|
|
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.
|
2011-07-23 18:55:15 +08:00
|
|
|
SmallVector<StringRef, 2> CommandLine;
|
2010-07-15 01:49:11 +08:00
|
|
|
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.
|
2011-07-23 18:55:15 +08:00
|
|
|
SmallVector<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
|
|
|
|
2011-07-23 18:55:15 +08:00
|
|
|
SmallVector<StringRef, 8> CmdLineLines;
|
2009-11-14 00:46:11 +08:00
|
|
|
Left.split(CmdLineLines, "\n", /*MaxSplit=*/-1, /*KeepEmpty=*/false);
|
2010-10-01 00:53:50 +08:00
|
|
|
|
|
|
|
// Pick out implicit #includes after the PCH and don't consider them for
|
|
|
|
// validation; we will insert them into SuggestedPredefines so that the
|
|
|
|
// preprocessor includes them.
|
|
|
|
std::string IncludesAfterPCH;
|
2011-07-23 18:55:15 +08:00
|
|
|
SmallVector<StringRef, 8> AfterPCHLines;
|
2010-10-01 00:53:50 +08:00
|
|
|
Right.split(AfterPCHLines, "\n", /*MaxSplit=*/-1, /*KeepEmpty=*/false);
|
|
|
|
for (unsigned i = 0, e = AfterPCHLines.size(); i != e; ++i) {
|
|
|
|
if (AfterPCHLines[i].startswith("#include ")) {
|
|
|
|
IncludesAfterPCH += AfterPCHLines[i];
|
|
|
|
IncludesAfterPCH += '\n';
|
|
|
|
} else {
|
|
|
|
CmdLineLines.push_back(AfterPCHLines[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make sure we add the includes last into SuggestedPredefines before we
|
|
|
|
// exit this function.
|
|
|
|
struct AddIncludesRAII {
|
|
|
|
std::string &SuggestedPredefines;
|
|
|
|
std::string &IncludesAfterPCH;
|
|
|
|
|
|
|
|
AddIncludesRAII(std::string &SuggestedPredefines,
|
|
|
|
std::string &IncludesAfterPCH)
|
|
|
|
: SuggestedPredefines(SuggestedPredefines),
|
|
|
|
IncludesAfterPCH(IncludesAfterPCH) { }
|
|
|
|
~AddIncludesRAII() {
|
|
|
|
SuggestedPredefines += IncludesAfterPCH;
|
|
|
|
}
|
|
|
|
} AddIncludes(SuggestedPredefines, IncludesAfterPCH);
|
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.
|
2011-07-23 18:55:15 +08:00
|
|
|
std::vector<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) {
|
2011-07-23 18:55:15 +08:00
|
|
|
StringRef Missing = MissingPredefines[I];
|
2010-10-01 00:53:50 +08:00
|
|
|
if (Missing.startswith("#include ")) {
|
|
|
|
// An -include was specified when generating the PCH; it is included in
|
|
|
|
// the PCH, just ignore it.
|
|
|
|
continue;
|
|
|
|
}
|
2009-11-11 13:26:28 +08:00
|
|
|
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");
|
2011-07-23 18:55:15 +08:00
|
|
|
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();
|
2011-07-23 18:55:15 +08:00
|
|
|
SmallVector<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.
|
2011-07-23 18:55:15 +08:00
|
|
|
std::pair<FileID, StringRef::size_type> MacroLoc =
|
2010-07-15 01:49:11 +08:00
|
|
|
FindMacro(Buffers, Missing);
|
2011-07-23 18:55:15 +08:00
|
|
|
assert(MacroLoc.second!=StringRef::npos && "Unable to find macro!");
|
2010-07-15 01:49:11 +08:00
|
|
|
SourceLocation PCHMissingLoc =
|
|
|
|
SourceMgr.getLocForStartOfFile(MacroLoc.first)
|
2011-09-20 04:40:19 +08:00
|
|
|
.getLocWithOffset(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.
|
2011-07-23 18:55:15 +08:00
|
|
|
std::pair<FileID, StringRef::size_type> MacroLoc =
|
2010-07-15 01:49:11 +08:00
|
|
|
FindMacro(Buffers, Missing);
|
2011-07-23 18:55:15 +08:00
|
|
|
assert(MacroLoc.second!=StringRef::npos && "Unable to find macro!");
|
2010-07-15 01:49:11 +08:00
|
|
|
SourceLocation PCHMissingLoc =
|
|
|
|
SourceMgr.getLocForStartOfFile(MacroLoc.first)
|
2011-09-20 04:40:19 +08:00
|
|
|
.getLocWithOffset(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.
|
2011-07-23 18:55:15 +08:00
|
|
|
std::vector<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) {
|
2011-07-23 18:55:15 +08:00
|
|
|
StringRef &Extra = ExtraPredefines[I];
|
2009-11-11 13:26:28 +08:00
|
|
|
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");
|
2011-07-23 18:55:15 +08:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
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
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
unsigned ASTSelectorLookupTrait::ComputeHash(Selector Sel) {
|
|
|
|
return serialization::ComputeHash(Sel);
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-04-25 05:10:55 +08:00
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
std::pair<unsigned, unsigned>
|
|
|
|
ASTSelectorLookupTrait::ReadKeyDataLength(const unsigned char*& d) {
|
|
|
|
using namespace clang::io;
|
|
|
|
unsigned KeyLen = ReadUnalignedLE16(d);
|
|
|
|
unsigned DataLen = ReadUnalignedLE16(d);
|
|
|
|
return std::make_pair(KeyLen, DataLen);
|
|
|
|
}
|
2009-04-25 05:10:55 +08:00
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
ASTSelectorLookupTrait::internal_key_type
|
|
|
|
ASTSelectorLookupTrait::ReadKey(const unsigned char* d, unsigned) {
|
|
|
|
using namespace clang::io;
|
2011-09-10 05:34:22 +08:00
|
|
|
SelectorTable &SelTable = Reader.getContext().Selectors;
|
2011-08-26 04:47:51 +08:00
|
|
|
unsigned N = ReadUnalignedLE16(d);
|
|
|
|
IdentifierInfo *FirstII
|
|
|
|
= Reader.getLocalIdentifier(F, ReadUnalignedLE32(d));
|
|
|
|
if (N == 0)
|
|
|
|
return SelTable.getNullarySelector(FirstII);
|
|
|
|
else if (N == 1)
|
|
|
|
return SelTable.getUnarySelector(FirstII);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
SmallVector<IdentifierInfo *, 16> Args;
|
|
|
|
Args.push_back(FirstII);
|
|
|
|
for (unsigned I = 1; I != N; ++I)
|
|
|
|
Args.push_back(Reader.getLocalIdentifier(F, ReadUnalignedLE32(d)));
|
2009-04-25 05:10:55 +08:00
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
return SelTable.getSelector(N, Args.data());
|
|
|
|
}
|
2009-04-25 05:10:55 +08:00
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
ASTSelectorLookupTrait::data_type
|
|
|
|
ASTSelectorLookupTrait::ReadData(Selector, const unsigned char* d,
|
|
|
|
unsigned DataLen) {
|
|
|
|
using namespace clang::io;
|
2010-08-05 01:20:04 +08:00
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
data_type Result;
|
2009-04-25 05:10:55 +08:00
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
Result.ID = Reader.getGlobalSelectorID(F, ReadUnalignedLE32(d));
|
|
|
|
unsigned NumInstanceMethods = ReadUnalignedLE16(d);
|
|
|
|
unsigned NumFactoryMethods = ReadUnalignedLE16(d);
|
2009-04-25 05:10:55 +08:00
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
// Load instance methods
|
|
|
|
for (unsigned I = 0; I != NumInstanceMethods; ++I) {
|
|
|
|
if (ObjCMethodDecl *Method
|
|
|
|
= Reader.GetLocalDeclAs<ObjCMethodDecl>(F, ReadUnalignedLE32(d)))
|
|
|
|
Result.Instance.push_back(Method);
|
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
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
// Load factory methods
|
|
|
|
for (unsigned I = 0; I != NumFactoryMethods; ++I) {
|
|
|
|
if (ObjCMethodDecl *Method
|
|
|
|
= Reader.GetLocalDeclAs<ObjCMethodDecl>(F, ReadUnalignedLE32(d)))
|
|
|
|
Result.Factory.push_back(Method);
|
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
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
return Result;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
unsigned ASTIdentifierLookupTrait::ComputeHash(const internal_key_type& a) {
|
|
|
|
return llvm::HashString(StringRef(a.first, a.second));
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
std::pair<unsigned, unsigned>
|
|
|
|
ASTIdentifierLookupTrait::ReadKeyDataLength(const unsigned char*& d) {
|
|
|
|
using namespace clang::io;
|
|
|
|
unsigned DataLen = ReadUnalignedLE16(d);
|
|
|
|
unsigned KeyLen = ReadUnalignedLE16(d);
|
|
|
|
return std::make_pair(KeyLen, DataLen);
|
|
|
|
}
|
2010-10-15 06:11:03 +08:00
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
std::pair<const char*, unsigned>
|
|
|
|
ASTIdentifierLookupTrait::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
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
IdentifierInfo *ASTIdentifierLookupTrait::ReadData(const internal_key_type& k,
|
|
|
|
const unsigned char* d,
|
|
|
|
unsigned DataLen) {
|
|
|
|
using namespace clang::io;
|
|
|
|
unsigned RawID = ReadUnalignedLE32(d);
|
|
|
|
bool IsInteresting = RawID & 0x01;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
// Wipe out the "is interesting" bit.
|
|
|
|
RawID = RawID >> 1;
|
2009-04-29 05:18:29 +08:00
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
IdentID ID = Reader.getGlobalIdentifierID(F, RawID);
|
|
|
|
if (!IsInteresting) {
|
|
|
|
// For uninteresting identifiers, just build the IdentifierInfo
|
|
|
|
// and associate it with the persistent ID.
|
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 *II = KnownII;
|
|
|
|
if (!II)
|
2011-08-04 05:49:18 +08:00
|
|
|
II = &Reader.getIdentifierTable().getOwn(StringRef(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);
|
2010-08-19 07:57:06 +08:00
|
|
|
II->setIsFromAST();
|
Make the loading of information attached to an IdentifierInfo from an
AST file more lazy, so that we don't eagerly load that information for
all known identifiers each time a new AST file is loaded. The eager
reloading made some sense in the context of precompiled headers, since
very few identifiers were defined before PCH load time. With modules,
however, a huge amount of code can get parsed before we see an
@import, so laziness becomes important here.
The approach taken to make this information lazy is fairly simple:
when we load a new AST file, we mark all of the existing identifiers
as being out-of-date. Whenever we want to access information that may
come from an AST (e.g., whether the identifier has a macro definition,
or what top-level declarations have that name), we check the
out-of-date bit and, if it's set, ask the AST reader to update the
IdentifierInfo from the AST files. The update is a merge, and we now
take care to merge declarations before/after imports with declarations
from multiple imports.
The results of this optimization are fairly dramatic. On a small
application that brings in 14 non-trivial modules, this takes modules
from being > 3x slower than a "perfect" PCH file down to 30% slower
for a full rebuild. A partial rebuild (where the PCH file or modules
can be re-used) is down to 7% slower. Making the PCH file just a
little imperfect (e.g., adding two smallish modules used by a bunch of
.m files that aren't in the PCH file) tips the scales in favor of the
modules approach, with 24% faster partial rebuilds.
This is just a first step; the lazy scheme could possibly be improved
by adding versioning, so we don't search into modules we already
searched. Moreover, we'll need similar lazy schemes for all of the
other lookup data structures, such as DeclContexts.
llvm-svn: 143100
2011-10-27 17:33:13 +08:00
|
|
|
II->setOutOfDate(false);
|
2009-06-19 08:03:23 +08:00
|
|
|
return II;
|
2009-04-29 02:58:38 +08:00
|
|
|
}
|
2010-08-21 00:04:27 +08:00
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
unsigned Bits = ReadUnalignedLE16(d);
|
|
|
|
bool CPlusPlusOperatorKeyword = Bits & 0x01;
|
|
|
|
Bits >>= 1;
|
|
|
|
bool HasRevertedTokenIDToIdentifier = Bits & 0x01;
|
|
|
|
Bits >>= 1;
|
|
|
|
bool Poisoned = Bits & 0x01;
|
|
|
|
Bits >>= 1;
|
|
|
|
bool ExtensionToken = Bits & 0x01;
|
|
|
|
Bits >>= 1;
|
|
|
|
bool hasMacroDefinition = Bits & 0x01;
|
|
|
|
Bits >>= 1;
|
2011-12-19 13:04:33 +08:00
|
|
|
unsigned ObjCOrBuiltinID = Bits & 0x7FF;
|
|
|
|
Bits >>= 11;
|
2011-08-26 04:47:51 +08:00
|
|
|
|
|
|
|
assert(Bits == 0 && "Extra bits in the identifier?");
|
|
|
|
DataLen -= 6;
|
|
|
|
|
|
|
|
// Build the IdentifierInfo itself and link the identifier ID with
|
|
|
|
// the new IdentifierInfo.
|
|
|
|
IdentifierInfo *II = KnownII;
|
|
|
|
if (!II)
|
|
|
|
II = &Reader.getIdentifierTable().getOwn(StringRef(k.first, k.second));
|
Make the loading of information attached to an IdentifierInfo from an
AST file more lazy, so that we don't eagerly load that information for
all known identifiers each time a new AST file is loaded. The eager
reloading made some sense in the context of precompiled headers, since
very few identifiers were defined before PCH load time. With modules,
however, a huge amount of code can get parsed before we see an
@import, so laziness becomes important here.
The approach taken to make this information lazy is fairly simple:
when we load a new AST file, we mark all of the existing identifiers
as being out-of-date. Whenever we want to access information that may
come from an AST (e.g., whether the identifier has a macro definition,
or what top-level declarations have that name), we check the
out-of-date bit and, if it's set, ask the AST reader to update the
IdentifierInfo from the AST files. The update is a merge, and we now
take care to merge declarations before/after imports with declarations
from multiple imports.
The results of this optimization are fairly dramatic. On a small
application that brings in 14 non-trivial modules, this takes modules
from being > 3x slower than a "perfect" PCH file down to 30% slower
for a full rebuild. A partial rebuild (where the PCH file or modules
can be re-used) is down to 7% slower. Making the PCH file just a
little imperfect (e.g., adding two smallish modules used by a bunch of
.m files that aren't in the PCH file) tips the scales in favor of the
modules approach, with 24% faster partial rebuilds.
This is just a first step; the lazy scheme could possibly be improved
by adding versioning, so we don't search into modules we already
searched. Moreover, we'll need similar lazy schemes for all of the
other lookup data structures, such as DeclContexts.
llvm-svn: 143100
2011-10-27 17:33:13 +08:00
|
|
|
II->setOutOfDate(false);
|
|
|
|
II->setIsFromAST();
|
2011-08-26 04:47:51 +08:00
|
|
|
|
|
|
|
// Set or check the various bits in the IdentifierInfo structure.
|
|
|
|
// Token IDs are read-only.
|
|
|
|
if (HasRevertedTokenIDToIdentifier)
|
|
|
|
II->RevertTokenIDToIdentifier();
|
|
|
|
II->setObjCOrBuiltinID(ObjCOrBuiltinID);
|
|
|
|
assert(II->isExtensionToken() == ExtensionToken &&
|
|
|
|
"Incorrect extension token flag");
|
|
|
|
(void)ExtensionToken;
|
|
|
|
if (Poisoned)
|
|
|
|
II->setIsPoisoned(true);
|
|
|
|
assert(II->isCPlusPlusOperatorKeyword() == CPlusPlusOperatorKeyword &&
|
|
|
|
"Incorrect C++ operator keyword flag");
|
|
|
|
(void)CPlusPlusOperatorKeyword;
|
|
|
|
|
|
|
|
// If this identifier is a macro, deserialize the macro
|
|
|
|
// definition.
|
|
|
|
if (hasMacroDefinition) {
|
|
|
|
// FIXME: Check for conflicts?
|
|
|
|
uint32_t Offset = ReadUnalignedLE32(d);
|
2011-12-02 23:45:10 +08:00
|
|
|
unsigned LocalSubmoduleID = ReadUnalignedLE32(d);
|
|
|
|
|
|
|
|
// Determine whether this macro definition should be visible now, or
|
|
|
|
// whether it is in a hidden submodule.
|
|
|
|
bool Visible = true;
|
|
|
|
if (SubmoduleID GlobalSubmoduleID
|
|
|
|
= Reader.getGlobalSubmoduleID(F, LocalSubmoduleID)) {
|
|
|
|
if (Module *Owner = Reader.getSubmodule(GlobalSubmoduleID)) {
|
|
|
|
if (Owner->NameVisibility == Module::Hidden) {
|
|
|
|
// The owning module is not visible, and this macro definition should
|
|
|
|
// not be, either.
|
|
|
|
Visible = false;
|
|
|
|
|
|
|
|
// Note that this macro definition was hidden because its owning
|
|
|
|
// module is not yet visible.
|
|
|
|
Reader.HiddenNamesMap[Owner].push_back(II);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Reader.setIdentifierIsMacro(II, F, Offset, Visible);
|
|
|
|
DataLen -= 8;
|
2011-08-26 04:47:51 +08:00
|
|
|
}
|
|
|
|
|
Make the loading of information attached to an IdentifierInfo from an
AST file more lazy, so that we don't eagerly load that information for
all known identifiers each time a new AST file is loaded. The eager
reloading made some sense in the context of precompiled headers, since
very few identifiers were defined before PCH load time. With modules,
however, a huge amount of code can get parsed before we see an
@import, so laziness becomes important here.
The approach taken to make this information lazy is fairly simple:
when we load a new AST file, we mark all of the existing identifiers
as being out-of-date. Whenever we want to access information that may
come from an AST (e.g., whether the identifier has a macro definition,
or what top-level declarations have that name), we check the
out-of-date bit and, if it's set, ask the AST reader to update the
IdentifierInfo from the AST files. The update is a merge, and we now
take care to merge declarations before/after imports with declarations
from multiple imports.
The results of this optimization are fairly dramatic. On a small
application that brings in 14 non-trivial modules, this takes modules
from being > 3x slower than a "perfect" PCH file down to 30% slower
for a full rebuild. A partial rebuild (where the PCH file or modules
can be re-used) is down to 7% slower. Making the PCH file just a
little imperfect (e.g., adding two smallish modules used by a bunch of
.m files that aren't in the PCH file) tips the scales in favor of the
modules approach, with 24% faster partial rebuilds.
This is just a first step; the lazy scheme could possibly be improved
by adding versioning, so we don't search into modules we already
searched. Moreover, we'll need similar lazy schemes for all of the
other lookup data structures, such as DeclContexts.
llvm-svn: 143100
2011-10-27 17:33:13 +08:00
|
|
|
Reader.SetIdentifierInfo(ID, II);
|
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
// Read all of the declarations visible at global scope with this
|
|
|
|
// name.
|
|
|
|
if (DataLen > 0) {
|
|
|
|
SmallVector<uint32_t, 4> DeclIDs;
|
|
|
|
for (; DataLen > 0; DataLen -= 4)
|
|
|
|
DeclIDs.push_back(Reader.getGlobalDeclID(F, ReadUnalignedLE32(d)));
|
|
|
|
Reader.SetGloballyVisibleDecls(II, DeclIDs);
|
|
|
|
}
|
|
|
|
|
|
|
|
return II;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned
|
|
|
|
ASTDeclContextNameLookupTrait::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::CXXOperatorName:
|
|
|
|
ID.AddInteger((OverloadedOperatorKind)Key.Data);
|
|
|
|
break;
|
|
|
|
case DeclarationName::CXXConstructorName:
|
|
|
|
case DeclarationName::CXXDestructorName:
|
|
|
|
case DeclarationName::CXXConversionFunctionName:
|
|
|
|
case DeclarationName::CXXUsingDirective:
|
|
|
|
break;
|
2010-08-21 00:04:27 +08:00
|
|
|
}
|
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
return ID.ComputeHash();
|
|
|
|
}
|
2010-10-21 11:16:25 +08:00
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
ASTDeclContextNameLookupTrait::internal_key_type
|
|
|
|
ASTDeclContextNameLookupTrait::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::CXXOperatorName:
|
|
|
|
Key.Data = Name.getCXXOverloadedOperator();
|
|
|
|
break;
|
|
|
|
case DeclarationName::CXXLiteralOperatorName:
|
|
|
|
Key.Data = (uint64_t)Name.getCXXLiteralIdentifier();
|
|
|
|
break;
|
|
|
|
case DeclarationName::CXXConstructorName:
|
|
|
|
case DeclarationName::CXXDestructorName:
|
|
|
|
case DeclarationName::CXXConversionFunctionName:
|
|
|
|
case DeclarationName::CXXUsingDirective:
|
|
|
|
Key.Data = 0;
|
|
|
|
break;
|
2010-08-21 00:04:27 +08:00
|
|
|
}
|
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
return Key;
|
|
|
|
}
|
2010-08-21 07:35:55 +08:00
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
ASTDeclContextNameLookupTrait::external_key_type
|
|
|
|
ASTDeclContextNameLookupTrait::GetExternalKey(
|
|
|
|
const internal_key_type& Key) const {
|
2011-09-10 05:34:22 +08:00
|
|
|
ASTContext &Context = Reader.getContext();
|
2011-08-26 04:47:51 +08:00
|
|
|
switch (Key.Kind) {
|
|
|
|
case DeclarationName::Identifier:
|
|
|
|
return DeclarationName((IdentifierInfo*)Key.Data);
|
2010-08-21 07:35:55 +08:00
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
case DeclarationName::ObjCZeroArgSelector:
|
|
|
|
case DeclarationName::ObjCOneArgSelector:
|
|
|
|
case DeclarationName::ObjCMultiArgSelector:
|
|
|
|
return DeclarationName(Selector(Key.Data));
|
2010-08-21 07:35:55 +08:00
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
case DeclarationName::CXXConstructorName:
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.DeclarationNames.getCXXConstructorName(
|
|
|
|
Context.getCanonicalType(Reader.getLocalType(F, Key.Data)));
|
2010-08-21 07:35:55 +08:00
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
case DeclarationName::CXXDestructorName:
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.DeclarationNames.getCXXDestructorName(
|
|
|
|
Context.getCanonicalType(Reader.getLocalType(F, Key.Data)));
|
2010-08-21 07:35:55 +08:00
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
case DeclarationName::CXXConversionFunctionName:
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.DeclarationNames.getCXXConversionFunctionName(
|
|
|
|
Context.getCanonicalType(Reader.getLocalType(F, Key.Data)));
|
2010-08-21 07:35:55 +08:00
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
case DeclarationName::CXXOperatorName:
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.DeclarationNames.getCXXOperatorName(
|
2011-08-26 04:47:51 +08:00
|
|
|
(OverloadedOperatorKind)Key.Data);
|
2010-08-21 07:35:55 +08:00
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
case DeclarationName::CXXLiteralOperatorName:
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.DeclarationNames.getCXXLiteralOperatorName(
|
2011-08-26 04:47:51 +08:00
|
|
|
(IdentifierInfo*)Key.Data);
|
2010-08-21 07:35:55 +08:00
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
case DeclarationName::CXXUsingDirective:
|
|
|
|
return DeclarationName::getUsingDirectiveName();
|
2010-08-21 00:04:27 +08:00
|
|
|
}
|
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
llvm_unreachable("Invalid Name Kind ?");
|
|
|
|
}
|
2010-08-21 00:04:27 +08:00
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
std::pair<unsigned, unsigned>
|
|
|
|
ASTDeclContextNameLookupTrait::ReadKeyDataLength(const unsigned char*& d) {
|
|
|
|
using namespace clang::io;
|
|
|
|
unsigned KeyLen = ReadUnalignedLE16(d);
|
|
|
|
unsigned DataLen = ReadUnalignedLE16(d);
|
|
|
|
return std::make_pair(KeyLen, DataLen);
|
|
|
|
}
|
2010-10-21 11:16:25 +08:00
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
ASTDeclContextNameLookupTrait::internal_key_type
|
|
|
|
ASTDeclContextNameLookupTrait::ReadKey(const unsigned char* d, unsigned) {
|
|
|
|
using namespace clang::io;
|
2010-08-21 00:04:27 +08:00
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
DeclNameKey Key;
|
|
|
|
Key.Kind = (DeclarationName::NameKind)*d++;
|
|
|
|
switch (Key.Kind) {
|
|
|
|
case DeclarationName::Identifier:
|
|
|
|
Key.Data = (uint64_t)Reader.getLocalIdentifier(F, ReadUnalignedLE32(d));
|
|
|
|
break;
|
|
|
|
case DeclarationName::ObjCZeroArgSelector:
|
|
|
|
case DeclarationName::ObjCOneArgSelector:
|
|
|
|
case DeclarationName::ObjCMultiArgSelector:
|
|
|
|
Key.Data =
|
|
|
|
(uint64_t)Reader.getLocalSelector(F, ReadUnalignedLE32(d))
|
|
|
|
.getAsOpaquePtr();
|
|
|
|
break;
|
|
|
|
case DeclarationName::CXXOperatorName:
|
|
|
|
Key.Data = *d++; // OverloadedOperatorKind
|
|
|
|
break;
|
|
|
|
case DeclarationName::CXXLiteralOperatorName:
|
|
|
|
Key.Data = (uint64_t)Reader.getLocalIdentifier(F, ReadUnalignedLE32(d));
|
|
|
|
break;
|
|
|
|
case DeclarationName::CXXConstructorName:
|
|
|
|
case DeclarationName::CXXDestructorName:
|
|
|
|
case DeclarationName::CXXConversionFunctionName:
|
|
|
|
case DeclarationName::CXXUsingDirective:
|
|
|
|
Key.Data = 0;
|
|
|
|
break;
|
2010-08-21 00:04:27 +08:00
|
|
|
}
|
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
return Key;
|
|
|
|
}
|
2010-08-21 00:04:27 +08:00
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
ASTDeclContextNameLookupTrait::data_type
|
|
|
|
ASTDeclContextNameLookupTrait::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);
|
|
|
|
}
|
2010-08-21 00:04:27 +08:00
|
|
|
|
2011-12-01 07:21:26 +08:00
|
|
|
bool ASTReader::ReadDeclContextStorage(ModuleFile &M,
|
2011-08-25 03:03:07 +08:00
|
|
|
llvm::BitstreamCursor &Cursor,
|
2010-08-21 00:04:35 +08:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2010-10-15 04:14:34 +08:00
|
|
|
Info.LexicalDecls = reinterpret_cast<const KindDeclIDPair*>(Blob);
|
|
|
|
Info.NumLexicalDecls = BlobLen / sizeof(KindDeclIDPair);
|
2010-08-21 00:04:35 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// 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,
|
2011-08-25 03:03:07 +08:00
|
|
|
ASTDeclContextNameLookupTrait(*this, M));
|
2010-08-21 00:04:35 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2011-07-23 18:55:15 +08:00
|
|
|
void ASTReader::Error(StringRef Msg) {
|
2011-04-26 06:23:56 +08:00
|
|
|
Error(diag::err_fe_pch_malformed, Msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ASTReader::Error(unsigned DiagID,
|
2011-07-23 18:55:15 +08:00
|
|
|
StringRef Arg1, StringRef Arg2) {
|
2011-04-26 06:23:56 +08:00
|
|
|
if (Diags.isDiagnosticInFlight())
|
|
|
|
Diags.SetDelayedDiagnostic(DiagID, Arg1, Arg2);
|
|
|
|
else
|
|
|
|
Diag(DiagID) << Arg1 << Arg2;
|
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,
|
2011-02-24 05:16:44 +08:00
|
|
|
SuggestedPredefines,
|
|
|
|
FileMgr);
|
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.
|
2010-10-05 23:59:54 +08:00
|
|
|
/// \returns true if there was an error.
|
2011-12-01 07:21:26 +08:00
|
|
|
bool ASTReader::ParseLineTable(ModuleFile &F,
|
2011-07-23 18:55:15 +08:00
|
|
|
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);
|
2011-06-21 23:13:30 +08:00
|
|
|
FileIDs[I] = LineTable.getLineTableFilenameID(Filename);
|
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++];
|
Revamp the SourceManager to separate the representation of parsed
source locations from source locations loaded from an AST/PCH file.
Previously, loading an AST/PCH file involved carefully pre-allocating
space at the beginning of the source manager for the source locations
and FileIDs that correspond to the prefix, and then appending the
source locations/FileIDs used for parsing the remaining translation
unit. This design forced us into loading PCH files early, as a prefix,
whic has become a rather significant limitation.
This patch splits the SourceManager space into two parts: for source
location "addresses", the lower values (growing upward) are used to
describe parsed code, while upper values (growing downward) are used
for source locations loaded from AST/PCH files. Similarly, positive
FileIDs are used to describe parsed code while negative FileIDs are
used to file/macro locations loaded from AST/PCH files. As a result,
we can load PCH/AST files even during parsing, making various
improvemnts in the future possible, e.g., teaching #include <foo.h> to
look for and load <foo.h.gch> if it happens to be already available.
This patch was originally written by Sebastian Redl, then brought
forward to the modern age by Jonathan Turner, and finally
polished/finished by me to be committed.
llvm-svn: 135484
2011-07-20 00:10:42 +08:00
|
|
|
assert(FID >= 0 && "Serialized line entries for non-local file.");
|
|
|
|
// Remap FileID from 1-based old view.
|
|
|
|
FID += F.SLocEntryBaseID - 1;
|
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 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)
|
2010-11-24 03:28:12 +08:00
|
|
|
: ino(i), dev(d), mode(mo), mtime(m), size(s) {}
|
2009-04-28 02:38:38 +08:00
|
|
|
};
|
|
|
|
|
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;
|
|
|
|
|
|
|
|
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-11-24 03:19:34 +08:00
|
|
|
class ASTStatCache : public FileSystemStatCache {
|
2010-08-19 07:57:06 +08:00
|
|
|
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-11-24 03:28:12 +08:00
|
|
|
ASTStatCache(const unsigned char *Buckets, const unsigned char *Base,
|
|
|
|
unsigned &NumStatHits, 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
|
|
|
|
2010-11-24 05:17:56 +08:00
|
|
|
LookupResult getStat(const char *Path, struct stat &StatBuf,
|
|
|
|
int *FileDescriptor) {
|
2010-08-19 07:57:06 +08:00
|
|
|
// Do the lookup for the file's data in the AST file.
|
2010-11-24 03:19:34 +08:00
|
|
|
CacheTy::iterator I = Cache->find(Path);
|
2009-04-28 02:38:38 +08:00
|
|
|
|
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;
|
2010-11-24 05:17:56 +08:00
|
|
|
return statChained(Path, StatBuf, FileDescriptor);
|
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
|
|
|
|
2010-11-24 03:19:34 +08:00
|
|
|
StatBuf.st_ino = Data.ino;
|
|
|
|
StatBuf.st_dev = Data.dev;
|
|
|
|
StatBuf.st_mtime = Data.mtime;
|
|
|
|
StatBuf.st_mode = Data.mode;
|
|
|
|
StatBuf.st_size = Data.size;
|
2010-11-24 04:05:15 +08:00
|
|
|
return CacheExists;
|
2009-04-28 02:38:38 +08:00
|
|
|
}
|
|
|
|
};
|
|
|
|
} // end anonymous namespace
|
|
|
|
|
|
|
|
|
2010-07-20 04:52:06 +08:00
|
|
|
/// \brief Read a source manager block
|
2011-12-01 07:21:26 +08:00
|
|
|
ASTReader::ASTReadResult ASTReader::ReadSourceManagerBlock(ModuleFile &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_SLOC_FILE_ENTRY:
|
|
|
|
case SM_SLOC_BUFFER_ENTRY:
|
2011-07-15 15:25:21 +08:00
|
|
|
case SM_SLOC_EXPANSION_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
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-02-16 01:54:22 +08:00
|
|
|
/// \brief If a header file is not found at the path that we expect it to be
|
|
|
|
/// and the PCH file was moved from its original location, try to resolve the
|
|
|
|
/// file by assuming that header+PCH were moved together and the header is in
|
|
|
|
/// the same place relative to the PCH.
|
|
|
|
static std::string
|
|
|
|
resolveFileRelativeToOriginalDir(const std::string &Filename,
|
|
|
|
const std::string &OriginalDir,
|
|
|
|
const std::string &CurrDir) {
|
|
|
|
assert(OriginalDir != CurrDir &&
|
|
|
|
"No point trying to resolve the file if the PCH dir didn't change");
|
|
|
|
using namespace llvm::sys;
|
|
|
|
llvm::SmallString<128> filePath(Filename);
|
|
|
|
fs::make_absolute(filePath);
|
|
|
|
assert(path::is_absolute(OriginalDir));
|
|
|
|
llvm::SmallString<128> currPCHPath(CurrDir);
|
|
|
|
|
|
|
|
path::const_iterator fileDirI = path::begin(path::parent_path(filePath)),
|
|
|
|
fileDirE = path::end(path::parent_path(filePath));
|
|
|
|
path::const_iterator origDirI = path::begin(OriginalDir),
|
|
|
|
origDirE = path::end(OriginalDir);
|
|
|
|
// Skip the common path components from filePath and OriginalDir.
|
|
|
|
while (fileDirI != fileDirE && origDirI != origDirE &&
|
|
|
|
*fileDirI == *origDirI) {
|
|
|
|
++fileDirI;
|
|
|
|
++origDirI;
|
|
|
|
}
|
|
|
|
for (; origDirI != origDirE; ++origDirI)
|
|
|
|
path::append(currPCHPath, "..");
|
|
|
|
path::append(currPCHPath, fileDirI, fileDirE);
|
|
|
|
path::append(currPCHPath, path::filename(Filename));
|
|
|
|
return currPCHPath.str();
|
|
|
|
}
|
|
|
|
|
2009-04-27 14:38:32 +08:00
|
|
|
/// \brief Read in the source location entry with the given ID.
|
Revamp the SourceManager to separate the representation of parsed
source locations from source locations loaded from an AST/PCH file.
Previously, loading an AST/PCH file involved carefully pre-allocating
space at the beginning of the source manager for the source locations
and FileIDs that correspond to the prefix, and then appending the
source locations/FileIDs used for parsing the remaining translation
unit. This design forced us into loading PCH files early, as a prefix,
whic has become a rather significant limitation.
This patch splits the SourceManager space into two parts: for source
location "addresses", the lower values (growing upward) are used to
describe parsed code, while upper values (growing downward) are used
for source locations loaded from AST/PCH files. Similarly, positive
FileIDs are used to describe parsed code while negative FileIDs are
used to file/macro locations loaded from AST/PCH files. As a result,
we can load PCH/AST files even during parsing, making various
improvemnts in the future possible, e.g., teaching #include <foo.h> to
look for and load <foo.h.gch> if it happens to be already available.
This patch was originally written by Sebastian Redl, then brought
forward to the modern age by Jonathan Turner, and finally
polished/finished by me to be committed.
llvm-svn: 135484
2011-07-20 00:10:42 +08:00
|
|
|
ASTReader::ASTReadResult ASTReader::ReadSLocEntryRecord(int ID) {
|
2009-04-27 14:38:32 +08:00
|
|
|
if (ID == 0)
|
|
|
|
return Success;
|
|
|
|
|
2011-07-22 02:46:38 +08:00
|
|
|
if (unsigned(-ID) - 2 >= getTotalNumSLocs() || ID > 0) {
|
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;
|
|
|
|
}
|
|
|
|
|
2011-12-01 07:21:26 +08:00
|
|
|
ModuleFile *F = GlobalSLocEntryMap.find(-ID)->second;
|
Revamp the SourceManager to separate the representation of parsed
source locations from source locations loaded from an AST/PCH file.
Previously, loading an AST/PCH file involved carefully pre-allocating
space at the beginning of the source manager for the source locations
and FileIDs that correspond to the prefix, and then appending the
source locations/FileIDs used for parsing the remaining translation
unit. This design forced us into loading PCH files early, as a prefix,
whic has become a rather significant limitation.
This patch splits the SourceManager space into two parts: for source
location "addresses", the lower values (growing upward) are used to
describe parsed code, while upper values (growing downward) are used
for source locations loaded from AST/PCH files. Similarly, positive
FileIDs are used to describe parsed code while negative FileIDs are
used to file/macro locations loaded from AST/PCH files. As a result,
we can load PCH/AST files even during parsing, making various
improvemnts in the future possible, e.g., teaching #include <foo.h> to
look for and load <foo.h.gch> if it happens to be already available.
This patch was originally written by Sebastian Redl, then brought
forward to the modern age by Jonathan Turner, and finally
polished/finished by me to be committed.
llvm-svn: 135484
2011-07-20 00:10:42 +08:00
|
|
|
F->SLocEntryCursor.JumpToBit(F->SLocEntryOffsets[ID - F->SLocEntryBaseID]);
|
2010-10-05 23:59:54 +08:00
|
|
|
llvm::BitstreamCursor &SLocEntryCursor = F->SLocEntryCursor;
|
Revamp the SourceManager to separate the representation of parsed
source locations from source locations loaded from an AST/PCH file.
Previously, loading an AST/PCH file involved carefully pre-allocating
space at the beginning of the source manager for the source locations
and FileIDs that correspond to the prefix, and then appending the
source locations/FileIDs used for parsing the remaining translation
unit. This design forced us into loading PCH files early, as a prefix,
whic has become a rather significant limitation.
This patch splits the SourceManager space into two parts: for source
location "addresses", the lower values (growing upward) are used to
describe parsed code, while upper values (growing downward) are used
for source locations loaded from AST/PCH files. Similarly, positive
FileIDs are used to describe parsed code while negative FileIDs are
used to file/macro locations loaded from AST/PCH files. As a result,
we can load PCH/AST files even during parsing, making various
improvemnts in the future possible, e.g., teaching #include <foo.h> to
look for and load <foo.h.gch> if it happens to be already available.
This patch was originally written by Sebastian Redl, then brought
forward to the modern age by Jonathan Turner, and finally
polished/finished by me to be committed.
llvm-svn: 135484
2011-07-20 00:10:42 +08:00
|
|
|
unsigned BaseOffset = F->SLocEntryBaseOffset;
|
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: {
|
2011-11-17 04:05:18 +08:00
|
|
|
if (Record.size() < 7) {
|
|
|
|
Error("source location entry is incorrect");
|
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool OverriddenBuffer = Record[6];
|
|
|
|
|
|
|
|
std::string OrigFilename(BlobStart, BlobStart + BlobLen);
|
|
|
|
std::string Filename = OrigFilename;
|
2009-07-07 08:12:59 +08:00
|
|
|
MaybeAddSystemRootToFilename(Filename);
|
2011-11-18 03:08:51 +08:00
|
|
|
const FileEntry *File =
|
|
|
|
OverriddenBuffer? FileMgr.getVirtualFile(Filename, (off_t)Record[4],
|
|
|
|
(time_t)Record[5])
|
|
|
|
: FileMgr.getFile(Filename, /*OpenFile=*/false);
|
2011-02-16 01:54:22 +08:00
|
|
|
if (File == 0 && !OriginalDir.empty() && !CurrentDir.empty() &&
|
|
|
|
OriginalDir != CurrentDir) {
|
|
|
|
std::string resolved = resolveFileRelativeToOriginalDir(Filename,
|
|
|
|
OriginalDir,
|
|
|
|
CurrentDir);
|
|
|
|
if (!resolved.empty())
|
|
|
|
File = FileMgr.getFile(resolved);
|
|
|
|
}
|
2011-01-27 18:55:51 +08:00
|
|
|
if (File == 0)
|
|
|
|
File = FileMgr.getVirtualFile(Filename, (off_t)Record[4],
|
|
|
|
(time_t)Record[5]);
|
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-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
|
|
|
)) {
|
2011-04-26 06:23:56 +08:00
|
|
|
Error(diag::err_fe_pch_file_modified, Filename);
|
2010-03-22 06:49:54 +08:00
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
|
Revamp the SourceManager to separate the representation of parsed
source locations from source locations loaded from an AST/PCH file.
Previously, loading an AST/PCH file involved carefully pre-allocating
space at the beginning of the source manager for the source locations
and FileIDs that correspond to the prefix, and then appending the
source locations/FileIDs used for parsing the remaining translation
unit. This design forced us into loading PCH files early, as a prefix,
whic has become a rather significant limitation.
This patch splits the SourceManager space into two parts: for source
location "addresses", the lower values (growing upward) are used to
describe parsed code, while upper values (growing downward) are used
for source locations loaded from AST/PCH files. Similarly, positive
FileIDs are used to describe parsed code while negative FileIDs are
used to file/macro locations loaded from AST/PCH files. As a result,
we can load PCH/AST files even during parsing, making various
improvemnts in the future possible, e.g., teaching #include <foo.h> to
look for and load <foo.h.gch> if it happens to be already available.
This patch was originally written by Sebastian Redl, then brought
forward to the modern age by Jonathan Turner, and finally
polished/finished by me to be committed.
llvm-svn: 135484
2011-07-20 00:10:42 +08:00
|
|
|
SourceLocation IncludeLoc = ReadSourceLocation(*F, Record[1]);
|
2011-07-23 00:00:58 +08:00
|
|
|
if (IncludeLoc.isInvalid() && F->Kind != MK_MainFile) {
|
Revamp the SourceManager to separate the representation of parsed
source locations from source locations loaded from an AST/PCH file.
Previously, loading an AST/PCH file involved carefully pre-allocating
space at the beginning of the source manager for the source locations
and FileIDs that correspond to the prefix, and then appending the
source locations/FileIDs used for parsing the remaining translation
unit. This design forced us into loading PCH files early, as a prefix,
whic has become a rather significant limitation.
This patch splits the SourceManager space into two parts: for source
location "addresses", the lower values (growing upward) are used to
describe parsed code, while upper values (growing downward) are used
for source locations loaded from AST/PCH files. Similarly, positive
FileIDs are used to describe parsed code while negative FileIDs are
used to file/macro locations loaded from AST/PCH files. As a result,
we can load PCH/AST files even during parsing, making various
improvemnts in the future possible, e.g., teaching #include <foo.h> to
look for and load <foo.h.gch> if it happens to be already available.
This patch was originally written by Sebastian Redl, then brought
forward to the modern age by Jonathan Turner, and finally
polished/finished by me to be committed.
llvm-svn: 135484
2011-07-20 00:10:42 +08:00
|
|
|
// This is the module's main file.
|
|
|
|
IncludeLoc = getImportLocation(F);
|
|
|
|
}
|
|
|
|
FileID FID = SourceMgr.createFileID(File, IncludeLoc,
|
2010-11-23 17:19:42 +08:00
|
|
|
(SrcMgr::CharacteristicKind)Record[2],
|
Revamp the SourceManager to separate the representation of parsed
source locations from source locations loaded from an AST/PCH file.
Previously, loading an AST/PCH file involved carefully pre-allocating
space at the beginning of the source manager for the source locations
and FileIDs that correspond to the prefix, and then appending the
source locations/FileIDs used for parsing the remaining translation
unit. This design forced us into loading PCH files early, as a prefix,
whic has become a rather significant limitation.
This patch splits the SourceManager space into two parts: for source
location "addresses", the lower values (growing upward) are used to
describe parsed code, while upper values (growing downward) are used
for source locations loaded from AST/PCH files. Similarly, positive
FileIDs are used to describe parsed code while negative FileIDs are
used to file/macro locations loaded from AST/PCH files. As a result,
we can load PCH/AST files even during parsing, making various
improvemnts in the future possible, e.g., teaching #include <foo.h> to
look for and load <foo.h.gch> if it happens to be already available.
This patch was originally written by Sebastian Redl, then brought
forward to the modern age by Jonathan Turner, and finally
polished/finished by me to be committed.
llvm-svn: 135484
2011-07-20 00:10:42 +08:00
|
|
|
ID, BaseOffset + Record[0]);
|
2011-08-22 07:33:04 +08:00
|
|
|
SrcMgr::FileInfo &FileInfo =
|
|
|
|
const_cast<SrcMgr::FileInfo&>(SourceMgr.getSLocEntry(FID).getFile());
|
2011-11-17 04:05:18 +08:00
|
|
|
FileInfo.NumCreatedFIDs = Record[7];
|
2009-04-27 14:38:32 +08:00
|
|
|
if (Record[3])
|
2011-08-22 07:33:04 +08:00
|
|
|
FileInfo.setHasLineDirectives();
|
2011-10-29 06:54:21 +08:00
|
|
|
|
2011-11-17 04:05:18 +08:00
|
|
|
const DeclID *FirstDecl = F->FileSortedDecls + Record[8];
|
|
|
|
unsigned NumFileDecls = Record[9];
|
2011-10-29 06:54:21 +08:00
|
|
|
if (NumFileDecls) {
|
|
|
|
assert(F->FileSortedDecls && "FILE_SORTED_DECLS not encountered yet ?");
|
2011-10-31 15:20:08 +08:00
|
|
|
FileDeclIDs[FID] = FileDeclsInfo(F, llvm::makeArrayRef(FirstDecl,
|
|
|
|
NumFileDecls));
|
2011-10-29 06:54:21 +08:00
|
|
|
}
|
Implement two related optimizations that make de-serialization of
AST/PCH files more lazy:
- Don't preload all of the file source-location entries when reading
the AST file. Instead, load them lazily, when needed.
- Only look up header-search information (whether a header was already
#import'd, how many times it's been included, etc.) when it's needed
by the preprocessor, rather than pre-populating it.
Previously, we would pre-load all of the file source-location entries,
which also populated the header-search information structure. This was
a relatively minor performance issue, since we would end up stat()'ing
all of the headers stored within a AST/PCH file when the AST/PCH file
was loaded. In the normal PCH use case, the stat()s were cached, so
the cost--of preloading ~860 source-location entries in the Cocoa.h
case---was relatively low.
However, the recent optimization that replaced stat+open with
open+fstat turned this into a major problem, since the preloading of
source-location entries would now end up opening those files. Worse,
those files wouldn't be closed until the file manager was destroyed,
so just opening a Cocoa.h PCH file would hold on to ~860 file
descriptors, and it was easy to blow through the process's limit on
the number of open file descriptors.
By eliminating the preloading of these files, we neither open nor stat
the headers stored in the PCH/AST file until they're actually needed
for something. Concretely, we went from
*** HeaderSearch Stats:
835 files tracked.
364 #import/#pragma once files.
823 included exactly once.
6 max times a file is included.
3 #include/#include_next/#import.
0 #includes skipped due to the multi-include optimization.
1 framework lookups.
0 subframework lookups.
*** Source Manager Stats:
835 files mapped, 3 mem buffers mapped.
37460 SLocEntry's allocated, 11215575B of Sloc address space used.
62 bytes of files mapped, 0 files with line #'s computed.
with a trivial program that uses a chained PCH including a Cocoa PCH
to
*** HeaderSearch Stats:
4 files tracked.
1 #import/#pragma once files.
3 included exactly once.
2 max times a file is included.
3 #include/#include_next/#import.
0 #includes skipped due to the multi-include optimization.
1 framework lookups.
0 subframework lookups.
*** Source Manager Stats:
3 files mapped, 3 mem buffers mapped.
37460 SLocEntry's allocated, 11215575B of Sloc address space used.
62 bytes of files mapped, 0 files with line #'s computed.
for the same program.
llvm-svn: 125286
2011-02-11 01:09:37 +08:00
|
|
|
|
2011-11-17 09:44:33 +08:00
|
|
|
const SrcMgr::ContentCache *ContentCache
|
|
|
|
= SourceMgr.getOrCreateContentCache(File);
|
|
|
|
if (OverriddenBuffer && !ContentCache->BufferOverridden &&
|
|
|
|
ContentCache->ContentsEntry == ContentCache->OrigEntry) {
|
2011-11-17 04:05:18 +08:00
|
|
|
unsigned Code = SLocEntryCursor.ReadCode();
|
|
|
|
Record.clear();
|
|
|
|
unsigned RecCode
|
|
|
|
= SLocEntryCursor.ReadRecord(Code, Record, &BlobStart, &BlobLen);
|
|
|
|
|
|
|
|
if (RecCode != SM_SLOC_BUFFER_BLOB) {
|
|
|
|
Error("AST record has invalid code");
|
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
|
|
|
|
llvm::MemoryBuffer *Buffer
|
|
|
|
= llvm::MemoryBuffer::getMemBuffer(StringRef(BlobStart, BlobLen - 1),
|
|
|
|
Filename);
|
|
|
|
SourceMgr.overrideFileContents(File, Buffer);
|
|
|
|
}
|
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
|
2011-11-17 04:05:18 +08:00
|
|
|
= llvm::MemoryBuffer::getMemBuffer(StringRef(BlobStart, BlobLen - 1),
|
|
|
|
Name);
|
Revamp the SourceManager to separate the representation of parsed
source locations from source locations loaded from an AST/PCH file.
Previously, loading an AST/PCH file involved carefully pre-allocating
space at the beginning of the source manager for the source locations
and FileIDs that correspond to the prefix, and then appending the
source locations/FileIDs used for parsing the remaining translation
unit. This design forced us into loading PCH files early, as a prefix,
whic has become a rather significant limitation.
This patch splits the SourceManager space into two parts: for source
location "addresses", the lower values (growing upward) are used to
describe parsed code, while upper values (growing downward) are used
for source locations loaded from AST/PCH files. Similarly, positive
FileIDs are used to describe parsed code while negative FileIDs are
used to file/macro locations loaded from AST/PCH files. As a result,
we can load PCH/AST files even during parsing, making various
improvemnts in the future possible, e.g., teaching #include <foo.h> to
look for and load <foo.h.gch> if it happens to be already available.
This patch was originally written by Sebastian Redl, then brought
forward to the modern age by Jonathan Turner, and finally
polished/finished by me to be committed.
llvm-svn: 135484
2011-07-20 00:10:42 +08:00
|
|
|
FileID BufferID = SourceMgr.createFileIDForMemBuffer(Buffer, ID,
|
|
|
|
BaseOffset + Offset);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2011-12-03 05:56:05 +08:00
|
|
|
if (strcmp(Name, "<built-in>") == 0 && F->Kind == MK_PCH) {
|
2010-07-15 01:49:11 +08:00
|
|
|
PCHPredefinesBlock Block = {
|
|
|
|
BufferID,
|
2011-07-23 18:55:15 +08:00
|
|
|
StringRef(BlobStart, BlobLen - 1)
|
2010-07-15 01:49:11 +08:00
|
|
|
};
|
|
|
|
PCHPredefinesBuffers.push_back(Block);
|
2009-04-29 04:33:11 +08:00
|
|
|
}
|
2009-04-27 14:38:32 +08:00
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2011-07-15 15:25:21 +08:00
|
|
|
case SM_SLOC_EXPANSION_ENTRY: {
|
2010-10-05 23:59:54 +08:00
|
|
|
SourceLocation SpellingLoc = ReadSourceLocation(*F, Record[1]);
|
2011-07-26 11:03:05 +08:00
|
|
|
SourceMgr.createExpansionLoc(SpellingLoc,
|
2010-10-05 23:59:54 +08:00
|
|
|
ReadSourceLocation(*F, Record[2]),
|
|
|
|
ReadSourceLocation(*F, Record[3]),
|
2009-04-27 14:38:32 +08:00
|
|
|
Record[4],
|
|
|
|
ID,
|
Revamp the SourceManager to separate the representation of parsed
source locations from source locations loaded from an AST/PCH file.
Previously, loading an AST/PCH file involved carefully pre-allocating
space at the beginning of the source manager for the source locations
and FileIDs that correspond to the prefix, and then appending the
source locations/FileIDs used for parsing the remaining translation
unit. This design forced us into loading PCH files early, as a prefix,
whic has become a rather significant limitation.
This patch splits the SourceManager space into two parts: for source
location "addresses", the lower values (growing upward) are used to
describe parsed code, while upper values (growing downward) are used
for source locations loaded from AST/PCH files. Similarly, positive
FileIDs are used to describe parsed code while negative FileIDs are
used to file/macro locations loaded from AST/PCH files. As a result,
we can load PCH/AST files even during parsing, making various
improvemnts in the future possible, e.g., teaching #include <foo.h> to
look for and load <foo.h.gch> if it happens to be already available.
This patch was originally written by Sebastian Redl, then brought
forward to the modern age by Jonathan Turner, and finally
polished/finished by me to be committed.
llvm-svn: 135484
2011-07-20 00:10:42 +08:00
|
|
|
BaseOffset + Record[0]);
|
2009-04-27 14:38:32 +08:00
|
|
|
break;
|
2009-09-09 23:08:12 +08:00
|
|
|
}
|
2009-04-27 14:38:32 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return Success;
|
|
|
|
}
|
|
|
|
|
Revamp the SourceManager to separate the representation of parsed
source locations from source locations loaded from an AST/PCH file.
Previously, loading an AST/PCH file involved carefully pre-allocating
space at the beginning of the source manager for the source locations
and FileIDs that correspond to the prefix, and then appending the
source locations/FileIDs used for parsing the remaining translation
unit. This design forced us into loading PCH files early, as a prefix,
whic has become a rather significant limitation.
This patch splits the SourceManager space into two parts: for source
location "addresses", the lower values (growing upward) are used to
describe parsed code, while upper values (growing downward) are used
for source locations loaded from AST/PCH files. Similarly, positive
FileIDs are used to describe parsed code while negative FileIDs are
used to file/macro locations loaded from AST/PCH files. As a result,
we can load PCH/AST files even during parsing, making various
improvemnts in the future possible, e.g., teaching #include <foo.h> to
look for and load <foo.h.gch> if it happens to be already available.
This patch was originally written by Sebastian Redl, then brought
forward to the modern age by Jonathan Turner, and finally
polished/finished by me to be committed.
llvm-svn: 135484
2011-07-20 00:10:42 +08:00
|
|
|
/// \brief Find the location where the module F is imported.
|
2011-12-01 07:21:26 +08:00
|
|
|
SourceLocation ASTReader::getImportLocation(ModuleFile *F) {
|
Revamp the SourceManager to separate the representation of parsed
source locations from source locations loaded from an AST/PCH file.
Previously, loading an AST/PCH file involved carefully pre-allocating
space at the beginning of the source manager for the source locations
and FileIDs that correspond to the prefix, and then appending the
source locations/FileIDs used for parsing the remaining translation
unit. This design forced us into loading PCH files early, as a prefix,
whic has become a rather significant limitation.
This patch splits the SourceManager space into two parts: for source
location "addresses", the lower values (growing upward) are used to
describe parsed code, while upper values (growing downward) are used
for source locations loaded from AST/PCH files. Similarly, positive
FileIDs are used to describe parsed code while negative FileIDs are
used to file/macro locations loaded from AST/PCH files. As a result,
we can load PCH/AST files even during parsing, making various
improvemnts in the future possible, e.g., teaching #include <foo.h> to
look for and load <foo.h.gch> if it happens to be already available.
This patch was originally written by Sebastian Redl, then brought
forward to the modern age by Jonathan Turner, and finally
polished/finished by me to be committed.
llvm-svn: 135484
2011-07-20 00:10:42 +08:00
|
|
|
if (F->ImportLoc.isValid())
|
|
|
|
return F->ImportLoc;
|
2011-07-30 02:09:09 +08:00
|
|
|
|
Revamp the SourceManager to separate the representation of parsed
source locations from source locations loaded from an AST/PCH file.
Previously, loading an AST/PCH file involved carefully pre-allocating
space at the beginning of the source manager for the source locations
and FileIDs that correspond to the prefix, and then appending the
source locations/FileIDs used for parsing the remaining translation
unit. This design forced us into loading PCH files early, as a prefix,
whic has become a rather significant limitation.
This patch splits the SourceManager space into two parts: for source
location "addresses", the lower values (growing upward) are used to
describe parsed code, while upper values (growing downward) are used
for source locations loaded from AST/PCH files. Similarly, positive
FileIDs are used to describe parsed code while negative FileIDs are
used to file/macro locations loaded from AST/PCH files. As a result,
we can load PCH/AST files even during parsing, making various
improvemnts in the future possible, e.g., teaching #include <foo.h> to
look for and load <foo.h.gch> if it happens to be already available.
This patch was originally written by Sebastian Redl, then brought
forward to the modern age by Jonathan Turner, and finally
polished/finished by me to be committed.
llvm-svn: 135484
2011-07-20 00:10:42 +08:00
|
|
|
// Otherwise we have a PCH. It's considered to be "imported" at the first
|
|
|
|
// location of its includer.
|
2011-07-30 02:09:09 +08:00
|
|
|
if (F->ImportedBy.empty() || !F->ImportedBy[0]) {
|
Revamp the SourceManager to separate the representation of parsed
source locations from source locations loaded from an AST/PCH file.
Previously, loading an AST/PCH file involved carefully pre-allocating
space at the beginning of the source manager for the source locations
and FileIDs that correspond to the prefix, and then appending the
source locations/FileIDs used for parsing the remaining translation
unit. This design forced us into loading PCH files early, as a prefix,
whic has become a rather significant limitation.
This patch splits the SourceManager space into two parts: for source
location "addresses", the lower values (growing upward) are used to
describe parsed code, while upper values (growing downward) are used
for source locations loaded from AST/PCH files. Similarly, positive
FileIDs are used to describe parsed code while negative FileIDs are
used to file/macro locations loaded from AST/PCH files. As a result,
we can load PCH/AST files even during parsing, making various
improvemnts in the future possible, e.g., teaching #include <foo.h> to
look for and load <foo.h.gch> if it happens to be already available.
This patch was originally written by Sebastian Redl, then brought
forward to the modern age by Jonathan Turner, and finally
polished/finished by me to be committed.
llvm-svn: 135484
2011-07-20 00:10:42 +08:00
|
|
|
// Main file is the importer. We assume that it is the first entry in the
|
|
|
|
// entry table. We can't ask the manager, because at the time of PCH loading
|
|
|
|
// the main file entry doesn't exist yet.
|
|
|
|
// The very first entry is the invalid instantiation loc, which takes up
|
|
|
|
// offsets 0 and 1.
|
|
|
|
return SourceLocation::getFromRawEncoding(2U);
|
|
|
|
}
|
2011-07-30 02:09:09 +08:00
|
|
|
//return F->Loaders[0]->FirstLoc;
|
|
|
|
return F->ImportedBy[0]->FirstLoc;
|
Revamp the SourceManager to separate the representation of parsed
source locations from source locations loaded from an AST/PCH file.
Previously, loading an AST/PCH file involved carefully pre-allocating
space at the beginning of the source manager for the source locations
and FileIDs that correspond to the prefix, and then appending the
source locations/FileIDs used for parsing the remaining translation
unit. This design forced us into loading PCH files early, as a prefix,
whic has become a rather significant limitation.
This patch splits the SourceManager space into two parts: for source
location "addresses", the lower values (growing upward) are used to
describe parsed code, while upper values (growing downward) are used
for source locations loaded from AST/PCH files. Similarly, positive
FileIDs are used to describe parsed code while negative FileIDs are
used to file/macro locations loaded from AST/PCH files. As a result,
we can load PCH/AST files even during parsing, making various
improvemnts in the future possible, e.g., teaching #include <foo.h> to
look for and load <foo.h.gch> if it happens to be already available.
This patch was originally written by Sebastian Redl, then brought
forward to the modern age by Jonathan Turner, and finally
polished/finished by me to be committed.
llvm-svn: 135484
2011-07-20 00:10:42 +08:00
|
|
|
}
|
|
|
|
|
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) {
|
2010-10-21 06:00:55 +08:00
|
|
|
uint64_t Offset = Cursor.GetCurrentBitNo();
|
2009-04-27 09:05:14 +08:00
|
|
|
unsigned Code = Cursor.ReadCode();
|
2010-10-21 11:16:25 +08:00
|
|
|
|
2009-04-27 09:05:14 +08:00
|
|
|
// We expect all abbrevs to be at the start of the block.
|
2010-10-21 06:00:55 +08:00
|
|
|
if (Code != llvm::bitc::DEFINE_ABBREV) {
|
|
|
|
Cursor.JumpToBit(Offset);
|
2009-04-27 09:05:14 +08:00
|
|
|
return false;
|
2010-10-21 06:00:55 +08:00
|
|
|
}
|
2009-04-27 09:05:14 +08:00
|
|
|
Cursor.ReadAbbrevRecord();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-12-01 07:21:26 +08:00
|
|
|
void ASTReader::ReadMacroRecord(ModuleFile &F, uint64_t Offset) {
|
2010-10-21 06:00:55 +08:00
|
|
|
llvm::BitstreamCursor &Stream = F.MacroCursor;
|
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;
|
2011-07-23 18:55:15 +08:00
|
|
|
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:
|
2011-08-05 02:09:14 +08:00
|
|
|
return;
|
2009-04-22 07:56:24 +08:00
|
|
|
|
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");
|
2011-08-05 02:09:14 +08:00
|
|
|
return;
|
2009-04-22 07:56:24 +08:00
|
|
|
}
|
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.
|
2010-10-21 06:00:55 +08:00
|
|
|
const char *BlobStart = 0;
|
|
|
|
unsigned BlobLen = 0;
|
2009-04-11 05:41:48 +08:00
|
|
|
Record.clear();
|
2010-08-19 07:57:32 +08:00
|
|
|
PreprocessorRecordTypes RecType =
|
2010-10-21 11:16:25 +08:00
|
|
|
(PreprocessorRecordTypes)Stream.ReadRecord(Code, Record, BlobStart,
|
2010-10-21 06:00:55 +08:00
|
|
|
BlobLen);
|
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)
|
2011-08-05 02:09:14 +08:00
|
|
|
return;
|
2009-04-22 07:56:24 +08:00
|
|
|
|
2011-07-29 04:55:49 +08:00
|
|
|
IdentifierInfo *II = getLocalIdentifier(F, 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");
|
2011-08-05 02:09:14 +08:00
|
|
|
return;
|
2009-04-22 07:56:24 +08:00
|
|
|
}
|
Make the loading of information attached to an IdentifierInfo from an
AST file more lazy, so that we don't eagerly load that information for
all known identifiers each time a new AST file is loaded. The eager
reloading made some sense in the context of precompiled headers, since
very few identifiers were defined before PCH load time. With modules,
however, a huge amount of code can get parsed before we see an
@import, so laziness becomes important here.
The approach taken to make this information lazy is fairly simple:
when we load a new AST file, we mark all of the existing identifiers
as being out-of-date. Whenever we want to access information that may
come from an AST (e.g., whether the identifier has a macro definition,
or what top-level declarations have that name), we check the
out-of-date bit and, if it's set, ask the AST reader to update the
IdentifierInfo from the AST files. The update is a merge, and we now
take care to merge declarations before/after imports with declarations
from multiple imports.
The results of this optimization are fairly dramatic. On a small
application that brings in 14 non-trivial modules, this takes modules
from being > 3x slower than a "perfect" PCH file down to 30% slower
for a full rebuild. A partial rebuild (where the PCH file or modules
can be re-used) is down to 7% slower. Making the PCH file just a
little imperfect (e.g., adding two smallish modules used by a bunch of
.m files that aren't in the PCH file) tips the scales in favor of the
modules approach, with 24% faster partial rebuilds.
This is just a first step; the lazy scheme could possibly be improved
by adding versioning, so we don't search into modules we already
searched. Moreover, we'll need similar lazy schemes for all of the
other lookup data structures, such as DeclContexts.
llvm-svn: 143100
2011-10-27 17:33:13 +08:00
|
|
|
|
2010-10-05 23:59:54 +08:00
|
|
|
SourceLocation Loc = ReadSourceLocation(F, Record[1]);
|
2009-04-11 05:41:48 +08:00
|
|
|
bool isUsed = Record[2];
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2011-09-10 06:02:16 +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
|
|
|
|
2011-10-17 23:32:29 +08:00
|
|
|
bool IsPublic = Record[3];
|
|
|
|
unsigned NextIndex = 4;
|
|
|
|
MI->setVisibility(IsPublic, ReadSourceLocation(F, Record, NextIndex));
|
2011-09-02 01:04:32 +08:00
|
|
|
|
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.
|
2011-09-02 01:04:32 +08:00
|
|
|
bool isC99VarArgs = Record[NextIndex++];
|
|
|
|
bool isGNUVarArgs = Record[NextIndex++];
|
2009-04-11 05:41:48 +08:00
|
|
|
MacroArgs.clear();
|
2011-09-02 01:04:32 +08:00
|
|
|
unsigned NumArgs = Record[NextIndex++];
|
2009-04-11 05:41:48 +08:00
|
|
|
for (unsigned i = 0; i != NumArgs; ++i)
|
2011-09-02 01:04:32 +08:00
|
|
|
MacroArgs.push_back(getLocalIdentifier(F, Record[NextIndex++]));
|
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(),
|
2011-09-10 06:02:16 +08:00
|
|
|
PP.getPreprocessorAllocator());
|
2009-04-11 05:41:48 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Finally, install the macro.
|
2011-09-10 06:02:16 +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;
|
2010-10-21 11:16:25 +08:00
|
|
|
|
2011-09-16 02:02:56 +08:00
|
|
|
if (NextIndex + 1 == Record.size() && PP.getPreprocessingRecord() &&
|
|
|
|
Record[NextIndex]) {
|
|
|
|
// We have a macro definition. Register the association
|
|
|
|
PreprocessedEntityID
|
|
|
|
GlobalID = getGlobalPreprocessedEntityID(F, Record[NextIndex]);
|
|
|
|
PreprocessingRecord &PPRec = *PP.getPreprocessingRecord();
|
|
|
|
PPRec.RegisterMacroDefinition(Macro,
|
|
|
|
PPRec.getPPEntityID(GlobalID-1, /*isLoaded=*/true));
|
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-10-21 11:16:25 +08:00
|
|
|
|
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();
|
2010-10-05 23:59:54 +08:00
|
|
|
Tok.setLocation(ReadSourceLocation(F, Record[0]));
|
2009-04-11 05:41:48 +08:00
|
|
|
Tok.setLength(Record[1]);
|
2011-07-29 04:55:49 +08:00
|
|
|
if (IdentifierInfo *II = getLocalIdentifier(F, Record[2]))
|
2009-04-12 05:15:38 +08:00
|
|
|
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;
|
|
|
|
}
|
2011-02-09 05:58:10 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-08-05 02:09:14 +08:00
|
|
|
return;
|
2011-02-09 05:58:10 +08:00
|
|
|
}
|
2010-10-21 11:16:25 +08:00
|
|
|
|
2011-07-29 06:39:26 +08:00
|
|
|
PreprocessedEntityID
|
2011-12-01 07:21:26 +08:00
|
|
|
ASTReader::getGlobalPreprocessedEntityID(ModuleFile &M, unsigned LocalID) const {
|
2011-09-20 04:40:02 +08:00
|
|
|
ContinuousRangeMap<uint32_t, int, 2>::const_iterator
|
2011-08-05 02:56:47 +08:00
|
|
|
I = M.PreprocessedEntityRemap.find(LocalID - NUM_PREDEF_PP_ENTITY_IDS);
|
|
|
|
assert(I != M.PreprocessedEntityRemap.end()
|
|
|
|
&& "Invalid index into preprocessed entity index remap");
|
|
|
|
|
|
|
|
return LocalID + I->second;
|
2011-07-29 06:39:26 +08:00
|
|
|
}
|
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
unsigned HeaderFileInfoTrait::ComputeHash(const char *path) {
|
|
|
|
return llvm::HashString(llvm::sys::path::filename(path));
|
|
|
|
}
|
Implement two related optimizations that make de-serialization of
AST/PCH files more lazy:
- Don't preload all of the file source-location entries when reading
the AST file. Instead, load them lazily, when needed.
- Only look up header-search information (whether a header was already
#import'd, how many times it's been included, etc.) when it's needed
by the preprocessor, rather than pre-populating it.
Previously, we would pre-load all of the file source-location entries,
which also populated the header-search information structure. This was
a relatively minor performance issue, since we would end up stat()'ing
all of the headers stored within a AST/PCH file when the AST/PCH file
was loaded. In the normal PCH use case, the stat()s were cached, so
the cost--of preloading ~860 source-location entries in the Cocoa.h
case---was relatively low.
However, the recent optimization that replaced stat+open with
open+fstat turned this into a major problem, since the preloading of
source-location entries would now end up opening those files. Worse,
those files wouldn't be closed until the file manager was destroyed,
so just opening a Cocoa.h PCH file would hold on to ~860 file
descriptors, and it was easy to blow through the process's limit on
the number of open file descriptors.
By eliminating the preloading of these files, we neither open nor stat
the headers stored in the PCH/AST file until they're actually needed
for something. Concretely, we went from
*** HeaderSearch Stats:
835 files tracked.
364 #import/#pragma once files.
823 included exactly once.
6 max times a file is included.
3 #include/#include_next/#import.
0 #includes skipped due to the multi-include optimization.
1 framework lookups.
0 subframework lookups.
*** Source Manager Stats:
835 files mapped, 3 mem buffers mapped.
37460 SLocEntry's allocated, 11215575B of Sloc address space used.
62 bytes of files mapped, 0 files with line #'s computed.
with a trivial program that uses a chained PCH including a Cocoa PCH
to
*** HeaderSearch Stats:
4 files tracked.
1 #import/#pragma once files.
3 included exactly once.
2 max times a file is included.
3 #include/#include_next/#import.
0 #includes skipped due to the multi-include optimization.
1 framework lookups.
0 subframework lookups.
*** Source Manager Stats:
3 files mapped, 3 mem buffers mapped.
37460 SLocEntry's allocated, 11215575B of Sloc address space used.
62 bytes of files mapped, 0 files with line #'s computed.
for the same program.
llvm-svn: 125286
2011-02-11 01:09:37 +08:00
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
HeaderFileInfoTrait::internal_key_type
|
|
|
|
HeaderFileInfoTrait::GetInternalKey(const char *path) { return path; }
|
Implement two related optimizations that make de-serialization of
AST/PCH files more lazy:
- Don't preload all of the file source-location entries when reading
the AST file. Instead, load them lazily, when needed.
- Only look up header-search information (whether a header was already
#import'd, how many times it's been included, etc.) when it's needed
by the preprocessor, rather than pre-populating it.
Previously, we would pre-load all of the file source-location entries,
which also populated the header-search information structure. This was
a relatively minor performance issue, since we would end up stat()'ing
all of the headers stored within a AST/PCH file when the AST/PCH file
was loaded. In the normal PCH use case, the stat()s were cached, so
the cost--of preloading ~860 source-location entries in the Cocoa.h
case---was relatively low.
However, the recent optimization that replaced stat+open with
open+fstat turned this into a major problem, since the preloading of
source-location entries would now end up opening those files. Worse,
those files wouldn't be closed until the file manager was destroyed,
so just opening a Cocoa.h PCH file would hold on to ~860 file
descriptors, and it was easy to blow through the process's limit on
the number of open file descriptors.
By eliminating the preloading of these files, we neither open nor stat
the headers stored in the PCH/AST file until they're actually needed
for something. Concretely, we went from
*** HeaderSearch Stats:
835 files tracked.
364 #import/#pragma once files.
823 included exactly once.
6 max times a file is included.
3 #include/#include_next/#import.
0 #includes skipped due to the multi-include optimization.
1 framework lookups.
0 subframework lookups.
*** Source Manager Stats:
835 files mapped, 3 mem buffers mapped.
37460 SLocEntry's allocated, 11215575B of Sloc address space used.
62 bytes of files mapped, 0 files with line #'s computed.
with a trivial program that uses a chained PCH including a Cocoa PCH
to
*** HeaderSearch Stats:
4 files tracked.
1 #import/#pragma once files.
3 included exactly once.
2 max times a file is included.
3 #include/#include_next/#import.
0 #includes skipped due to the multi-include optimization.
1 framework lookups.
0 subframework lookups.
*** Source Manager Stats:
3 files mapped, 3 mem buffers mapped.
37460 SLocEntry's allocated, 11215575B of Sloc address space used.
62 bytes of files mapped, 0 files with line #'s computed.
for the same program.
llvm-svn: 125286
2011-02-11 01:09:37 +08:00
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
bool HeaderFileInfoTrait::EqualKey(internal_key_type a, internal_key_type b) {
|
|
|
|
if (strcmp(a, b) == 0)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if (llvm::sys::path::filename(a) != llvm::sys::path::filename(b))
|
|
|
|
return false;
|
2011-12-10 00:22:07 +08:00
|
|
|
|
|
|
|
// Determine whether the actual files are equivalent.
|
|
|
|
bool Result = false;
|
|
|
|
if (llvm::sys::fs::equivalent(a, b, Result))
|
2011-08-26 04:47:51 +08:00
|
|
|
return false;
|
|
|
|
|
2011-12-10 00:22:07 +08:00
|
|
|
return Result;
|
2011-08-26 04:47:51 +08:00
|
|
|
}
|
Implement two related optimizations that make de-serialization of
AST/PCH files more lazy:
- Don't preload all of the file source-location entries when reading
the AST file. Instead, load them lazily, when needed.
- Only look up header-search information (whether a header was already
#import'd, how many times it's been included, etc.) when it's needed
by the preprocessor, rather than pre-populating it.
Previously, we would pre-load all of the file source-location entries,
which also populated the header-search information structure. This was
a relatively minor performance issue, since we would end up stat()'ing
all of the headers stored within a AST/PCH file when the AST/PCH file
was loaded. In the normal PCH use case, the stat()s were cached, so
the cost--of preloading ~860 source-location entries in the Cocoa.h
case---was relatively low.
However, the recent optimization that replaced stat+open with
open+fstat turned this into a major problem, since the preloading of
source-location entries would now end up opening those files. Worse,
those files wouldn't be closed until the file manager was destroyed,
so just opening a Cocoa.h PCH file would hold on to ~860 file
descriptors, and it was easy to blow through the process's limit on
the number of open file descriptors.
By eliminating the preloading of these files, we neither open nor stat
the headers stored in the PCH/AST file until they're actually needed
for something. Concretely, we went from
*** HeaderSearch Stats:
835 files tracked.
364 #import/#pragma once files.
823 included exactly once.
6 max times a file is included.
3 #include/#include_next/#import.
0 #includes skipped due to the multi-include optimization.
1 framework lookups.
0 subframework lookups.
*** Source Manager Stats:
835 files mapped, 3 mem buffers mapped.
37460 SLocEntry's allocated, 11215575B of Sloc address space used.
62 bytes of files mapped, 0 files with line #'s computed.
with a trivial program that uses a chained PCH including a Cocoa PCH
to
*** HeaderSearch Stats:
4 files tracked.
1 #import/#pragma once files.
3 included exactly once.
2 max times a file is included.
3 #include/#include_next/#import.
0 #includes skipped due to the multi-include optimization.
1 framework lookups.
0 subframework lookups.
*** Source Manager Stats:
3 files mapped, 3 mem buffers mapped.
37460 SLocEntry's allocated, 11215575B of Sloc address space used.
62 bytes of files mapped, 0 files with line #'s computed.
for the same program.
llvm-svn: 125286
2011-02-11 01:09:37 +08:00
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
std::pair<unsigned, unsigned>
|
|
|
|
HeaderFileInfoTrait::ReadKeyDataLength(const unsigned char*& d) {
|
|
|
|
unsigned KeyLen = (unsigned) clang::io::ReadUnalignedLE16(d);
|
|
|
|
unsigned DataLen = (unsigned) *d++;
|
|
|
|
return std::make_pair(KeyLen + 1, DataLen);
|
|
|
|
}
|
Implement two related optimizations that make de-serialization of
AST/PCH files more lazy:
- Don't preload all of the file source-location entries when reading
the AST file. Instead, load them lazily, when needed.
- Only look up header-search information (whether a header was already
#import'd, how many times it's been included, etc.) when it's needed
by the preprocessor, rather than pre-populating it.
Previously, we would pre-load all of the file source-location entries,
which also populated the header-search information structure. This was
a relatively minor performance issue, since we would end up stat()'ing
all of the headers stored within a AST/PCH file when the AST/PCH file
was loaded. In the normal PCH use case, the stat()s were cached, so
the cost--of preloading ~860 source-location entries in the Cocoa.h
case---was relatively low.
However, the recent optimization that replaced stat+open with
open+fstat turned this into a major problem, since the preloading of
source-location entries would now end up opening those files. Worse,
those files wouldn't be closed until the file manager was destroyed,
so just opening a Cocoa.h PCH file would hold on to ~860 file
descriptors, and it was easy to blow through the process's limit on
the number of open file descriptors.
By eliminating the preloading of these files, we neither open nor stat
the headers stored in the PCH/AST file until they're actually needed
for something. Concretely, we went from
*** HeaderSearch Stats:
835 files tracked.
364 #import/#pragma once files.
823 included exactly once.
6 max times a file is included.
3 #include/#include_next/#import.
0 #includes skipped due to the multi-include optimization.
1 framework lookups.
0 subframework lookups.
*** Source Manager Stats:
835 files mapped, 3 mem buffers mapped.
37460 SLocEntry's allocated, 11215575B of Sloc address space used.
62 bytes of files mapped, 0 files with line #'s computed.
with a trivial program that uses a chained PCH including a Cocoa PCH
to
*** HeaderSearch Stats:
4 files tracked.
1 #import/#pragma once files.
3 included exactly once.
2 max times a file is included.
3 #include/#include_next/#import.
0 #includes skipped due to the multi-include optimization.
1 framework lookups.
0 subframework lookups.
*** Source Manager Stats:
3 files mapped, 3 mem buffers mapped.
37460 SLocEntry's allocated, 11215575B of Sloc address space used.
62 bytes of files mapped, 0 files with line #'s computed.
for the same program.
llvm-svn: 125286
2011-02-11 01:09:37 +08:00
|
|
|
|
2011-08-26 04:47:51 +08:00
|
|
|
HeaderFileInfoTrait::data_type
|
|
|
|
HeaderFileInfoTrait::ReadData(const internal_key_type, const unsigned char *d,
|
|
|
|
unsigned DataLen) {
|
|
|
|
const unsigned char *End = d + DataLen;
|
|
|
|
using namespace clang::io;
|
|
|
|
HeaderFileInfo HFI;
|
|
|
|
unsigned Flags = *d++;
|
|
|
|
HFI.isImport = (Flags >> 5) & 0x01;
|
|
|
|
HFI.isPragmaOnce = (Flags >> 4) & 0x01;
|
|
|
|
HFI.DirInfo = (Flags >> 2) & 0x03;
|
|
|
|
HFI.Resolved = (Flags >> 1) & 0x01;
|
|
|
|
HFI.IndexHeaderMapHeader = Flags & 0x01;
|
|
|
|
HFI.NumIncludes = ReadUnalignedLE16(d);
|
2011-10-18 02:53:12 +08:00
|
|
|
HFI.ControllingMacroID = Reader.getGlobalIdentifierID(M,
|
|
|
|
ReadUnalignedLE32(d));
|
2011-08-26 04:47:51 +08:00
|
|
|
if (unsigned FrameworkOffset = ReadUnalignedLE32(d)) {
|
|
|
|
// The framework offset is 1 greater than the actual offset,
|
|
|
|
// since 0 is used as an indicator for "no framework name".
|
|
|
|
StringRef FrameworkName(FrameworkStrings + FrameworkOffset - 1);
|
|
|
|
HFI.Framework = HS->getUniqueFrameworkName(FrameworkName);
|
|
|
|
}
|
|
|
|
|
|
|
|
assert(End == d && "Wrong data length in HeaderFileInfo deserialization");
|
|
|
|
(void)End;
|
|
|
|
|
|
|
|
// This HeaderFileInfo was externally loaded.
|
|
|
|
HFI.External = true;
|
|
|
|
return HFI;
|
Implement two related optimizations that make de-serialization of
AST/PCH files more lazy:
- Don't preload all of the file source-location entries when reading
the AST file. Instead, load them lazily, when needed.
- Only look up header-search information (whether a header was already
#import'd, how many times it's been included, etc.) when it's needed
by the preprocessor, rather than pre-populating it.
Previously, we would pre-load all of the file source-location entries,
which also populated the header-search information structure. This was
a relatively minor performance issue, since we would end up stat()'ing
all of the headers stored within a AST/PCH file when the AST/PCH file
was loaded. In the normal PCH use case, the stat()s were cached, so
the cost--of preloading ~860 source-location entries in the Cocoa.h
case---was relatively low.
However, the recent optimization that replaced stat+open with
open+fstat turned this into a major problem, since the preloading of
source-location entries would now end up opening those files. Worse,
those files wouldn't be closed until the file manager was destroyed,
so just opening a Cocoa.h PCH file would hold on to ~860 file
descriptors, and it was easy to blow through the process's limit on
the number of open file descriptors.
By eliminating the preloading of these files, we neither open nor stat
the headers stored in the PCH/AST file until they're actually needed
for something. Concretely, we went from
*** HeaderSearch Stats:
835 files tracked.
364 #import/#pragma once files.
823 included exactly once.
6 max times a file is included.
3 #include/#include_next/#import.
0 #includes skipped due to the multi-include optimization.
1 framework lookups.
0 subframework lookups.
*** Source Manager Stats:
835 files mapped, 3 mem buffers mapped.
37460 SLocEntry's allocated, 11215575B of Sloc address space used.
62 bytes of files mapped, 0 files with line #'s computed.
with a trivial program that uses a chained PCH including a Cocoa PCH
to
*** HeaderSearch Stats:
4 files tracked.
1 #import/#pragma once files.
3 included exactly once.
2 max times a file is included.
3 #include/#include_next/#import.
0 #includes skipped due to the multi-include optimization.
1 framework lookups.
0 subframework lookups.
*** Source Manager Stats:
3 files mapped, 3 mem buffers mapped.
37460 SLocEntry's allocated, 11215575B of Sloc address space used.
62 bytes of files mapped, 0 files with line #'s computed.
for the same program.
llvm-svn: 125286
2011-02-11 01:09:37 +08:00
|
|
|
}
|
|
|
|
|
2011-12-02 23:45:10 +08:00
|
|
|
void ASTReader::setIdentifierIsMacro(IdentifierInfo *II, ModuleFile &F,
|
|
|
|
uint64_t LocalOffset, bool Visible) {
|
|
|
|
if (Visible) {
|
|
|
|
// Note that this identifier has a macro definition.
|
|
|
|
II->setHasMacroDefinition(true);
|
|
|
|
}
|
2010-10-30 08:23:06 +08:00
|
|
|
|
Introduce a global bit-offset continuous range map into the ASTReader,
so that we have one, simple way to map from global bit offsets to
local bit offsets. Eliminates a number of loops over the chain, and
generalizes for more interesting bit remappings.
Also, as an amusing oddity, we were computing global bit offsets
*backwards* for preprocessed entities (e.g., the directly included PCH
file in the chain would start at offset zero, rather than the original
PCH that occurs first in translation unit). Even more amusingly, it
made precompiled preambles work, because we were forgetting to adjust
the local bit offset to a global bit offset when storing preprocessed
entity offsets in the ASTUnit. Two wrongs made a right, and now
they're both right.
llvm-svn: 135750
2011-07-22 14:10:01 +08:00
|
|
|
// Adjust the offset to a global offset.
|
2011-07-29 05:16:51 +08:00
|
|
|
UnreadMacroRecordOffsets[II] = F.GlobalBitOffset + LocalOffset;
|
2010-10-30 08:23:06 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:43 +08:00
|
|
|
void ASTReader::ReadDefinedMacros() {
|
2011-07-26 04:32:21 +08:00
|
|
|
for (ModuleReverseIterator I = ModuleMgr.rbegin(),
|
|
|
|
E = ModuleMgr.rend(); I != E; ++I) {
|
|
|
|
llvm::BitstreamCursor &MacroCursor = (*I)->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;
|
2011-07-26 04:32:21 +08:00
|
|
|
Cursor.JumpToBit((*I)->MacroStartOffset);
|
2010-10-21 11:16:25 +08:00
|
|
|
|
2010-07-22 06:31:37 +08:00
|
|
|
RecordData Record;
|
|
|
|
while (true) {
|
|
|
|
unsigned Code = Cursor.ReadCode();
|
2010-10-21 06:00:55 +08:00
|
|
|
if (Code == llvm::bitc::END_BLOCK)
|
2010-07-22 06:31:37 +08:00
|
|
|
break;
|
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:
|
2011-07-29 04:55:49 +08:00
|
|
|
getLocalIdentifier(**I, Record[0]);
|
2010-07-22 06:31:37 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case PP_TOKEN:
|
2010-07-22 06:31:37 +08:00
|
|
|
// Ignore tokens.
|
|
|
|
break;
|
|
|
|
}
|
2010-01-05 03:18:44 +08:00
|
|
|
}
|
|
|
|
}
|
2010-10-30 08:23:06 +08:00
|
|
|
|
|
|
|
// Drain the unread macro-record offsets map.
|
|
|
|
while (!UnreadMacroRecordOffsets.empty())
|
|
|
|
LoadMacroDefinition(UnreadMacroRecordOffsets.begin());
|
|
|
|
}
|
|
|
|
|
|
|
|
void ASTReader::LoadMacroDefinition(
|
|
|
|
llvm::DenseMap<IdentifierInfo *, uint64_t>::iterator Pos) {
|
|
|
|
assert(Pos != UnreadMacroRecordOffsets.end() && "Unknown macro definition");
|
|
|
|
uint64_t Offset = Pos->second;
|
|
|
|
UnreadMacroRecordOffsets.erase(Pos);
|
|
|
|
|
Introduce a global bit-offset continuous range map into the ASTReader,
so that we have one, simple way to map from global bit offsets to
local bit offsets. Eliminates a number of loops over the chain, and
generalizes for more interesting bit remappings.
Also, as an amusing oddity, we were computing global bit offsets
*backwards* for preprocessed entities (e.g., the directly included PCH
file in the chain would start at offset zero, rather than the original
PCH that occurs first in translation unit). Even more amusingly, it
made precompiled preambles work, because we were forgetting to adjust
the local bit offset to a global bit offset when storing preprocessed
entity offsets in the ASTUnit. Two wrongs made a right, and now
they're both right.
llvm-svn: 135750
2011-07-22 14:10:01 +08:00
|
|
|
RecordLocation Loc = getLocalBitOffset(Offset);
|
|
|
|
ReadMacroRecord(*Loc.F, Loc.Offset);
|
2010-10-30 08:23:06 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void ASTReader::LoadMacroDefinition(IdentifierInfo *II) {
|
|
|
|
llvm::DenseMap<IdentifierInfo *, uint64_t>::iterator Pos
|
|
|
|
= UnreadMacroRecordOffsets.find(II);
|
|
|
|
LoadMacroDefinition(Pos);
|
2010-01-05 03:18:44 +08:00
|
|
|
}
|
|
|
|
|
Make the loading of information attached to an IdentifierInfo from an
AST file more lazy, so that we don't eagerly load that information for
all known identifiers each time a new AST file is loaded. The eager
reloading made some sense in the context of precompiled headers, since
very few identifiers were defined before PCH load time. With modules,
however, a huge amount of code can get parsed before we see an
@import, so laziness becomes important here.
The approach taken to make this information lazy is fairly simple:
when we load a new AST file, we mark all of the existing identifiers
as being out-of-date. Whenever we want to access information that may
come from an AST (e.g., whether the identifier has a macro definition,
or what top-level declarations have that name), we check the
out-of-date bit and, if it's set, ask the AST reader to update the
IdentifierInfo from the AST files. The update is a merge, and we now
take care to merge declarations before/after imports with declarations
from multiple imports.
The results of this optimization are fairly dramatic. On a small
application that brings in 14 non-trivial modules, this takes modules
from being > 3x slower than a "perfect" PCH file down to 30% slower
for a full rebuild. A partial rebuild (where the PCH file or modules
can be re-used) is down to 7% slower. Making the PCH file just a
little imperfect (e.g., adding two smallish modules used by a bunch of
.m files that aren't in the PCH file) tips the scales in favor of the
modules approach, with 24% faster partial rebuilds.
This is just a first step; the lazy scheme could possibly be improved
by adding versioning, so we don't search into modules we already
searched. Moreover, we'll need similar lazy schemes for all of the
other lookup data structures, such as DeclContexts.
llvm-svn: 143100
2011-10-27 17:33:13 +08:00
|
|
|
namespace {
|
|
|
|
/// \brief Visitor class used to look up identifirs in an AST file.
|
|
|
|
class IdentifierLookupVisitor {
|
|
|
|
StringRef Name;
|
|
|
|
IdentifierInfo *Found;
|
|
|
|
public:
|
|
|
|
explicit IdentifierLookupVisitor(StringRef Name) : Name(Name), Found() { }
|
|
|
|
|
2011-12-01 07:21:26 +08:00
|
|
|
static bool visit(ModuleFile &M, void *UserData) {
|
Make the loading of information attached to an IdentifierInfo from an
AST file more lazy, so that we don't eagerly load that information for
all known identifiers each time a new AST file is loaded. The eager
reloading made some sense in the context of precompiled headers, since
very few identifiers were defined before PCH load time. With modules,
however, a huge amount of code can get parsed before we see an
@import, so laziness becomes important here.
The approach taken to make this information lazy is fairly simple:
when we load a new AST file, we mark all of the existing identifiers
as being out-of-date. Whenever we want to access information that may
come from an AST (e.g., whether the identifier has a macro definition,
or what top-level declarations have that name), we check the
out-of-date bit and, if it's set, ask the AST reader to update the
IdentifierInfo from the AST files. The update is a merge, and we now
take care to merge declarations before/after imports with declarations
from multiple imports.
The results of this optimization are fairly dramatic. On a small
application that brings in 14 non-trivial modules, this takes modules
from being > 3x slower than a "perfect" PCH file down to 30% slower
for a full rebuild. A partial rebuild (where the PCH file or modules
can be re-used) is down to 7% slower. Making the PCH file just a
little imperfect (e.g., adding two smallish modules used by a bunch of
.m files that aren't in the PCH file) tips the scales in favor of the
modules approach, with 24% faster partial rebuilds.
This is just a first step; the lazy scheme could possibly be improved
by adding versioning, so we don't search into modules we already
searched. Moreover, we'll need similar lazy schemes for all of the
other lookup data structures, such as DeclContexts.
llvm-svn: 143100
2011-10-27 17:33:13 +08:00
|
|
|
IdentifierLookupVisitor *This
|
|
|
|
= static_cast<IdentifierLookupVisitor *>(UserData);
|
|
|
|
|
|
|
|
ASTIdentifierLookupTable *IdTable
|
|
|
|
= (ASTIdentifierLookupTable *)M.IdentifierLookupTable;
|
|
|
|
if (!IdTable)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
std::pair<const char*, unsigned> Key(This->Name.begin(),
|
|
|
|
This->Name.size());
|
|
|
|
ASTIdentifierLookupTable::iterator Pos = IdTable->find(Key);
|
|
|
|
if (Pos == IdTable->end())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Dereferencing the iterator has the effect of building the
|
|
|
|
// IdentifierInfo node and populating it with the various
|
|
|
|
// declarations it needs.
|
|
|
|
This->Found = *Pos;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// \brief Retrieve the identifier info found within the module
|
|
|
|
// files.
|
|
|
|
IdentifierInfo *getIdentifierInfo() const { return Found; }
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
void ASTReader::updateOutOfDateIdentifier(IdentifierInfo &II) {
|
|
|
|
get(II.getName());
|
|
|
|
}
|
|
|
|
|
2011-07-23 18:55:15 +08:00
|
|
|
const FileEntry *ASTReader::getFileEntry(StringRef filenameStrRef) {
|
2011-06-01 13:43:53 +08:00
|
|
|
std::string Filename = filenameStrRef;
|
|
|
|
MaybeAddSystemRootToFilename(Filename);
|
|
|
|
const FileEntry *File = FileMgr.getFile(Filename);
|
|
|
|
if (File == 0 && !OriginalDir.empty() && !CurrentDir.empty() &&
|
|
|
|
OriginalDir != CurrentDir) {
|
|
|
|
std::string resolved = resolveFileRelativeToOriginalDir(Filename,
|
|
|
|
OriginalDir,
|
|
|
|
CurrentDir);
|
|
|
|
if (!resolved.empty())
|
|
|
|
File = FileMgr.getFile(resolved);
|
|
|
|
}
|
|
|
|
|
|
|
|
return File;
|
|
|
|
}
|
|
|
|
|
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
|
|
|
|
2010-12-18 05:22:22 +08:00
|
|
|
if (Filename.empty() || llvm::sys::path::is_absolute(Filename))
|
2009-07-07 08:12:59 +08:00
|
|
|
return;
|
|
|
|
|
2011-07-23 00:35:34 +08:00
|
|
|
if (isysroot.empty()) {
|
2009-07-07 08:12:59 +08:00
|
|
|
// If no system root was given, default to '/'
|
|
|
|
Filename.insert(Filename.begin(), '/');
|
|
|
|
return;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2011-07-23 00:35:34 +08:00
|
|
|
unsigned Length = isysroot.size();
|
2009-07-07 08:12:59 +08:00
|
|
|
if (isysroot[Length - 1] != '/')
|
|
|
|
Filename.insert(Filename.begin(), '/');
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2011-07-23 00:35:34 +08:00
|
|
|
Filename.insert(Filename.begin(), isysroot.begin(), isysroot.end());
|
2009-07-07 08:12:59 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:43 +08:00
|
|
|
ASTReader::ASTReadResult
|
2011-12-01 07:21:26 +08:00
|
|
|
ASTReader::ReadASTBlock(ModuleFile &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;
|
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-10-25 01:26:36 +08:00
|
|
|
case DECL_UPDATES_BLOCK_ID:
|
|
|
|
if (Stream.SkipBlock()) {
|
|
|
|
Error("malformed block record in AST file");
|
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case PREPROCESSOR_BLOCK_ID:
|
2010-07-17 01:50:48 +08:00
|
|
|
F.MacroCursor = Stream;
|
2011-09-10 06:02:16 +08:00
|
|
|
if (!PP.getExternalSource())
|
|
|
|
PP.setExternalSource(this);
|
2010-01-05 03:18:44 +08:00
|
|
|
|
2010-10-21 06:00:55 +08:00
|
|
|
if (Stream.SkipBlock() ||
|
|
|
|
ReadBlockAbbrevs(F.MacroCursor, PREPROCESSOR_BLOCK_ID)) {
|
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;
|
|
|
|
}
|
2010-10-21 06:00:55 +08:00
|
|
|
F.MacroStartOffset = F.MacroCursor.GetCurrentBitNo();
|
2009-04-12 05:15:38 +08:00
|
|
|
break;
|
2009-04-23 18:39:46 +08:00
|
|
|
|
2011-02-09 05:58:10 +08:00
|
|
|
case PREPROCESSOR_DETAIL_BLOCK_ID:
|
|
|
|
F.PreprocessorDetailCursor = Stream;
|
|
|
|
if (Stream.SkipBlock() ||
|
|
|
|
ReadBlockAbbrevs(F.PreprocessorDetailCursor,
|
|
|
|
PREPROCESSOR_DETAIL_BLOCK_ID)) {
|
|
|
|
Error("malformed preprocessor detail record in AST file");
|
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
F.PreprocessorDetailStartOffset
|
|
|
|
= F.PreprocessorDetailCursor.GetCurrentBitNo();
|
2011-09-10 06:02:16 +08:00
|
|
|
|
|
|
|
if (!PP.getPreprocessingRecord())
|
|
|
|
PP.createPreprocessingRecord(true);
|
|
|
|
if (!PP.getPreprocessingRecord()->getExternalSource())
|
|
|
|
PP.getPreprocessingRecord()->SetExternalSource(*this);
|
2011-02-09 05:58:10 +08:00
|
|
|
break;
|
|
|
|
|
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;
|
2011-12-01 01:33:56 +08:00
|
|
|
|
|
|
|
case SUBMODULE_BLOCK_ID:
|
|
|
|
switch (ReadSubmoduleBlock(F)) {
|
|
|
|
case Success:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Failure:
|
|
|
|
Error("malformed submodule block in AST file");
|
|
|
|
return Failure;
|
|
|
|
|
|
|
|
case IgnorePCH:
|
|
|
|
return IgnorePCH;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
if (!Stream.SkipBlock())
|
|
|
|
break;
|
|
|
|
Error("malformed block record in AST file");
|
|
|
|
return Failure;
|
2009-04-11 01:25:41 +08:00
|
|
|
}
|
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,
|
2010-10-05 23:59:54 +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;
|
|
|
|
}
|
|
|
|
|
2011-08-18 05:07:30 +08:00
|
|
|
case IMPORTS: {
|
|
|
|
// Load each of the imported PCH files.
|
|
|
|
unsigned Idx = 0, N = Record.size();
|
|
|
|
while (Idx < N) {
|
|
|
|
// Read information about the AST file.
|
|
|
|
ModuleKind ImportedKind = (ModuleKind)Record[Idx++];
|
|
|
|
unsigned Length = Record[Idx++];
|
|
|
|
llvm::SmallString<128> ImportedFile(Record.begin() + Idx,
|
|
|
|
Record.begin() + Idx + Length);
|
|
|
|
Idx += Length;
|
|
|
|
|
|
|
|
// Load the AST file.
|
2011-08-18 12:12:04 +08:00
|
|
|
switch(ReadASTCore(ImportedFile, ImportedKind, &F)) {
|
2011-08-18 05:07:30 +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;
|
|
|
|
}
|
2010-07-17 08:12:06 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2011-08-03 00:26:37 +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];
|
Change the hashing function for DeclContext lookup within an AST file
by eliminating the type ID from constructor, destructor, and
conversion function names. There are several reasons for this change:
- A given type (say, int*) isn't guaranteed to have a single, unique
type ID within a chain of PCH files. Hence, we could end up hashing
based on the wrong type ID, causing name lookup to fail.
- The mapping from types back to type IDs required one DenseMap
entry for every type that was ever deserialized, which was an
unacceptable cost to support just the name lookup of constructors,
destructors, and conversion functions. Plus, this mapping could
never actually work with chained or multiple PCH, based on the first
bullet.
Once we have eliminated the type from the hash function, these
problems go away, as does my horrible "reverse type remap" hack, which
was doomed from the start (see bullet #1 above) and far too
complicated.
However, note that removing the type from the hash function means that
all constructors, destructors, and conversion functions have the same
hash key, so I've updated the caller to double-check that the
declarations found have the appropriate name.
llvm-svn: 136708
2011-08-03 02:32:54 +08:00
|
|
|
unsigned LocalBaseTypeIndex = Record[1];
|
|
|
|
F.BaseTypeIndex = getTotalNumTypes();
|
2011-07-29 08:21:44 +08:00
|
|
|
|
2011-08-03 00:26:37 +08:00
|
|
|
if (F.LocalNumTypes > 0) {
|
|
|
|
// Introduce the global -> local mapping for types within this module.
|
|
|
|
GlobalTypeMap.insert(std::make_pair(getTotalNumTypes(), &F));
|
|
|
|
|
|
|
|
// Introduce the local -> global mapping for types within this module.
|
2011-12-20 00:14:14 +08:00
|
|
|
F.TypeRemap.insertOrReplace(
|
|
|
|
std::make_pair(LocalBaseTypeIndex,
|
|
|
|
F.BaseTypeIndex - LocalBaseTypeIndex));
|
2011-08-03 00:26:37 +08:00
|
|
|
|
|
|
|
TypesLoaded.resize(TypesLoaded.size() + F.LocalNumTypes);
|
|
|
|
}
|
2009-04-10 06:27:44 +08:00
|
|
|
break;
|
2011-08-03 00:26:37 +08:00
|
|
|
}
|
|
|
|
|
2011-08-03 23:48:04 +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;
|
|
|
|
}
|
2011-10-28 02:47:35 +08:00
|
|
|
F.DeclOffsets = (const DeclOffset *)BlobStart;
|
2010-07-20 06:06:55 +08:00
|
|
|
F.LocalNumDecls = Record[0];
|
2011-08-03 23:48:04 +08:00
|
|
|
unsigned LocalBaseDeclID = Record[1];
|
2011-07-29 08:56:45 +08:00
|
|
|
F.BaseDeclID = getTotalNumDecls();
|
2011-07-20 08:27:43 +08:00
|
|
|
|
2011-08-03 23:48:04 +08:00
|
|
|
if (F.LocalNumDecls > 0) {
|
|
|
|
// Introduce the global -> local mapping for declarations within this
|
|
|
|
// module.
|
2011-08-12 08:15:20 +08:00
|
|
|
GlobalDeclMap.insert(
|
|
|
|
std::make_pair(getTotalNumDecls() + NUM_PREDEF_DECL_IDS, &F));
|
2011-08-03 23:48:04 +08:00
|
|
|
|
|
|
|
// Introduce the local -> global mapping for declarations within this
|
|
|
|
// module.
|
2011-12-20 00:14:14 +08:00
|
|
|
F.DeclRemap.insertOrReplace(
|
|
|
|
std::make_pair(LocalBaseDeclID, F.BaseDeclID - LocalBaseDeclID));
|
2011-08-03 23:48:04 +08:00
|
|
|
|
Completely re-implement (de-)serialization of declaration
chains. The previous implementation relied heavily on the declaration
chain being stored as a (circular) linked list on disk, as it is in
memory. However, when deserializing from multiple modules, the
different chains could get mixed up, leading to broken declaration chains.
The new solution keeps track of the first and last declarations in the
chain for each module file. When we load a declaration, we search all
of the module files for redeclarations of that declaration, then
splice together all of the lists into a coherent whole (along with any
redeclarations that were actually parsed).
As a drive-by fix, (de-)serialize the redeclaration chains of
TypedefNameDecls, which had somehow gotten missed previously. Add a
test of this serialization.
This new scheme creates a redeclaration table that is fairly large in
the PCH file (on the order of 400k for Cocoa.h's 12MB PCH file). The
table is mmap'd in and searched via a binary search, but it's still
quite large. A future tweak will eliminate entries for declarations
that have no redeclarations anywhere, and should
drastically reduce the size of this table.
llvm-svn: 146841
2011-12-18 07:38:30 +08:00
|
|
|
// Introduce the global -> local mapping for declarations within this
|
|
|
|
// module.
|
|
|
|
F.GlobalToLocalDeclIDs[&F] = LocalBaseDeclID;
|
|
|
|
|
2011-08-03 23:48:04 +08:00
|
|
|
DeclsLoaded.resize(DeclsLoaded.size() + F.LocalNumDecls);
|
|
|
|
}
|
2009-04-10 06:27:44 +08:00
|
|
|
break;
|
2011-08-03 23:48:04 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TU_UPDATE_LEXICAL: {
|
2011-09-10 05:34:22 +08:00
|
|
|
DeclContext *TU = Context.getTranslationUnitDecl();
|
2011-08-25 03:03:07 +08:00
|
|
|
DeclContextInfo &Info = F.DeclContextInfos[TU];
|
|
|
|
Info.LexicalDecls = reinterpret_cast<const KindDeclIDPair *>(BlobStart);
|
|
|
|
Info.NumLexicalDecls
|
|
|
|
= static_cast<unsigned int>(BlobLen / sizeof(KindDeclIDPair));
|
2011-09-10 05:34:22 +08:00
|
|
|
TU->setHasExternalLexicalStorage(true);
|
2010-07-28 02:24:41 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2010-08-24 08:50:04 +08:00
|
|
|
case UPDATE_VISIBLE: {
|
2011-08-03 23:48:04 +08:00
|
|
|
unsigned Idx = 0;
|
|
|
|
serialization::DeclID ID = ReadDeclID(F, Record, Idx);
|
2010-08-24 08:50:04 +08:00
|
|
|
void *Table = ASTDeclContextNameLookupTable::Create(
|
2011-08-03 23:48:04 +08:00
|
|
|
(const unsigned char *)BlobStart + Record[Idx++],
|
2010-08-24 08:50:04 +08:00
|
|
|
(const unsigned char *)BlobStart,
|
2011-07-22 08:38:23 +08:00
|
|
|
ASTDeclContextNameLookupTrait(*this, F));
|
2011-09-10 05:34:22 +08:00
|
|
|
if (ID == PREDEF_DECL_TRANSLATION_UNIT_ID) { // Is it the TU?
|
|
|
|
DeclContext *TU = Context.getTranslationUnitDecl();
|
2011-08-25 03:03:07 +08:00
|
|
|
F.DeclContextInfos[TU].NameLookupTableData = Table;
|
2011-08-12 08:15:20 +08:00
|
|
|
TU->setHasExternalVisibleStorage(true);
|
2010-08-24 08:50:04 +08:00
|
|
|
} else
|
2011-08-03 23:48:04 +08:00
|
|
|
PendingVisibleUpdates[ID].push_back(std::make_pair(Table, &F));
|
2010-08-24 08:50:04 +08:00
|
|
|
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");
|
2011-08-03 23:48:04 +08:00
|
|
|
for (unsigned i = 0, e = Record.size(); i < e; /* in loop */) {
|
|
|
|
DeclID First = ReadDeclID(F, Record, i);
|
|
|
|
DeclID Latest = ReadDeclID(F, Record, i);
|
2010-08-04 01:30:10 +08:00
|
|
|
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-10-05 23:59:54 +08:00
|
|
|
ASTIdentifierLookupTrait(*this, F));
|
2011-09-10 06:02:16 +08:00
|
|
|
|
|
|
|
PP.getIdentifierTable().setExternalIdentifierLookup(this);
|
2009-04-26 03:10:14 +08:00
|
|
|
}
|
2009-04-11 08:14:32 +08:00
|
|
|
break;
|
|
|
|
|
2011-08-04 05:49:18 +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];
|
2011-08-04 05:49:18 +08:00
|
|
|
unsigned LocalBaseIdentifierID = Record[1];
|
2011-07-29 08:56:45 +08:00
|
|
|
F.BaseIdentifierID = getTotalNumIdentifiers();
|
2011-07-20 08:59:32 +08:00
|
|
|
|
2011-08-04 05:49:18 +08:00
|
|
|
if (F.LocalNumIdentifiers > 0) {
|
|
|
|
// Introduce the global -> local mapping for identifiers within this
|
|
|
|
// module.
|
|
|
|
GlobalIdentifierMap.insert(std::make_pair(getTotalNumIdentifiers() + 1,
|
|
|
|
&F));
|
|
|
|
|
|
|
|
// Introduce the local -> global mapping for identifiers within this
|
|
|
|
// module.
|
2011-12-20 00:14:14 +08:00
|
|
|
F.IdentifierRemap.insertOrReplace(
|
|
|
|
std::make_pair(LocalBaseIdentifierID,
|
|
|
|
F.BaseIdentifierID - LocalBaseIdentifierID));
|
2011-08-04 05:49:18 +08:00
|
|
|
|
|
|
|
IdentifiersLoaded.resize(IdentifiersLoaded.size()
|
|
|
|
+ F.LocalNumIdentifiers);
|
|
|
|
}
|
2009-04-11 08:14:32 +08:00
|
|
|
break;
|
2011-08-04 05:49:18 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case EXTERNAL_DEFINITIONS:
|
2011-07-22 06:35:25 +08:00
|
|
|
for (unsigned I = 0, N = Record.size(); I != N; ++I)
|
|
|
|
ExternalDefinitions.push_back(getGlobalDeclID(F, Record[I]));
|
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:
|
2011-07-22 08:38:23 +08:00
|
|
|
for (unsigned I = 0, N = Record.size(); I != N; ++I)
|
|
|
|
SpecialTypes.push_back(getGlobalTypeID(F, Record[I]));
|
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 UNUSED_FILESCOPED_DECLS:
|
2011-07-22 06:35:25 +08:00
|
|
|
for (unsigned I = 0, N = Record.size(); I != N; ++I)
|
|
|
|
UnusedFileScopedDecls.push_back(getGlobalDeclID(F, Record[I]));
|
2010-02-12 08:07:30 +08:00
|
|
|
break;
|
2010-03-08 03:10:13 +08:00
|
|
|
|
2011-05-05 07:29:54 +08:00
|
|
|
case DELEGATING_CTORS:
|
2011-07-22 06:35:25 +08:00
|
|
|
for (unsigned I = 0, N = Record.size(); I != N; ++I)
|
|
|
|
DelegatingCtorDecls.push_back(getGlobalDeclID(F, Record[I]));
|
2011-05-05 07:29:54 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case WEAK_UNDECLARED_IDENTIFIERS:
|
2011-07-29 02:09:57 +08:00
|
|
|
if (Record.size() % 4 != 0) {
|
|
|
|
Error("invalid weak identifiers record");
|
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
|
|
|
|
// FIXME: Ignore weak undeclared identifiers from non-original PCH
|
|
|
|
// files. This isn't the way to do it :)
|
|
|
|
WeakUndeclaredIdentifiers.clear();
|
|
|
|
|
|
|
|
// Translate the weak, undeclared identifiers into global IDs.
|
|
|
|
for (unsigned I = 0, N = Record.size(); I < N; /* in loop */) {
|
|
|
|
WeakUndeclaredIdentifiers.push_back(
|
|
|
|
getGlobalIdentifierID(F, Record[I++]));
|
|
|
|
WeakUndeclaredIdentifiers.push_back(
|
|
|
|
getGlobalIdentifierID(F, Record[I++]));
|
|
|
|
WeakUndeclaredIdentifiers.push_back(
|
|
|
|
ReadSourceLocation(F, Record, I).getRawEncoding());
|
|
|
|
WeakUndeclaredIdentifiers.push_back(Record[I++]);
|
|
|
|
}
|
2010-08-05 17:48:08 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case LOCALLY_SCOPED_EXTERNAL_DECLS:
|
2011-07-22 06:35:25 +08:00
|
|
|
for (unsigned I = 0, N = Record.size(); I != N; ++I)
|
|
|
|
LocallyScopedExternalDecls.push_back(getGlobalDeclID(F, Record[I]));
|
2009-04-23 06:18:58 +08:00
|
|
|
break;
|
2009-04-25 05:10:55 +08:00
|
|
|
|
2011-08-04 07:28:44 +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];
|
2011-08-04 07:28:44 +08:00
|
|
|
unsigned LocalBaseSelectorID = Record[1];
|
2011-07-29 08:56:45 +08:00
|
|
|
F.BaseSelectorID = getTotalNumSelectors();
|
2011-07-20 09:10:58 +08:00
|
|
|
|
2011-08-04 07:28:44 +08:00
|
|
|
if (F.LocalNumSelectors > 0) {
|
|
|
|
// Introduce the global -> local mapping for selectors within this
|
|
|
|
// module.
|
|
|
|
GlobalSelectorMap.insert(std::make_pair(getTotalNumSelectors()+1, &F));
|
|
|
|
|
|
|
|
// Introduce the local -> global mapping for selectors within this
|
|
|
|
// module.
|
2011-12-20 00:14:14 +08:00
|
|
|
F.SelectorRemap.insertOrReplace(
|
|
|
|
std::make_pair(LocalBaseSelectorID,
|
|
|
|
F.BaseSelectorID - LocalBaseSelectorID));
|
2009-04-26 01:48:32 +08:00
|
|
|
|
2011-08-04 07:28:44 +08:00
|
|
|
SelectorsLoaded.resize(SelectorsLoaded.size() + F.LocalNumSelectors);
|
|
|
|
}
|
|
|
|
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,
|
2011-07-22 06:35:25 +08:00
|
|
|
ASTSelectorLookupTrait(*this, F));
|
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:
|
2011-07-28 22:41:43 +08:00
|
|
|
if (!Record.empty()) {
|
|
|
|
for (unsigned Idx = 0, N = Record.size() - 1; Idx < N; /* in loop */) {
|
|
|
|
ReferencedSelectorsData.push_back(getGlobalSelectorID(F,
|
|
|
|
Record[Idx++]));
|
|
|
|
ReferencedSelectorsData.push_back(ReadSourceLocation(F, Record, Idx).
|
|
|
|
getRawEncoding());
|
|
|
|
}
|
|
|
|
}
|
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;
|
2011-10-29 06:54:21 +08:00
|
|
|
|
|
|
|
case FILE_SORTED_DECLS:
|
|
|
|
F.FileSortedDecls = (const DeclID *)BlobStart;
|
|
|
|
break;
|
2009-04-27 14:38:32 +08:00
|
|
|
|
Revamp the SourceManager to separate the representation of parsed
source locations from source locations loaded from an AST/PCH file.
Previously, loading an AST/PCH file involved carefully pre-allocating
space at the beginning of the source manager for the source locations
and FileIDs that correspond to the prefix, and then appending the
source locations/FileIDs used for parsing the remaining translation
unit. This design forced us into loading PCH files early, as a prefix,
whic has become a rather significant limitation.
This patch splits the SourceManager space into two parts: for source
location "addresses", the lower values (growing upward) are used to
describe parsed code, while upper values (growing downward) are used
for source locations loaded from AST/PCH files. Similarly, positive
FileIDs are used to describe parsed code while negative FileIDs are
used to file/macro locations loaded from AST/PCH files. As a result,
we can load PCH/AST files even during parsing, making various
improvemnts in the future possible, e.g., teaching #include <foo.h> to
look for and load <foo.h.gch> if it happens to be already available.
This patch was originally written by Sebastian Redl, then brought
forward to the modern age by Jonathan Turner, and finally
polished/finished by me to be committed.
llvm-svn: 135484
2011-07-20 00:10:42 +08:00
|
|
|
case SOURCE_LOCATION_OFFSETS: {
|
|
|
|
F.SLocEntryOffsets = (const uint32_t *)BlobStart;
|
2010-07-21 05:20:32 +08:00
|
|
|
F.LocalNumSLocEntries = Record[0];
|
2011-09-20 04:40:25 +08:00
|
|
|
unsigned SLocSpaceSize = Record[1];
|
Revamp the SourceManager to separate the representation of parsed
source locations from source locations loaded from an AST/PCH file.
Previously, loading an AST/PCH file involved carefully pre-allocating
space at the beginning of the source manager for the source locations
and FileIDs that correspond to the prefix, and then appending the
source locations/FileIDs used for parsing the remaining translation
unit. This design forced us into loading PCH files early, as a prefix,
whic has become a rather significant limitation.
This patch splits the SourceManager space into two parts: for source
location "addresses", the lower values (growing upward) are used to
describe parsed code, while upper values (growing downward) are used
for source locations loaded from AST/PCH files. Similarly, positive
FileIDs are used to describe parsed code while negative FileIDs are
used to file/macro locations loaded from AST/PCH files. As a result,
we can load PCH/AST files even during parsing, making various
improvemnts in the future possible, e.g., teaching #include <foo.h> to
look for and load <foo.h.gch> if it happens to be already available.
This patch was originally written by Sebastian Redl, then brought
forward to the modern age by Jonathan Turner, and finally
polished/finished by me to be committed.
llvm-svn: 135484
2011-07-20 00:10:42 +08:00
|
|
|
llvm::tie(F.SLocEntryBaseID, F.SLocEntryBaseOffset) =
|
2011-09-20 04:40:25 +08:00
|
|
|
SourceMgr.AllocateLoadedSLocEntries(F.LocalNumSLocEntries,
|
|
|
|
SLocSpaceSize);
|
Revamp the SourceManager to separate the representation of parsed
source locations from source locations loaded from an AST/PCH file.
Previously, loading an AST/PCH file involved carefully pre-allocating
space at the beginning of the source manager for the source locations
and FileIDs that correspond to the prefix, and then appending the
source locations/FileIDs used for parsing the remaining translation
unit. This design forced us into loading PCH files early, as a prefix,
whic has become a rather significant limitation.
This patch splits the SourceManager space into two parts: for source
location "addresses", the lower values (growing upward) are used to
describe parsed code, while upper values (growing downward) are used
for source locations loaded from AST/PCH files. Similarly, positive
FileIDs are used to describe parsed code while negative FileIDs are
used to file/macro locations loaded from AST/PCH files. As a result,
we can load PCH/AST files even during parsing, making various
improvemnts in the future possible, e.g., teaching #include <foo.h> to
look for and load <foo.h.gch> if it happens to be already available.
This patch was originally written by Sebastian Redl, then brought
forward to the modern age by Jonathan Turner, and finally
polished/finished by me to be committed.
llvm-svn: 135484
2011-07-20 00:10:42 +08:00
|
|
|
// Make our entry in the range map. BaseID is negative and growing, so
|
|
|
|
// we invert it. Because we invert it, though, we need the other end of
|
|
|
|
// the range.
|
|
|
|
unsigned RangeStart =
|
|
|
|
unsigned(-F.SLocEntryBaseID) - F.LocalNumSLocEntries + 1;
|
|
|
|
GlobalSLocEntryMap.insert(std::make_pair(RangeStart, &F));
|
|
|
|
F.FirstLoc = SourceLocation::getFromRawEncoding(F.SLocEntryBaseOffset);
|
|
|
|
|
2011-09-20 04:40:25 +08:00
|
|
|
// SLocEntryBaseOffset is lower than MaxLoadedOffset and decreasing.
|
|
|
|
assert((F.SLocEntryBaseOffset & (1U << 31U)) == 0);
|
|
|
|
GlobalSLocOffsetMap.insert(
|
|
|
|
std::make_pair(SourceManager::MaxLoadedOffset - F.SLocEntryBaseOffset
|
|
|
|
- SLocSpaceSize,&F));
|
|
|
|
|
Revamp the SourceManager to separate the representation of parsed
source locations from source locations loaded from an AST/PCH file.
Previously, loading an AST/PCH file involved carefully pre-allocating
space at the beginning of the source manager for the source locations
and FileIDs that correspond to the prefix, and then appending the
source locations/FileIDs used for parsing the remaining translation
unit. This design forced us into loading PCH files early, as a prefix,
whic has become a rather significant limitation.
This patch splits the SourceManager space into two parts: for source
location "addresses", the lower values (growing upward) are used to
describe parsed code, while upper values (growing downward) are used
for source locations loaded from AST/PCH files. Similarly, positive
FileIDs are used to describe parsed code while negative FileIDs are
used to file/macro locations loaded from AST/PCH files. As a result,
we can load PCH/AST files even during parsing, making various
improvemnts in the future possible, e.g., teaching #include <foo.h> to
look for and load <foo.h.gch> if it happens to be already available.
This patch was originally written by Sebastian Redl, then brought
forward to the modern age by Jonathan Turner, and finally
polished/finished by me to be committed.
llvm-svn: 135484
2011-07-20 00:10:42 +08:00
|
|
|
// Initialize the remapping table.
|
|
|
|
// Invalid stays invalid.
|
|
|
|
F.SLocRemap.insert(std::make_pair(0U, 0));
|
|
|
|
// This module. Base was 2 when being compiled.
|
|
|
|
F.SLocRemap.insert(std::make_pair(2U,
|
|
|
|
static_cast<int>(F.SLocEntryBaseOffset - 2)));
|
2011-07-22 02:46:38 +08:00
|
|
|
|
|
|
|
TotalNumSLocEntries += F.LocalNumSLocEntries;
|
Revamp the SourceManager to separate the representation of parsed
source locations from source locations loaded from an AST/PCH file.
Previously, loading an AST/PCH file involved carefully pre-allocating
space at the beginning of the source manager for the source locations
and FileIDs that correspond to the prefix, and then appending the
source locations/FileIDs used for parsing the remaining translation
unit. This design forced us into loading PCH files early, as a prefix,
whic has become a rather significant limitation.
This patch splits the SourceManager space into two parts: for source
location "addresses", the lower values (growing upward) are used to
describe parsed code, while upper values (growing downward) are used
for source locations loaded from AST/PCH files. Similarly, positive
FileIDs are used to describe parsed code while negative FileIDs are
used to file/macro locations loaded from AST/PCH files. As a result,
we can load PCH/AST files even during parsing, making various
improvemnts in the future possible, e.g., teaching #include <foo.h> to
look for and load <foo.h.gch> if it happens to be already available.
This patch was originally written by Sebastian Redl, then brought
forward to the modern age by Jonathan Turner, and finally
polished/finished by me to be committed.
llvm-svn: 135484
2011-07-20 00:10:42 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2011-08-02 00:01:55 +08:00
|
|
|
case MODULE_OFFSET_MAP: {
|
Revamp the SourceManager to separate the representation of parsed
source locations from source locations loaded from an AST/PCH file.
Previously, loading an AST/PCH file involved carefully pre-allocating
space at the beginning of the source manager for the source locations
and FileIDs that correspond to the prefix, and then appending the
source locations/FileIDs used for parsing the remaining translation
unit. This design forced us into loading PCH files early, as a prefix,
whic has become a rather significant limitation.
This patch splits the SourceManager space into two parts: for source
location "addresses", the lower values (growing upward) are used to
describe parsed code, while upper values (growing downward) are used
for source locations loaded from AST/PCH files. Similarly, positive
FileIDs are used to describe parsed code while negative FileIDs are
used to file/macro locations loaded from AST/PCH files. As a result,
we can load PCH/AST files even during parsing, making various
improvemnts in the future possible, e.g., teaching #include <foo.h> to
look for and load <foo.h.gch> if it happens to be already available.
This patch was originally written by Sebastian Redl, then brought
forward to the modern age by Jonathan Turner, and finally
polished/finished by me to be committed.
llvm-svn: 135484
2011-07-20 00:10:42 +08:00
|
|
|
// Additional remapping information.
|
|
|
|
const unsigned char *Data = (const unsigned char*)BlobStart;
|
|
|
|
const unsigned char *DataEnd = Data + BlobLen;
|
2011-08-02 18:56:51 +08:00
|
|
|
|
|
|
|
// Continuous range maps we may be updating in our module.
|
|
|
|
ContinuousRangeMap<uint32_t, int, 2>::Builder SLocRemap(F.SLocRemap);
|
2011-08-04 05:49:18 +08:00
|
|
|
ContinuousRangeMap<uint32_t, int, 2>::Builder
|
|
|
|
IdentifierRemap(F.IdentifierRemap);
|
2011-08-05 02:56:47 +08:00
|
|
|
ContinuousRangeMap<uint32_t, int, 2>::Builder
|
|
|
|
PreprocessedEntityRemap(F.PreprocessedEntityRemap);
|
2011-12-01 08:59:36 +08:00
|
|
|
ContinuousRangeMap<uint32_t, int, 2>::Builder
|
|
|
|
SubmoduleRemap(F.SubmoduleRemap);
|
2011-08-04 07:28:44 +08:00
|
|
|
ContinuousRangeMap<uint32_t, int, 2>::Builder
|
|
|
|
SelectorRemap(F.SelectorRemap);
|
2011-08-03 23:48:04 +08:00
|
|
|
ContinuousRangeMap<uint32_t, int, 2>::Builder DeclRemap(F.DeclRemap);
|
2011-08-03 00:26:37 +08:00
|
|
|
ContinuousRangeMap<uint32_t, int, 2>::Builder TypeRemap(F.TypeRemap);
|
|
|
|
|
Revamp the SourceManager to separate the representation of parsed
source locations from source locations loaded from an AST/PCH file.
Previously, loading an AST/PCH file involved carefully pre-allocating
space at the beginning of the source manager for the source locations
and FileIDs that correspond to the prefix, and then appending the
source locations/FileIDs used for parsing the remaining translation
unit. This design forced us into loading PCH files early, as a prefix,
whic has become a rather significant limitation.
This patch splits the SourceManager space into two parts: for source
location "addresses", the lower values (growing upward) are used to
describe parsed code, while upper values (growing downward) are used
for source locations loaded from AST/PCH files. Similarly, positive
FileIDs are used to describe parsed code while negative FileIDs are
used to file/macro locations loaded from AST/PCH files. As a result,
we can load PCH/AST files even during parsing, making various
improvemnts in the future possible, e.g., teaching #include <foo.h> to
look for and load <foo.h.gch> if it happens to be already available.
This patch was originally written by Sebastian Redl, then brought
forward to the modern age by Jonathan Turner, and finally
polished/finished by me to be committed.
llvm-svn: 135484
2011-07-20 00:10:42 +08:00
|
|
|
while(Data < DataEnd) {
|
|
|
|
uint16_t Len = io::ReadUnalignedLE16(Data);
|
2011-07-23 18:55:15 +08:00
|
|
|
StringRef Name = StringRef((const char*)Data, Len);
|
2011-08-02 18:56:51 +08:00
|
|
|
Data += Len;
|
2011-12-01 07:21:26 +08:00
|
|
|
ModuleFile *OM = ModuleMgr.lookup(Name);
|
Revamp the SourceManager to separate the representation of parsed
source locations from source locations loaded from an AST/PCH file.
Previously, loading an AST/PCH file involved carefully pre-allocating
space at the beginning of the source manager for the source locations
and FileIDs that correspond to the prefix, and then appending the
source locations/FileIDs used for parsing the remaining translation
unit. This design forced us into loading PCH files early, as a prefix,
whic has become a rather significant limitation.
This patch splits the SourceManager space into two parts: for source
location "addresses", the lower values (growing upward) are used to
describe parsed code, while upper values (growing downward) are used
for source locations loaded from AST/PCH files. Similarly, positive
FileIDs are used to describe parsed code while negative FileIDs are
used to file/macro locations loaded from AST/PCH files. As a result,
we can load PCH/AST files even during parsing, making various
improvemnts in the future possible, e.g., teaching #include <foo.h> to
look for and load <foo.h.gch> if it happens to be already available.
This patch was originally written by Sebastian Redl, then brought
forward to the modern age by Jonathan Turner, and finally
polished/finished by me to be committed.
llvm-svn: 135484
2011-07-20 00:10:42 +08:00
|
|
|
if (!OM) {
|
|
|
|
Error("SourceLocation remap refers to unknown module");
|
|
|
|
return Failure;
|
|
|
|
}
|
2011-08-02 18:56:51 +08:00
|
|
|
|
|
|
|
uint32_t SLocOffset = io::ReadUnalignedLE32(Data);
|
|
|
|
uint32_t IdentifierIDOffset = io::ReadUnalignedLE32(Data);
|
|
|
|
uint32_t PreprocessedEntityIDOffset = io::ReadUnalignedLE32(Data);
|
2011-12-01 08:59:36 +08:00
|
|
|
uint32_t SubmoduleIDOffset = io::ReadUnalignedLE32(Data);
|
2011-08-02 18:56:51 +08:00
|
|
|
uint32_t SelectorIDOffset = io::ReadUnalignedLE32(Data);
|
|
|
|
uint32_t DeclIDOffset = io::ReadUnalignedLE32(Data);
|
2011-08-03 00:26:37 +08:00
|
|
|
uint32_t TypeIndexOffset = io::ReadUnalignedLE32(Data);
|
2011-08-02 18:56:51 +08:00
|
|
|
|
|
|
|
// Source location offset is mapped to OM->SLocEntryBaseOffset.
|
|
|
|
SLocRemap.insert(std::make_pair(SLocOffset,
|
|
|
|
static_cast<int>(OM->SLocEntryBaseOffset - SLocOffset)));
|
2011-08-04 05:49:18 +08:00
|
|
|
IdentifierRemap.insert(
|
|
|
|
std::make_pair(IdentifierIDOffset,
|
|
|
|
OM->BaseIdentifierID - IdentifierIDOffset));
|
2011-08-05 02:56:47 +08:00
|
|
|
PreprocessedEntityRemap.insert(
|
|
|
|
std::make_pair(PreprocessedEntityIDOffset,
|
|
|
|
OM->BasePreprocessedEntityID - PreprocessedEntityIDOffset));
|
2011-12-01 08:59:36 +08:00
|
|
|
SubmoduleRemap.insert(std::make_pair(SubmoduleIDOffset,
|
|
|
|
OM->BaseSubmoduleID - SubmoduleIDOffset));
|
2011-08-04 07:28:44 +08:00
|
|
|
SelectorRemap.insert(std::make_pair(SelectorIDOffset,
|
|
|
|
OM->BaseSelectorID - SelectorIDOffset));
|
2011-08-03 23:48:04 +08:00
|
|
|
DeclRemap.insert(std::make_pair(DeclIDOffset,
|
|
|
|
OM->BaseDeclID - DeclIDOffset));
|
|
|
|
|
2011-08-03 00:26:37 +08:00
|
|
|
TypeRemap.insert(std::make_pair(TypeIndexOffset,
|
Change the hashing function for DeclContext lookup within an AST file
by eliminating the type ID from constructor, destructor, and
conversion function names. There are several reasons for this change:
- A given type (say, int*) isn't guaranteed to have a single, unique
type ID within a chain of PCH files. Hence, we could end up hashing
based on the wrong type ID, causing name lookup to fail.
- The mapping from types back to type IDs required one DenseMap
entry for every type that was ever deserialized, which was an
unacceptable cost to support just the name lookup of constructors,
destructors, and conversion functions. Plus, this mapping could
never actually work with chained or multiple PCH, based on the first
bullet.
Once we have eliminated the type from the hash function, these
problems go away, as does my horrible "reverse type remap" hack, which
was doomed from the start (see bullet #1 above) and far too
complicated.
However, note that removing the type from the hash function means that
all constructors, destructors, and conversion functions have the same
hash key, so I've updated the caller to double-check that the
declarations found have the appropriate name.
llvm-svn: 136708
2011-08-03 02:32:54 +08:00
|
|
|
OM->BaseTypeIndex - TypeIndexOffset));
|
Completely re-implement (de-)serialization of declaration
chains. The previous implementation relied heavily on the declaration
chain being stored as a (circular) linked list on disk, as it is in
memory. However, when deserializing from multiple modules, the
different chains could get mixed up, leading to broken declaration chains.
The new solution keeps track of the first and last declarations in the
chain for each module file. When we load a declaration, we search all
of the module files for redeclarations of that declaration, then
splice together all of the lists into a coherent whole (along with any
redeclarations that were actually parsed).
As a drive-by fix, (de-)serialize the redeclaration chains of
TypedefNameDecls, which had somehow gotten missed previously. Add a
test of this serialization.
This new scheme creates a redeclaration table that is fairly large in
the PCH file (on the order of 400k for Cocoa.h's 12MB PCH file). The
table is mmap'd in and searched via a binary search, but it's still
quite large. A future tweak will eliminate entries for declarations
that have no redeclarations anywhere, and should
drastically reduce the size of this table.
llvm-svn: 146841
2011-12-18 07:38:30 +08:00
|
|
|
|
|
|
|
// Global -> local mappings.
|
|
|
|
F.GlobalToLocalDeclIDs[OM] = DeclIDOffset;
|
Revamp the SourceManager to separate the representation of parsed
source locations from source locations loaded from an AST/PCH file.
Previously, loading an AST/PCH file involved carefully pre-allocating
space at the beginning of the source manager for the source locations
and FileIDs that correspond to the prefix, and then appending the
source locations/FileIDs used for parsing the remaining translation
unit. This design forced us into loading PCH files early, as a prefix,
whic has become a rather significant limitation.
This patch splits the SourceManager space into two parts: for source
location "addresses", the lower values (growing upward) are used to
describe parsed code, while upper values (growing downward) are used
for source locations loaded from AST/PCH files. Similarly, positive
FileIDs are used to describe parsed code while negative FileIDs are
used to file/macro locations loaded from AST/PCH files. As a result,
we can load PCH/AST files even during parsing, making various
improvemnts in the future possible, e.g., teaching #include <foo.h> to
look for and load <foo.h.gch> if it happens to be already available.
This patch was originally written by Sebastian Redl, then brought
forward to the modern age by Jonathan Turner, and finally
polished/finished by me to be committed.
llvm-svn: 135484
2011-07-20 00:10:42 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case SOURCE_MANAGER_LINE_TABLE:
|
|
|
|
if (ParseLineTable(F, Record))
|
|
|
|
return Failure;
|
2009-04-27 14:38:32 +08:00
|
|
|
break;
|
|
|
|
|
2011-06-03 04:01:46 +08:00
|
|
|
case FILE_SOURCE_LOCATION_OFFSETS:
|
|
|
|
F.SLocFileOffsets = (const uint32_t *)BlobStart;
|
|
|
|
F.LocalNumSLocFileEntries = Record[0];
|
|
|
|
break;
|
|
|
|
|
Revamp the SourceManager to separate the representation of parsed
source locations from source locations loaded from an AST/PCH file.
Previously, loading an AST/PCH file involved carefully pre-allocating
space at the beginning of the source manager for the source locations
and FileIDs that correspond to the prefix, and then appending the
source locations/FileIDs used for parsing the remaining translation
unit. This design forced us into loading PCH files early, as a prefix,
whic has become a rather significant limitation.
This patch splits the SourceManager space into two parts: for source
location "addresses", the lower values (growing upward) are used to
describe parsed code, while upper values (growing downward) are used
for source locations loaded from AST/PCH files. Similarly, positive
FileIDs are used to describe parsed code while negative FileIDs are
used to file/macro locations loaded from AST/PCH files. As a result,
we can load PCH/AST files even during parsing, making various
improvemnts in the future possible, e.g., teaching #include <foo.h> to
look for and load <foo.h.gch> if it happens to be already available.
This patch was originally written by Sebastian Redl, then brought
forward to the modern age by Jonathan Turner, and finally
polished/finished by me to be committed.
llvm-svn: 135484
2011-07-20 00:10:42 +08:00
|
|
|
case SOURCE_LOCATION_PRELOADS: {
|
|
|
|
// Need to transform from the local view (1-based IDs) to the global view,
|
|
|
|
// which is based off F.SLocEntryBaseID.
|
2011-08-26 05:09:44 +08:00
|
|
|
if (!F.PreloadSLocEntries.empty()) {
|
|
|
|
Error("Multiple SOURCE_LOCATION_PRELOADS records in AST file");
|
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
|
|
|
|
F.PreloadSLocEntries.swap(Record);
|
2009-04-27 14:38:32 +08:00
|
|
|
break;
|
Revamp the SourceManager to separate the representation of parsed
source locations from source locations loaded from an AST/PCH file.
Previously, loading an AST/PCH file involved carefully pre-allocating
space at the beginning of the source manager for the source locations
and FileIDs that correspond to the prefix, and then appending the
source locations/FileIDs used for parsing the remaining translation
unit. This design forced us into loading PCH files early, as a prefix,
whic has become a rather significant limitation.
This patch splits the SourceManager space into two parts: for source
location "addresses", the lower values (growing upward) are used to
describe parsed code, while upper values (growing downward) are used
for source locations loaded from AST/PCH files. Similarly, positive
FileIDs are used to describe parsed code while negative FileIDs are
used to file/macro locations loaded from AST/PCH files. As a result,
we can load PCH/AST files even during parsing, making various
improvemnts in the future possible, e.g., teaching #include <foo.h> to
look for and load <foo.h.gch> if it happens to be already available.
This patch was originally written by Sebastian Redl, then brought
forward to the modern age by Jonathan Turner, and finally
polished/finished by me to be committed.
llvm-svn: 135484
2011-07-20 00:10:42 +08:00
|
|
|
}
|
2009-04-28 02:38:38 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case STAT_CACHE: {
|
2011-02-06 03:42:43 +08:00
|
|
|
if (!DisableStatCache) {
|
|
|
|
ASTStatCache *MyStatCache =
|
|
|
|
new ASTStatCache((const unsigned char *)BlobStart + Record[0],
|
|
|
|
(const unsigned char *)BlobStart,
|
|
|
|
NumStatHits, NumStatMisses);
|
|
|
|
FileMgr.addStatCache(MyStatCache);
|
|
|
|
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:
|
2011-07-22 06:35:25 +08:00
|
|
|
for (unsigned I = 0, N = Record.size(); I != N; ++I)
|
|
|
|
ExtVectorDecls.push_back(getGlobalDeclID(F, Record[I]));
|
2009-04-28 04:06:05 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case VTABLE_USES:
|
2011-07-29 03:11:31 +08:00
|
|
|
if (Record.size() % 3 != 0) {
|
|
|
|
Error("Invalid VTABLE_USES record");
|
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
|
2010-08-06 02:21:25 +08:00
|
|
|
// Later tables overwrite earlier ones.
|
2011-07-29 03:11:31 +08:00
|
|
|
// FIXME: Modules will have some trouble with this. This is clearly not
|
|
|
|
// the right way to do this.
|
2011-07-22 06:35:25 +08:00
|
|
|
VTableUses.clear();
|
2011-07-29 03:11:31 +08:00
|
|
|
|
|
|
|
for (unsigned Idx = 0, N = Record.size(); Idx != N; /* In loop */) {
|
|
|
|
VTableUses.push_back(getGlobalDeclID(F, Record[Idx++]));
|
|
|
|
VTableUses.push_back(
|
|
|
|
ReadSourceLocation(F, Record, Idx).getRawEncoding());
|
|
|
|
VTableUses.push_back(Record[Idx++]);
|
|
|
|
}
|
2010-07-06 23:37:04 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case DYNAMIC_CLASSES:
|
2011-07-22 06:35:25 +08:00
|
|
|
for (unsigned I = 0, N = Record.size(); I != N; ++I)
|
|
|
|
DynamicClasses.push_back(getGlobalDeclID(F, Record[I]));
|
2010-07-06 23:37:04 +08:00
|
|
|
break;
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case PENDING_IMPLICIT_INSTANTIATIONS:
|
2011-07-29 03:26:52 +08:00
|
|
|
if (PendingInstantiations.size() % 2 != 0) {
|
|
|
|
Error("Invalid PENDING_IMPLICIT_INSTANTIATIONS block");
|
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Later lists of pending instantiations overwrite earlier ones.
|
|
|
|
// FIXME: This is most certainly wrong for modules.
|
|
|
|
PendingInstantiations.clear();
|
|
|
|
for (unsigned I = 0, N = Record.size(); I != N; /* in loop */) {
|
|
|
|
PendingInstantiations.push_back(getGlobalDeclID(F, Record[I++]));
|
|
|
|
PendingInstantiations.push_back(
|
|
|
|
ReadSourceLocation(F, Record, I).getRawEncoding());
|
|
|
|
}
|
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.
|
2011-07-22 06:35:25 +08:00
|
|
|
// FIXME: Modules will have some trouble with this.
|
|
|
|
SemaDeclRefs.clear();
|
|
|
|
for (unsigned I = 0, N = Record.size(); I != N; ++I)
|
|
|
|
SemaDeclRefs.push_back(getGlobalDeclID(F, Record[I]));
|
2010-08-02 15:14:54 +08:00
|
|
|
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
|
|
|
|
2011-05-07 05:43:30 +08:00
|
|
|
case ORIGINAL_FILE_ID:
|
|
|
|
OriginalFileID = FileID::get(Record[0]);
|
|
|
|
break;
|
|
|
|
|
2011-02-16 01:54:22 +08:00
|
|
|
case ORIGINAL_PCH_DIR:
|
|
|
|
// The primary AST will be the last to get here, so it will be the one
|
|
|
|
// that's used.
|
|
|
|
OriginalDir.assign(BlobStart, BlobLen);
|
|
|
|
break;
|
|
|
|
|
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();
|
2011-07-23 18:55:15 +08:00
|
|
|
StringRef ASTBranch(BlobStart, BlobLen);
|
|
|
|
if (StringRef(CurBranch) != ASTBranch && !DisableValidation) {
|
2010-08-19 07:57:06 +08:00
|
|
|
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
|
|
|
|
2011-09-16 02:02:56 +08:00
|
|
|
case PPD_ENTITIES_OFFSETS: {
|
2011-09-20 04:40:25 +08:00
|
|
|
F.PreprocessedEntityOffsets = (const PPEntityOffset *)BlobStart;
|
|
|
|
assert(BlobLen % sizeof(PPEntityOffset) == 0);
|
|
|
|
F.NumPreprocessedEntities = BlobLen / sizeof(PPEntityOffset);
|
2011-09-16 02:02:56 +08:00
|
|
|
|
|
|
|
unsigned LocalBasePreprocessedEntityID = Record[0];
|
2011-08-05 00:36:56 +08:00
|
|
|
|
2011-07-21 08:47:40 +08:00
|
|
|
unsigned StartingID;
|
2011-09-10 06:02:16 +08:00
|
|
|
if (!PP.getPreprocessingRecord())
|
|
|
|
PP.createPreprocessingRecord(true);
|
|
|
|
if (!PP.getPreprocessingRecord()->getExternalSource())
|
|
|
|
PP.getPreprocessingRecord()->SetExternalSource(*this);
|
|
|
|
StartingID
|
|
|
|
= PP.getPreprocessingRecord()
|
2011-09-16 02:02:56 +08:00
|
|
|
->allocateLoadedEntities(F.NumPreprocessedEntities);
|
2011-07-29 08:56:45 +08:00
|
|
|
F.BasePreprocessedEntityID = StartingID;
|
2011-07-21 08:47:40 +08:00
|
|
|
|
2011-09-16 02:02:56 +08:00
|
|
|
if (F.NumPreprocessedEntities > 0) {
|
2011-08-05 02:56:47 +08:00
|
|
|
// Introduce the global -> local mapping for preprocessed entities in
|
|
|
|
// this module.
|
|
|
|
GlobalPreprocessedEntityMap.insert(std::make_pair(StartingID, &F));
|
|
|
|
|
|
|
|
// Introduce the local -> global mapping for preprocessed entities in
|
|
|
|
// this module.
|
2011-12-20 00:14:14 +08:00
|
|
|
F.PreprocessedEntityRemap.insertOrReplace(
|
2011-08-05 02:56:47 +08:00
|
|
|
std::make_pair(LocalBasePreprocessedEntityID,
|
|
|
|
F.BasePreprocessedEntityID - LocalBasePreprocessedEntityID));
|
|
|
|
}
|
|
|
|
|
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;
|
2011-07-21 08:47:40 +08:00
|
|
|
}
|
|
|
|
|
2010-10-25 01:26:36 +08:00
|
|
|
case DECL_UPDATE_OFFSETS: {
|
|
|
|
if (Record.size() % 2 != 0) {
|
|
|
|
Error("invalid DECL_UPDATE_OFFSETS block in AST file");
|
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
for (unsigned I = 0, N = Record.size(); I != N; I += 2)
|
2011-08-03 23:48:04 +08:00
|
|
|
DeclUpdateOffsets[getGlobalDeclID(F, Record[I])]
|
|
|
|
.push_back(std::make_pair(&F, Record[I+1]));
|
2010-10-25 01:26:36 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case DECL_REPLACEMENTS: {
|
2011-10-31 15:20:15 +08:00
|
|
|
if (Record.size() % 3 != 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;
|
|
|
|
}
|
2011-10-31 15:20:15 +08:00
|
|
|
for (unsigned I = 0, N = Record.size(); I != N; I += 3)
|
2011-08-03 23:48:04 +08:00
|
|
|
ReplacedDecls[getGlobalDeclID(F, Record[I])]
|
2011-10-31 15:20:15 +08:00
|
|
|
= ReplacedDeclInfo(&F, Record[I+1], Record[I+2]);
|
2010-08-13 08:28:03 +08:00
|
|
|
break;
|
|
|
|
}
|
2011-09-01 08:58:55 +08:00
|
|
|
|
|
|
|
case OBJC_CHAINED_CATEGORIES: {
|
|
|
|
if (Record.size() % 3 != 0) {
|
|
|
|
Error("invalid OBJC_CHAINED_CATEGORIES block in AST file");
|
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
for (unsigned I = 0, N = Record.size(); I != N; I += 3) {
|
|
|
|
serialization::GlobalDeclID GlobID = getGlobalDeclID(F, Record[I]);
|
|
|
|
F.ChainedObjCCategories[GlobID] = std::make_pair(Record[I+1],
|
|
|
|
Record[I+2]);
|
|
|
|
ObjCChainedCategoriesInterfaces.insert(GlobID);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2010-10-30 06:39:52 +08:00
|
|
|
|
|
|
|
case CXX_BASE_SPECIFIER_OFFSETS: {
|
|
|
|
if (F.LocalNumCXXBaseSpecifiers != 0) {
|
|
|
|
Error("duplicate CXX_BASE_SPECIFIER_OFFSETS record in AST file");
|
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
|
|
|
|
F.LocalNumCXXBaseSpecifiers = Record[0];
|
|
|
|
F.CXXBaseSpecifiersOffsets = (const uint32_t *)BlobStart;
|
2011-07-22 05:15:19 +08:00
|
|
|
NumCXXBaseSpecifiersLoaded += F.LocalNumCXXBaseSpecifiers;
|
2010-10-30 06:39:52 +08:00
|
|
|
break;
|
|
|
|
}
|
2010-11-06 06:10:18 +08:00
|
|
|
|
2011-01-15 04:54:07 +08:00
|
|
|
case DIAG_PRAGMA_MAPPINGS:
|
2010-11-06 06:10:18 +08:00
|
|
|
if (Record.size() % 2 != 0) {
|
|
|
|
Error("invalid DIAG_USER_MAPPINGS block in AST file");
|
|
|
|
return Failure;
|
|
|
|
}
|
Revamp the SourceManager to separate the representation of parsed
source locations from source locations loaded from an AST/PCH file.
Previously, loading an AST/PCH file involved carefully pre-allocating
space at the beginning of the source manager for the source locations
and FileIDs that correspond to the prefix, and then appending the
source locations/FileIDs used for parsing the remaining translation
unit. This design forced us into loading PCH files early, as a prefix,
whic has become a rather significant limitation.
This patch splits the SourceManager space into two parts: for source
location "addresses", the lower values (growing upward) are used to
describe parsed code, while upper values (growing downward) are used
for source locations loaded from AST/PCH files. Similarly, positive
FileIDs are used to describe parsed code while negative FileIDs are
used to file/macro locations loaded from AST/PCH files. As a result,
we can load PCH/AST files even during parsing, making various
improvemnts in the future possible, e.g., teaching #include <foo.h> to
look for and load <foo.h.gch> if it happens to be already available.
This patch was originally written by Sebastian Redl, then brought
forward to the modern age by Jonathan Turner, and finally
polished/finished by me to be committed.
llvm-svn: 135484
2011-07-20 00:10:42 +08:00
|
|
|
|
|
|
|
if (F.PragmaDiagMappings.empty())
|
|
|
|
F.PragmaDiagMappings.swap(Record);
|
2010-11-06 06:10:18 +08:00
|
|
|
else
|
Revamp the SourceManager to separate the representation of parsed
source locations from source locations loaded from an AST/PCH file.
Previously, loading an AST/PCH file involved carefully pre-allocating
space at the beginning of the source manager for the source locations
and FileIDs that correspond to the prefix, and then appending the
source locations/FileIDs used for parsing the remaining translation
unit. This design forced us into loading PCH files early, as a prefix,
whic has become a rather significant limitation.
This patch splits the SourceManager space into two parts: for source
location "addresses", the lower values (growing upward) are used to
describe parsed code, while upper values (growing downward) are used
for source locations loaded from AST/PCH files. Similarly, positive
FileIDs are used to describe parsed code while negative FileIDs are
used to file/macro locations loaded from AST/PCH files. As a result,
we can load PCH/AST files even during parsing, making various
improvemnts in the future possible, e.g., teaching #include <foo.h> to
look for and load <foo.h.gch> if it happens to be already available.
This patch was originally written by Sebastian Redl, then brought
forward to the modern age by Jonathan Turner, and finally
polished/finished by me to be committed.
llvm-svn: 135484
2011-07-20 00:10:42 +08:00
|
|
|
F.PragmaDiagMappings.insert(F.PragmaDiagMappings.end(),
|
|
|
|
Record.begin(), Record.end());
|
2010-11-06 06:10:18 +08:00
|
|
|
break;
|
Implement two related optimizations that make de-serialization of
AST/PCH files more lazy:
- Don't preload all of the file source-location entries when reading
the AST file. Instead, load them lazily, when needed.
- Only look up header-search information (whether a header was already
#import'd, how many times it's been included, etc.) when it's needed
by the preprocessor, rather than pre-populating it.
Previously, we would pre-load all of the file source-location entries,
which also populated the header-search information structure. This was
a relatively minor performance issue, since we would end up stat()'ing
all of the headers stored within a AST/PCH file when the AST/PCH file
was loaded. In the normal PCH use case, the stat()s were cached, so
the cost--of preloading ~860 source-location entries in the Cocoa.h
case---was relatively low.
However, the recent optimization that replaced stat+open with
open+fstat turned this into a major problem, since the preloading of
source-location entries would now end up opening those files. Worse,
those files wouldn't be closed until the file manager was destroyed,
so just opening a Cocoa.h PCH file would hold on to ~860 file
descriptors, and it was easy to blow through the process's limit on
the number of open file descriptors.
By eliminating the preloading of these files, we neither open nor stat
the headers stored in the PCH/AST file until they're actually needed
for something. Concretely, we went from
*** HeaderSearch Stats:
835 files tracked.
364 #import/#pragma once files.
823 included exactly once.
6 max times a file is included.
3 #include/#include_next/#import.
0 #includes skipped due to the multi-include optimization.
1 framework lookups.
0 subframework lookups.
*** Source Manager Stats:
835 files mapped, 3 mem buffers mapped.
37460 SLocEntry's allocated, 11215575B of Sloc address space used.
62 bytes of files mapped, 0 files with line #'s computed.
with a trivial program that uses a chained PCH including a Cocoa PCH
to
*** HeaderSearch Stats:
4 files tracked.
1 #import/#pragma once files.
3 included exactly once.
2 max times a file is included.
3 #include/#include_next/#import.
0 #includes skipped due to the multi-include optimization.
1 framework lookups.
0 subframework lookups.
*** Source Manager Stats:
3 files mapped, 3 mem buffers mapped.
37460 SLocEntry's allocated, 11215575B of Sloc address space used.
62 bytes of files mapped, 0 files with line #'s computed.
for the same program.
llvm-svn: 125286
2011-02-11 01:09:37 +08:00
|
|
|
|
2011-02-10 05:04:32 +08:00
|
|
|
case CUDA_SPECIAL_DECL_REFS:
|
|
|
|
// Later tables overwrite earlier ones.
|
2011-07-22 06:35:25 +08:00
|
|
|
// FIXME: Modules will have trouble with this.
|
|
|
|
CUDASpecialDeclRefs.clear();
|
|
|
|
for (unsigned I = 0, N = Record.size(); I != N; ++I)
|
|
|
|
CUDASpecialDeclRefs.push_back(getGlobalDeclID(F, Record[I]));
|
2011-02-10 05:04:32 +08:00
|
|
|
break;
|
Implement two related optimizations that make de-serialization of
AST/PCH files more lazy:
- Don't preload all of the file source-location entries when reading
the AST file. Instead, load them lazily, when needed.
- Only look up header-search information (whether a header was already
#import'd, how many times it's been included, etc.) when it's needed
by the preprocessor, rather than pre-populating it.
Previously, we would pre-load all of the file source-location entries,
which also populated the header-search information structure. This was
a relatively minor performance issue, since we would end up stat()'ing
all of the headers stored within a AST/PCH file when the AST/PCH file
was loaded. In the normal PCH use case, the stat()s were cached, so
the cost--of preloading ~860 source-location entries in the Cocoa.h
case---was relatively low.
However, the recent optimization that replaced stat+open with
open+fstat turned this into a major problem, since the preloading of
source-location entries would now end up opening those files. Worse,
those files wouldn't be closed until the file manager was destroyed,
so just opening a Cocoa.h PCH file would hold on to ~860 file
descriptors, and it was easy to blow through the process's limit on
the number of open file descriptors.
By eliminating the preloading of these files, we neither open nor stat
the headers stored in the PCH/AST file until they're actually needed
for something. Concretely, we went from
*** HeaderSearch Stats:
835 files tracked.
364 #import/#pragma once files.
823 included exactly once.
6 max times a file is included.
3 #include/#include_next/#import.
0 #includes skipped due to the multi-include optimization.
1 framework lookups.
0 subframework lookups.
*** Source Manager Stats:
835 files mapped, 3 mem buffers mapped.
37460 SLocEntry's allocated, 11215575B of Sloc address space used.
62 bytes of files mapped, 0 files with line #'s computed.
with a trivial program that uses a chained PCH including a Cocoa PCH
to
*** HeaderSearch Stats:
4 files tracked.
1 #import/#pragma once files.
3 included exactly once.
2 max times a file is included.
3 #include/#include_next/#import.
0 #includes skipped due to the multi-include optimization.
1 framework lookups.
0 subframework lookups.
*** Source Manager Stats:
3 files mapped, 3 mem buffers mapped.
37460 SLocEntry's allocated, 11215575B of Sloc address space used.
62 bytes of files mapped, 0 files with line #'s computed.
for the same program.
llvm-svn: 125286
2011-02-11 01:09:37 +08:00
|
|
|
|
2011-07-28 12:50:02 +08:00
|
|
|
case HEADER_SEARCH_TABLE: {
|
Implement two related optimizations that make de-serialization of
AST/PCH files more lazy:
- Don't preload all of the file source-location entries when reading
the AST file. Instead, load them lazily, when needed.
- Only look up header-search information (whether a header was already
#import'd, how many times it's been included, etc.) when it's needed
by the preprocessor, rather than pre-populating it.
Previously, we would pre-load all of the file source-location entries,
which also populated the header-search information structure. This was
a relatively minor performance issue, since we would end up stat()'ing
all of the headers stored within a AST/PCH file when the AST/PCH file
was loaded. In the normal PCH use case, the stat()s were cached, so
the cost--of preloading ~860 source-location entries in the Cocoa.h
case---was relatively low.
However, the recent optimization that replaced stat+open with
open+fstat turned this into a major problem, since the preloading of
source-location entries would now end up opening those files. Worse,
those files wouldn't be closed until the file manager was destroyed,
so just opening a Cocoa.h PCH file would hold on to ~860 file
descriptors, and it was easy to blow through the process's limit on
the number of open file descriptors.
By eliminating the preloading of these files, we neither open nor stat
the headers stored in the PCH/AST file until they're actually needed
for something. Concretely, we went from
*** HeaderSearch Stats:
835 files tracked.
364 #import/#pragma once files.
823 included exactly once.
6 max times a file is included.
3 #include/#include_next/#import.
0 #includes skipped due to the multi-include optimization.
1 framework lookups.
0 subframework lookups.
*** Source Manager Stats:
835 files mapped, 3 mem buffers mapped.
37460 SLocEntry's allocated, 11215575B of Sloc address space used.
62 bytes of files mapped, 0 files with line #'s computed.
with a trivial program that uses a chained PCH including a Cocoa PCH
to
*** HeaderSearch Stats:
4 files tracked.
1 #import/#pragma once files.
3 included exactly once.
2 max times a file is included.
3 #include/#include_next/#import.
0 #includes skipped due to the multi-include optimization.
1 framework lookups.
0 subframework lookups.
*** Source Manager Stats:
3 files mapped, 3 mem buffers mapped.
37460 SLocEntry's allocated, 11215575B of Sloc address space used.
62 bytes of files mapped, 0 files with line #'s computed.
for the same program.
llvm-svn: 125286
2011-02-11 01:09:37 +08:00
|
|
|
F.HeaderFileInfoTableData = BlobStart;
|
|
|
|
F.LocalNumHeaderFileInfos = Record[1];
|
2011-07-28 12:50:02 +08:00
|
|
|
F.HeaderFileFrameworkStrings = BlobStart + Record[2];
|
Implement two related optimizations that make de-serialization of
AST/PCH files more lazy:
- Don't preload all of the file source-location entries when reading
the AST file. Instead, load them lazily, when needed.
- Only look up header-search information (whether a header was already
#import'd, how many times it's been included, etc.) when it's needed
by the preprocessor, rather than pre-populating it.
Previously, we would pre-load all of the file source-location entries,
which also populated the header-search information structure. This was
a relatively minor performance issue, since we would end up stat()'ing
all of the headers stored within a AST/PCH file when the AST/PCH file
was loaded. In the normal PCH use case, the stat()s were cached, so
the cost--of preloading ~860 source-location entries in the Cocoa.h
case---was relatively low.
However, the recent optimization that replaced stat+open with
open+fstat turned this into a major problem, since the preloading of
source-location entries would now end up opening those files. Worse,
those files wouldn't be closed until the file manager was destroyed,
so just opening a Cocoa.h PCH file would hold on to ~860 file
descriptors, and it was easy to blow through the process's limit on
the number of open file descriptors.
By eliminating the preloading of these files, we neither open nor stat
the headers stored in the PCH/AST file until they're actually needed
for something. Concretely, we went from
*** HeaderSearch Stats:
835 files tracked.
364 #import/#pragma once files.
823 included exactly once.
6 max times a file is included.
3 #include/#include_next/#import.
0 #includes skipped due to the multi-include optimization.
1 framework lookups.
0 subframework lookups.
*** Source Manager Stats:
835 files mapped, 3 mem buffers mapped.
37460 SLocEntry's allocated, 11215575B of Sloc address space used.
62 bytes of files mapped, 0 files with line #'s computed.
with a trivial program that uses a chained PCH including a Cocoa PCH
to
*** HeaderSearch Stats:
4 files tracked.
1 #import/#pragma once files.
3 included exactly once.
2 max times a file is included.
3 #include/#include_next/#import.
0 #includes skipped due to the multi-include optimization.
1 framework lookups.
0 subframework lookups.
*** Source Manager Stats:
3 files mapped, 3 mem buffers mapped.
37460 SLocEntry's allocated, 11215575B of Sloc address space used.
62 bytes of files mapped, 0 files with line #'s computed.
for the same program.
llvm-svn: 125286
2011-02-11 01:09:37 +08:00
|
|
|
if (Record[0]) {
|
|
|
|
F.HeaderFileInfoTable
|
|
|
|
= HeaderFileInfoLookupTable::Create(
|
|
|
|
(const unsigned char *)F.HeaderFileInfoTableData + Record[0],
|
2011-07-28 12:50:02 +08:00
|
|
|
(const unsigned char *)F.HeaderFileInfoTableData,
|
2011-07-29 04:55:49 +08:00
|
|
|
HeaderFileInfoTrait(*this, F,
|
2011-09-10 06:02:16 +08:00
|
|
|
&PP.getHeaderSearchInfo(),
|
2011-07-28 12:50:02 +08:00
|
|
|
BlobStart + Record[2]));
|
2011-09-10 06:02:16 +08:00
|
|
|
|
|
|
|
PP.getHeaderSearchInfo().SetExternalSource(this);
|
|
|
|
if (!PP.getHeaderSearchInfo().getExternalLookup())
|
|
|
|
PP.getHeaderSearchInfo().SetExternalLookup(this);
|
Implement two related optimizations that make de-serialization of
AST/PCH files more lazy:
- Don't preload all of the file source-location entries when reading
the AST file. Instead, load them lazily, when needed.
- Only look up header-search information (whether a header was already
#import'd, how many times it's been included, etc.) when it's needed
by the preprocessor, rather than pre-populating it.
Previously, we would pre-load all of the file source-location entries,
which also populated the header-search information structure. This was
a relatively minor performance issue, since we would end up stat()'ing
all of the headers stored within a AST/PCH file when the AST/PCH file
was loaded. In the normal PCH use case, the stat()s were cached, so
the cost--of preloading ~860 source-location entries in the Cocoa.h
case---was relatively low.
However, the recent optimization that replaced stat+open with
open+fstat turned this into a major problem, since the preloading of
source-location entries would now end up opening those files. Worse,
those files wouldn't be closed until the file manager was destroyed,
so just opening a Cocoa.h PCH file would hold on to ~860 file
descriptors, and it was easy to blow through the process's limit on
the number of open file descriptors.
By eliminating the preloading of these files, we neither open nor stat
the headers stored in the PCH/AST file until they're actually needed
for something. Concretely, we went from
*** HeaderSearch Stats:
835 files tracked.
364 #import/#pragma once files.
823 included exactly once.
6 max times a file is included.
3 #include/#include_next/#import.
0 #includes skipped due to the multi-include optimization.
1 framework lookups.
0 subframework lookups.
*** Source Manager Stats:
835 files mapped, 3 mem buffers mapped.
37460 SLocEntry's allocated, 11215575B of Sloc address space used.
62 bytes of files mapped, 0 files with line #'s computed.
with a trivial program that uses a chained PCH including a Cocoa PCH
to
*** HeaderSearch Stats:
4 files tracked.
1 #import/#pragma once files.
3 included exactly once.
2 max times a file is included.
3 #include/#include_next/#import.
0 #includes skipped due to the multi-include optimization.
1 framework lookups.
0 subframework lookups.
*** Source Manager Stats:
3 files mapped, 3 mem buffers mapped.
37460 SLocEntry's allocated, 11215575B of Sloc address space used.
62 bytes of files mapped, 0 files with line #'s computed.
for the same program.
llvm-svn: 125286
2011-02-11 01:09:37 +08:00
|
|
|
}
|
|
|
|
break;
|
2011-07-28 12:50:02 +08:00
|
|
|
}
|
|
|
|
|
2011-02-16 03:46:30 +08:00
|
|
|
case FP_PRAGMA_OPTIONS:
|
|
|
|
// Later tables overwrite earlier ones.
|
|
|
|
FPPragmaOptions.swap(Record);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case OPENCL_EXTENSIONS:
|
|
|
|
// Later tables overwrite earlier ones.
|
|
|
|
OpenCLExtensions.swap(Record);
|
|
|
|
break;
|
2011-05-05 07:29:54 +08:00
|
|
|
|
|
|
|
case TENTATIVE_DEFINITIONS:
|
2011-07-22 06:35:25 +08:00
|
|
|
for (unsigned I = 0, N = Record.size(); I != N; ++I)
|
|
|
|
TentativeDefinitions.push_back(getGlobalDeclID(F, Record[I]));
|
2011-05-05 07:29:54 +08:00
|
|
|
break;
|
2011-06-29 00:20:02 +08:00
|
|
|
|
|
|
|
case KNOWN_NAMESPACES:
|
2011-07-22 06:35:25 +08:00
|
|
|
for (unsigned I = 0, N = Record.size(); I != N; ++I)
|
|
|
|
KnownNamespaces.push_back(getGlobalDeclID(F, Record[I]));
|
2011-06-29 00:20:02 +08:00
|
|
|
break;
|
2011-12-03 08:59:55 +08:00
|
|
|
|
|
|
|
case IMPORTED_MODULES: {
|
|
|
|
if (F.Kind != MK_Module) {
|
|
|
|
// If we aren't loading a module (which has its own exports), make
|
|
|
|
// all of the imported modules visible.
|
|
|
|
// FIXME: Deal with macros-only imports.
|
|
|
|
for (unsigned I = 0, N = Record.size(); I != N; ++I) {
|
|
|
|
if (unsigned GlobalID = getGlobalSubmoduleID(F, Record[I]))
|
|
|
|
ImportedModules.push_back(GlobalID);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
Completely re-implement (de-)serialization of declaration
chains. The previous implementation relied heavily on the declaration
chain being stored as a (circular) linked list on disk, as it is in
memory. However, when deserializing from multiple modules, the
different chains could get mixed up, leading to broken declaration chains.
The new solution keeps track of the first and last declarations in the
chain for each module file. When we load a declaration, we search all
of the module files for redeclarations of that declaration, then
splice together all of the lists into a coherent whole (along with any
redeclarations that were actually parsed).
As a drive-by fix, (de-)serialize the redeclaration chains of
TypedefNameDecls, which had somehow gotten missed previously. Add a
test of this serialization.
This new scheme creates a redeclaration table that is fairly large in
the PCH file (on the order of 400k for Cocoa.h's 12MB PCH file). The
table is mmap'd in and searched via a binary search, but it's still
quite large. A future tweak will eliminate entries for declarations
that have no redeclarations anywhere, and should
drastically reduce the size of this table.
llvm-svn: 146841
2011-12-18 07:38:30 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
case LOCAL_REDECLARATIONS: {
|
|
|
|
if (F.LocalNumRedeclarationsInfos != 0) {
|
|
|
|
Error("duplicate LOCAL_REDECLARATIONS record in AST file");
|
|
|
|
return Failure;
|
|
|
|
}
|
2011-12-03 08:59:55 +08:00
|
|
|
|
Completely re-implement (de-)serialization of declaration
chains. The previous implementation relied heavily on the declaration
chain being stored as a (circular) linked list on disk, as it is in
memory. However, when deserializing from multiple modules, the
different chains could get mixed up, leading to broken declaration chains.
The new solution keeps track of the first and last declarations in the
chain for each module file. When we load a declaration, we search all
of the module files for redeclarations of that declaration, then
splice together all of the lists into a coherent whole (along with any
redeclarations that were actually parsed).
As a drive-by fix, (de-)serialize the redeclaration chains of
TypedefNameDecls, which had somehow gotten missed previously. Add a
test of this serialization.
This new scheme creates a redeclaration table that is fairly large in
the PCH file (on the order of 400k for Cocoa.h's 12MB PCH file). The
table is mmap'd in and searched via a binary search, but it's still
quite large. A future tweak will eliminate entries for declarations
that have no redeclarations anywhere, and should
drastically reduce the size of this table.
llvm-svn: 146841
2011-12-18 07:38:30 +08:00
|
|
|
F.LocalNumRedeclarationsInfos = Record[0];
|
|
|
|
F.RedeclarationsInfo = (const LocalRedeclarationsInfo *)BlobStart;
|
|
|
|
break;
|
2011-12-03 08:59:55 +08:00
|
|
|
}
|
2009-04-11 08:14:32 +08:00
|
|
|
}
|
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
|
|
|
}
|
|
|
|
|
2011-12-01 07:21:26 +08:00
|
|
|
ASTReader::ASTReadResult ASTReader::validateFileEntries(ModuleFile &M) {
|
2011-08-26 04:58:51 +08:00
|
|
|
llvm::BitstreamCursor &SLocEntryCursor = M.SLocEntryCursor;
|
|
|
|
|
|
|
|
for (unsigned i = 0, e = M.LocalNumSLocFileEntries; i != e; ++i) {
|
|
|
|
SLocEntryCursor.JumpToBit(M.SLocFileOffsets[i]);
|
|
|
|
unsigned Code = SLocEntryCursor.ReadCode();
|
|
|
|
if (Code == llvm::bitc::END_BLOCK ||
|
|
|
|
Code == llvm::bitc::ENTER_SUBBLOCK ||
|
|
|
|
Code == llvm::bitc::DEFINE_ABBREV) {
|
|
|
|
Error("incorrectly-formatted source location entry in AST file");
|
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
|
|
|
|
RecordData Record;
|
|
|
|
const char *BlobStart;
|
|
|
|
unsigned BlobLen;
|
|
|
|
switch (SLocEntryCursor.ReadRecord(Code, Record, &BlobStart, &BlobLen)) {
|
|
|
|
default:
|
|
|
|
Error("incorrectly-formatted source location entry in AST file");
|
|
|
|
return Failure;
|
|
|
|
|
|
|
|
case SM_SLOC_FILE_ENTRY: {
|
2011-11-17 04:05:18 +08:00
|
|
|
// If the buffer was overridden, the file need not exist.
|
|
|
|
if (Record[6])
|
|
|
|
break;
|
|
|
|
|
2011-08-26 04:58:51 +08:00
|
|
|
StringRef Filename(BlobStart, BlobLen);
|
|
|
|
const FileEntry *File = getFileEntry(Filename);
|
|
|
|
|
|
|
|
if (File == 0) {
|
|
|
|
std::string ErrorStr = "could not find file '";
|
|
|
|
ErrorStr += Filename;
|
|
|
|
ErrorStr += "' referenced by AST file";
|
|
|
|
Error(ErrorStr.c_str());
|
|
|
|
return IgnorePCH;
|
2011-06-01 13:43:53 +08:00
|
|
|
}
|
2011-08-26 04:58:51 +08:00
|
|
|
|
2011-11-17 04:05:18 +08:00
|
|
|
if (Record.size() < 7) {
|
2011-08-26 04:58:51 +08:00
|
|
|
Error("source location entry is incorrect");
|
2011-06-01 13:43:53 +08:00
|
|
|
return Failure;
|
2011-08-26 04:58:51 +08:00
|
|
|
}
|
2011-06-01 13:43:53 +08:00
|
|
|
|
2011-08-26 04:58:51 +08:00
|
|
|
// The stat info from the FileEntry came from the cached stat
|
|
|
|
// info of the PCH, so we cannot trust it.
|
|
|
|
struct stat StatBuf;
|
|
|
|
if (::stat(File->getName(), &StatBuf) != 0) {
|
|
|
|
StatBuf.st_size = File->getSize();
|
|
|
|
StatBuf.st_mtime = File->getModificationTime();
|
|
|
|
}
|
2011-06-01 13:43:53 +08:00
|
|
|
|
2011-08-26 04:58:51 +08:00
|
|
|
if (((off_t)Record[4] != StatBuf.st_size
|
2011-06-01 13:43:53 +08:00
|
|
|
#if !defined(LLVM_ON_WIN32)
|
2011-08-26 04:58:51 +08:00
|
|
|
// In our regression testing, the Windows file system seems to
|
|
|
|
// have inconsistent modification times that sometimes
|
|
|
|
// erroneously trigger this error-handling path.
|
|
|
|
|| (time_t)Record[5] != StatBuf.st_mtime
|
2011-06-01 13:43:53 +08:00
|
|
|
#endif
|
2011-08-26 04:58:51 +08:00
|
|
|
)) {
|
|
|
|
Error(diag::err_fe_pch_file_modified, Filename);
|
|
|
|
return IgnorePCH;
|
2011-06-01 13:43:53 +08:00
|
|
|
}
|
2011-08-26 04:58:51 +08:00
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
2011-06-01 13:43:53 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Success;
|
|
|
|
}
|
|
|
|
|
2011-12-02 06:20:10 +08:00
|
|
|
void ASTReader::makeNamesVisible(const HiddenNames &Names) {
|
2011-12-02 23:45:10 +08:00
|
|
|
for (unsigned I = 0, N = Names.size(); I != N; ++I) {
|
|
|
|
if (Decl *D = Names[I].dyn_cast<Decl *>())
|
|
|
|
D->ModulePrivate = false;
|
|
|
|
else
|
|
|
|
Names[I].get<IdentifierInfo *>()->setHasMacroDefinition(true);
|
|
|
|
}
|
2011-12-02 06:20:10 +08:00
|
|
|
}
|
|
|
|
|
2011-12-02 01:11:21 +08:00
|
|
|
void ASTReader::makeModuleVisible(Module *Mod,
|
|
|
|
Module::NameVisibilityKind NameVisibility) {
|
|
|
|
llvm::SmallPtrSet<Module *, 4> Visited;
|
|
|
|
llvm::SmallVector<Module *, 4> Stack;
|
|
|
|
Stack.push_back(Mod);
|
|
|
|
while (!Stack.empty()) {
|
|
|
|
Mod = Stack.back();
|
|
|
|
Stack.pop_back();
|
|
|
|
|
|
|
|
if (NameVisibility <= Mod->NameVisibility) {
|
|
|
|
// This module already has this level of visibility (or greater), so
|
|
|
|
// there is nothing more to do.
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update the module's name visibility.
|
|
|
|
Mod->NameVisibility = NameVisibility;
|
|
|
|
|
2011-12-02 06:20:10 +08:00
|
|
|
// If we've already deserialized any names from this module,
|
2011-12-02 01:11:21 +08:00
|
|
|
// mark them as visible.
|
2011-12-02 06:20:10 +08:00
|
|
|
HiddenNamesMapType::iterator Hidden = HiddenNamesMap.find(Mod);
|
|
|
|
if (Hidden != HiddenNamesMap.end()) {
|
|
|
|
makeNamesVisible(Hidden->second);
|
|
|
|
HiddenNamesMap.erase(Hidden);
|
|
|
|
}
|
2011-12-02 01:11:21 +08:00
|
|
|
|
|
|
|
// Push any non-explicit submodules onto the stack to be marked as
|
|
|
|
// visible.
|
|
|
|
for (llvm::StringMap<Module *>::iterator Sub = Mod->SubModules.begin(),
|
|
|
|
SubEnd = Mod->SubModules.end();
|
|
|
|
Sub != SubEnd; ++Sub) {
|
|
|
|
if (!Sub->getValue()->IsExplicit && Visited.insert(Sub->getValue()))
|
|
|
|
Stack.push_back(Sub->getValue());
|
|
|
|
}
|
2011-12-03 03:11:09 +08:00
|
|
|
|
|
|
|
// Push any exported modules onto the stack to be marked as visible.
|
2011-12-06 01:28:06 +08:00
|
|
|
bool AnyWildcard = false;
|
|
|
|
bool UnrestrictedWildcard = false;
|
|
|
|
llvm::SmallVector<Module *, 4> WildcardRestrictions;
|
2011-12-03 03:11:09 +08:00
|
|
|
for (unsigned I = 0, N = Mod->Exports.size(); I != N; ++I) {
|
|
|
|
Module *Exported = Mod->Exports[I].getPointer();
|
2011-12-06 01:28:06 +08:00
|
|
|
if (!Mod->Exports[I].getInt()) {
|
|
|
|
// Export a named module directly; no wildcards involved.
|
|
|
|
if (Visited.insert(Exported))
|
2011-12-03 03:11:09 +08:00
|
|
|
Stack.push_back(Exported);
|
2011-12-06 01:28:06 +08:00
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Wildcard export: export all of the imported modules that match
|
|
|
|
// the given pattern.
|
|
|
|
AnyWildcard = true;
|
|
|
|
if (UnrestrictedWildcard)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (Module *Restriction = Mod->Exports[I].getPointer())
|
|
|
|
WildcardRestrictions.push_back(Restriction);
|
|
|
|
else {
|
|
|
|
WildcardRestrictions.clear();
|
|
|
|
UnrestrictedWildcard = true;
|
2011-12-03 03:11:09 +08:00
|
|
|
}
|
|
|
|
}
|
2011-12-06 01:28:06 +08:00
|
|
|
|
|
|
|
// If there were any wildcards, push any imported modules that were
|
|
|
|
// re-exported by the wildcard restriction.
|
|
|
|
if (!AnyWildcard)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
for (unsigned I = 0, N = Mod->Imports.size(); I != N; ++I) {
|
|
|
|
Module *Imported = Mod->Imports[I];
|
|
|
|
if (Visited.count(Imported))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
bool Acceptable = UnrestrictedWildcard;
|
|
|
|
if (!Acceptable) {
|
|
|
|
// Check whether this module meets one of the restrictions.
|
|
|
|
for (unsigned R = 0, NR = WildcardRestrictions.size(); R != NR; ++R) {
|
|
|
|
Module *Restriction = WildcardRestrictions[R];
|
|
|
|
if (Imported == Restriction || Imported->isSubModuleOf(Restriction)) {
|
|
|
|
Acceptable = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!Acceptable)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
Visited.insert(Imported);
|
|
|
|
Stack.push_back(Imported);
|
|
|
|
}
|
2011-12-02 01:11:21 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-10-06 00:15:19 +08:00
|
|
|
ASTReader::ASTReadResult ASTReader::ReadAST(const std::string &FileName,
|
2011-07-23 00:00:58 +08:00
|
|
|
ModuleKind Type) {
|
2011-08-18 12:12:04 +08:00
|
|
|
switch(ReadASTCore(FileName, Type, /*ImportedBy=*/0)) {
|
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.
|
2011-07-22 02:46:38 +08:00
|
|
|
|
2010-07-17 04:41:52 +08:00
|
|
|
// Check the predefines buffers.
|
2011-12-03 05:56:05 +08:00
|
|
|
if (!DisableValidation && Type == MK_PCH &&
|
2011-09-13 02:09:38 +08:00
|
|
|
// FIXME: CheckPredefinesBuffers also sets the SuggestedPredefines;
|
|
|
|
// if DisableValidation is true, defines that were set on command-line
|
|
|
|
// but not in the PCH file will not be added to SuggestedPredefines.
|
|
|
|
CheckPredefinesBuffers())
|
2009-04-29 04:33:11 +08:00
|
|
|
return IgnorePCH;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
Make the loading of information attached to an IdentifierInfo from an
AST file more lazy, so that we don't eagerly load that information for
all known identifiers each time a new AST file is loaded. The eager
reloading made some sense in the context of precompiled headers, since
very few identifiers were defined before PCH load time. With modules,
however, a huge amount of code can get parsed before we see an
@import, so laziness becomes important here.
The approach taken to make this information lazy is fairly simple:
when we load a new AST file, we mark all of the existing identifiers
as being out-of-date. Whenever we want to access information that may
come from an AST (e.g., whether the identifier has a macro definition,
or what top-level declarations have that name), we check the
out-of-date bit and, if it's set, ask the AST reader to update the
IdentifierInfo from the AST files. The update is a merge, and we now
take care to merge declarations before/after imports with declarations
from multiple imports.
The results of this optimization are fairly dramatic. On a small
application that brings in 14 non-trivial modules, this takes modules
from being > 3x slower than a "perfect" PCH file down to 30% slower
for a full rebuild. A partial rebuild (where the PCH file or modules
can be re-used) is down to 7% slower. Making the PCH file just a
little imperfect (e.g., adding two smallish modules used by a bunch of
.m files that aren't in the PCH file) tips the scales in favor of the
modules approach, with 24% faster partial rebuilds.
This is just a first step; the lazy scheme could possibly be improved
by adding versioning, so we don't search into modules we already
searched. Moreover, we'll need similar lazy schemes for all of the
other lookup data structures, such as DeclContexts.
llvm-svn: 143100
2011-10-27 17:33:13 +08:00
|
|
|
// Mark all of the identifiers in the identifier table as being out of date,
|
|
|
|
// so that various accessors know to check the loaded modules when the
|
|
|
|
// identifier is used.
|
2011-09-10 06:02:16 +08:00
|
|
|
for (IdentifierTable::iterator Id = PP.getIdentifierTable().begin(),
|
|
|
|
IdEnd = PP.getIdentifierTable().end();
|
|
|
|
Id != IdEnd; ++Id)
|
Make the loading of information attached to an IdentifierInfo from an
AST file more lazy, so that we don't eagerly load that information for
all known identifiers each time a new AST file is loaded. The eager
reloading made some sense in the context of precompiled headers, since
very few identifiers were defined before PCH load time. With modules,
however, a huge amount of code can get parsed before we see an
@import, so laziness becomes important here.
The approach taken to make this information lazy is fairly simple:
when we load a new AST file, we mark all of the existing identifiers
as being out-of-date. Whenever we want to access information that may
come from an AST (e.g., whether the identifier has a macro definition,
or what top-level declarations have that name), we check the
out-of-date bit and, if it's set, ask the AST reader to update the
IdentifierInfo from the AST files. The update is a merge, and we now
take care to merge declarations before/after imports with declarations
from multiple imports.
The results of this optimization are fairly dramatic. On a small
application that brings in 14 non-trivial modules, this takes modules
from being > 3x slower than a "perfect" PCH file down to 30% slower
for a full rebuild. A partial rebuild (where the PCH file or modules
can be re-used) is down to 7% slower. Making the PCH file just a
little imperfect (e.g., adding two smallish modules used by a bunch of
.m files that aren't in the PCH file) tips the scales in favor of the
modules approach, with 24% faster partial rebuilds.
This is just a first step; the lazy scheme could possibly be improved
by adding versioning, so we don't search into modules we already
searched. Moreover, we'll need similar lazy schemes for all of the
other lookup data structures, such as DeclContexts.
llvm-svn: 143100
2011-10-27 17:33:13 +08:00
|
|
|
Id->second->setOutOfDate(true);
|
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
|
|
|
|
2011-12-03 02:58:38 +08:00
|
|
|
// Resolve any unresolved module exports.
|
2011-12-06 00:33:54 +08:00
|
|
|
for (unsigned I = 0, N = UnresolvedModuleImportExports.size(); I != N; ++I) {
|
|
|
|
UnresolvedModuleImportExport &Unresolved = UnresolvedModuleImportExports[I];
|
|
|
|
SubmoduleID GlobalID = getGlobalSubmoduleID(*Unresolved.File,Unresolved.ID);
|
2011-12-06 01:28:06 +08:00
|
|
|
Module *ResolvedMod = getSubmodule(GlobalID);
|
|
|
|
|
|
|
|
if (Unresolved.IsImport) {
|
|
|
|
if (ResolvedMod)
|
2011-12-06 00:33:54 +08:00
|
|
|
Unresolved.Mod->Imports.push_back(ResolvedMod);
|
2011-12-06 01:28:06 +08:00
|
|
|
continue;
|
2011-12-03 02:58:38 +08:00
|
|
|
}
|
2011-12-06 01:28:06 +08:00
|
|
|
|
|
|
|
if (ResolvedMod || Unresolved.IsWildcard)
|
|
|
|
Unresolved.Mod->Exports.push_back(
|
|
|
|
Module::ExportDecl(ResolvedMod, Unresolved.IsWildcard));
|
2011-12-03 02:58:38 +08:00
|
|
|
}
|
2011-12-06 00:33:54 +08:00
|
|
|
UnresolvedModuleImportExports.clear();
|
2011-12-03 02:58:38 +08:00
|
|
|
|
2011-09-10 05:34:22 +08:00
|
|
|
InitializeContext();
|
2009-04-15 05:18:50 +08:00
|
|
|
|
2010-10-25 01:26:36 +08:00
|
|
|
if (DeserializationListener)
|
|
|
|
DeserializationListener->ReaderInitialized(this);
|
|
|
|
|
2011-09-20 04:40:35 +08:00
|
|
|
// If this AST file is a precompiled preamble, then set the preamble file ID
|
|
|
|
// of the source manager to the file source file from which the preamble was
|
|
|
|
// built.
|
2011-07-23 00:00:58 +08:00
|
|
|
if (Type == MK_Preamble) {
|
2011-09-20 04:40:35 +08:00
|
|
|
if (!OriginalFileID.isInvalid()) {
|
2011-07-26 04:32:21 +08:00
|
|
|
OriginalFileID = FileID::get(ModuleMgr.getPrimaryModule().SLocEntryBaseID
|
Revamp the SourceManager to separate the representation of parsed
source locations from source locations loaded from an AST/PCH file.
Previously, loading an AST/PCH file involved carefully pre-allocating
space at the beginning of the source manager for the source locations
and FileIDs that correspond to the prefix, and then appending the
source locations/FileIDs used for parsing the remaining translation
unit. This design forced us into loading PCH files early, as a prefix,
whic has become a rather significant limitation.
This patch splits the SourceManager space into two parts: for source
location "addresses", the lower values (growing upward) are used to
describe parsed code, while upper values (growing downward) are used
for source locations loaded from AST/PCH files. Similarly, positive
FileIDs are used to describe parsed code while negative FileIDs are
used to file/macro locations loaded from AST/PCH files. As a result,
we can load PCH/AST files even during parsing, making various
improvemnts in the future possible, e.g., teaching #include <foo.h> to
look for and load <foo.h.gch> if it happens to be already available.
This patch was originally written by Sebastian Redl, then brought
forward to the modern age by Jonathan Turner, and finally
polished/finished by me to be committed.
llvm-svn: 135484
2011-07-20 00:10:42 +08:00
|
|
|
+ OriginalFileID.getOpaqueValue() - 1);
|
2011-09-20 04:40:35 +08:00
|
|
|
SourceMgr.setPreambleFileID(OriginalFileID);
|
Revamp the SourceManager to separate the representation of parsed
source locations from source locations loaded from an AST/PCH file.
Previously, loading an AST/PCH file involved carefully pre-allocating
space at the beginning of the source manager for the source locations
and FileIDs that correspond to the prefix, and then appending the
source locations/FileIDs used for parsing the remaining translation
unit. This design forced us into loading PCH files early, as a prefix,
whic has become a rather significant limitation.
This patch splits the SourceManager space into two parts: for source
location "addresses", the lower values (growing upward) are used to
describe parsed code, while upper values (growing downward) are used
for source locations loaded from AST/PCH files. Similarly, positive
FileIDs are used to describe parsed code while negative FileIDs are
used to file/macro locations loaded from AST/PCH files. As a result,
we can load PCH/AST files even during parsing, making various
improvemnts in the future possible, e.g., teaching #include <foo.h> to
look for and load <foo.h.gch> if it happens to be already available.
This patch was originally written by Sebastian Redl, then brought
forward to the modern age by Jonathan Turner, and finally
polished/finished by me to be committed.
llvm-svn: 135484
2011-07-20 00:10:42 +08:00
|
|
|
}
|
2010-11-30 13:23:00 +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
|
|
|
}
|
|
|
|
|
2011-07-23 18:55:15 +08:00
|
|
|
ASTReader::ASTReadResult ASTReader::ReadASTCore(StringRef FileName,
|
2011-08-18 12:12:04 +08:00
|
|
|
ModuleKind Type,
|
2011-12-01 07:21:26 +08:00
|
|
|
ModuleFile *ImportedBy) {
|
|
|
|
ModuleFile *M;
|
2011-08-19 10:29:29 +08:00
|
|
|
bool NewModule;
|
|
|
|
std::string ErrorStr;
|
|
|
|
llvm::tie(M, NewModule) = ModuleMgr.addModule(FileName, Type, ImportedBy,
|
|
|
|
ErrorStr);
|
|
|
|
|
|
|
|
if (!M) {
|
|
|
|
// We couldn't load the module.
|
|
|
|
std::string Msg = "Unable to load module \"" + FileName.str() + "\": "
|
|
|
|
+ ErrorStr;
|
|
|
|
Error(Msg);
|
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!NewModule) {
|
|
|
|
// We've already loaded this module.
|
|
|
|
return Success;
|
|
|
|
}
|
2010-07-17 04:41:52 +08:00
|
|
|
|
2011-08-19 10:29:29 +08:00
|
|
|
// FIXME: This seems rather a hack. Should CurrentDir be part of the
|
|
|
|
// module?
|
2011-02-16 01:54:22 +08:00
|
|
|
if (FileName != "-") {
|
|
|
|
CurrentDir = llvm::sys::path::parent_path(FileName);
|
|
|
|
if (CurrentDir.empty()) CurrentDir = ".";
|
|
|
|
}
|
|
|
|
|
2011-12-01 07:21:26 +08:00
|
|
|
ModuleFile &F = *M;
|
2010-07-17 04:41:52 +08:00
|
|
|
llvm::BitstreamCursor &Stream = F.Stream;
|
|
|
|
Stream.init(F.StreamFile);
|
2010-07-22 04:07:32 +08:00
|
|
|
F.SizeInBits = F.Buffer->getBufferSize() * 8;
|
Introduce a global bit-offset continuous range map into the ASTReader,
so that we have one, simple way to map from global bit offsets to
local bit offsets. Eliminates a number of loops over the chain, and
generalizes for more interesting bit remappings.
Also, as an amusing oddity, we were computing global bit offsets
*backwards* for preprocessed entities (e.g., the directly included PCH
file in the chain would start at offset zero, rather than the original
PCH that occurs first in translation unit). Even more amusingly, it
made precompiled preambles work, because we were forgetting to adjust
the local bit offset to a global bit offset when storing preprocessed
entity offsets in the ASTUnit. Two wrongs made a right, and now
they're both right.
llvm-svn: 135750
2011-07-22 14:10:01 +08:00
|
|
|
|
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
|
|
|
|
Revamp the SourceManager to separate the representation of parsed
source locations from source locations loaded from an AST/PCH file.
Previously, loading an AST/PCH file involved carefully pre-allocating
space at the beginning of the source manager for the source locations
and FileIDs that correspond to the prefix, and then appending the
source locations/FileIDs used for parsing the remaining translation
unit. This design forced us into loading PCH files early, as a prefix,
whic has become a rather significant limitation.
This patch splits the SourceManager space into two parts: for source
location "addresses", the lower values (growing upward) are used to
describe parsed code, while upper values (growing downward) are used
for source locations loaded from AST/PCH files. Similarly, positive
FileIDs are used to describe parsed code while negative FileIDs are
used to file/macro locations loaded from AST/PCH files. As a result,
we can load PCH/AST files even during parsing, making various
improvemnts in the future possible, e.g., teaching #include <foo.h> to
look for and load <foo.h.gch> if it happens to be already available.
This patch was originally written by Sebastian Redl, then brought
forward to the modern age by Jonathan Turner, and finally
polished/finished by me to be committed.
llvm-svn: 135484
2011-07-20 00:10:42 +08:00
|
|
|
// FIXME: We can't clear loaded slocentries anymore.
|
|
|
|
//SourceMgr.ClearPreallocatedSLocEntries();
|
2010-07-17 08:12:06 +08:00
|
|
|
|
|
|
|
// 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
|
|
|
}
|
Introduce a global bit-offset continuous range map into the ASTReader,
so that we have one, simple way to map from global bit offsets to
local bit offsets. Eliminates a number of loops over the chain, and
generalizes for more interesting bit remappings.
Also, as an amusing oddity, we were computing global bit offsets
*backwards* for preprocessed entities (e.g., the directly included PCH
file in the chain would start at offset zero, rather than the original
PCH that occurs first in translation unit). Even more amusingly, it
made precompiled preambles work, because we were forgetting to adjust
the local bit offset to a global bit offset when storing preprocessed
entity offsets in the ASTUnit. Two wrongs made a right, and now
they're both right.
llvm-svn: 135750
2011-07-22 14:10:01 +08:00
|
|
|
|
2011-12-01 07:21:26 +08:00
|
|
|
// Once read, set the ModuleFile bit base offset and update the size in
|
Introduce a global bit-offset continuous range map into the ASTReader,
so that we have one, simple way to map from global bit offsets to
local bit offsets. Eliminates a number of loops over the chain, and
generalizes for more interesting bit remappings.
Also, as an amusing oddity, we were computing global bit offsets
*backwards* for preprocessed entities (e.g., the directly included PCH
file in the chain would start at offset zero, rather than the original
PCH that occurs first in translation unit). Even more amusingly, it
made precompiled preambles work, because we were forgetting to adjust
the local bit offset to a global bit offset when storing preprocessed
entity offsets in the ASTUnit. Two wrongs made a right, and now
they're both right.
llvm-svn: 135750
2011-07-22 14:10:01 +08:00
|
|
|
// bits of all files we've seen.
|
|
|
|
F.GlobalBitOffset = TotalModulesSizeInBits;
|
|
|
|
TotalModulesSizeInBits += F.SizeInBits;
|
|
|
|
GlobalBitOffsetsMap.insert(std::make_pair(F.GlobalBitOffset, &F));
|
2011-08-26 04:58:51 +08:00
|
|
|
|
|
|
|
// Make sure that the files this module was built against are still available.
|
|
|
|
if (!DisableValidation) {
|
|
|
|
switch(validateFileEntries(*M)) {
|
|
|
|
case Failure: return Failure;
|
|
|
|
case IgnorePCH: return IgnorePCH;
|
|
|
|
case Success: break;
|
|
|
|
}
|
|
|
|
}
|
2011-08-26 05:09:44 +08:00
|
|
|
|
|
|
|
// Preload SLocEntries.
|
|
|
|
for (unsigned I = 0, N = M->PreloadSLocEntries.size(); I != N; ++I) {
|
|
|
|
int Index = int(M->PreloadSLocEntries[I] - 1) + F.SLocEntryBaseID;
|
2011-09-20 04:39:54 +08:00
|
|
|
// Load it through the SourceManager and don't call ReadSLocEntryRecord()
|
|
|
|
// directly because the entry may have already been loaded in which case
|
|
|
|
// calling ReadSLocEntryRecord() directly would trigger an assertion in
|
|
|
|
// SourceManager.
|
|
|
|
SourceMgr.getLoadedSLocEntryByID(Index);
|
2011-08-26 05:09:44 +08:00
|
|
|
}
|
|
|
|
|
2011-08-26 04:58:51 +08:00
|
|
|
|
2010-07-17 04:41:52 +08:00
|
|
|
return Success;
|
|
|
|
}
|
|
|
|
|
2011-09-10 06:02:16 +08:00
|
|
|
void ASTReader::InitializeContext() {
|
2011-08-12 08:15:20 +08:00
|
|
|
// If there's a listener, notify them that we "read" the translation unit.
|
|
|
|
if (DeserializationListener)
|
2011-09-10 05:34:22 +08:00
|
|
|
DeserializationListener->DeclRead(PREDEF_DECL_TRANSLATION_UNIT_ID,
|
|
|
|
Context.getTranslationUnitDecl());
|
2010-10-01 09:18:02 +08:00
|
|
|
|
2011-08-12 08:15:20 +08:00
|
|
|
// Make sure we load the declaration update records for the translation unit,
|
|
|
|
// if there are any.
|
2011-09-10 05:34:22 +08:00
|
|
|
loadDeclUpdateRecords(PREDEF_DECL_TRANSLATION_UNIT_ID,
|
|
|
|
Context.getTranslationUnitDecl());
|
2011-08-12 08:15:20 +08:00
|
|
|
|
2011-08-12 06:18:49 +08:00
|
|
|
// FIXME: Find a better way to deal with collisions between these
|
|
|
|
// built-in types. Right now, we just ignore the problem.
|
|
|
|
|
|
|
|
// Load the special types.
|
2011-09-10 08:30:18 +08:00
|
|
|
if (SpecialTypes.size() > NumSpecialTypeIDs) {
|
|
|
|
if (Context.getBuiltinVaListType().isNull()) {
|
|
|
|
Context.setBuiltinVaListType(
|
|
|
|
GetType(SpecialTypes[SPECIAL_TYPE_BUILTIN_VA_LIST]));
|
2011-08-12 06:18:49 +08:00
|
|
|
}
|
|
|
|
|
2011-09-10 08:30:18 +08:00
|
|
|
if (unsigned Proto = SpecialTypes[SPECIAL_TYPE_OBJC_PROTOCOL]) {
|
|
|
|
if (Context.ObjCProtoType.isNull())
|
|
|
|
Context.ObjCProtoType = GetType(Proto);
|
2009-07-08 00:35:42 +08:00
|
|
|
}
|
2011-09-10 08:30:18 +08:00
|
|
|
|
|
|
|
if (unsigned String = SpecialTypes[SPECIAL_TYPE_CF_CONSTANT_STRING]) {
|
|
|
|
if (!Context.CFConstantStringTypeDecl)
|
|
|
|
Context.setCFConstantStringType(GetType(String));
|
2011-08-12 06:18:49 +08:00
|
|
|
}
|
|
|
|
|
2011-09-10 08:30:18 +08:00
|
|
|
if (unsigned File = SpecialTypes[SPECIAL_TYPE_FILE]) {
|
|
|
|
QualType FileType = GetType(File);
|
|
|
|
if (FileType.isNull()) {
|
|
|
|
Error("FILE type is NULL");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!Context.FILEDecl) {
|
|
|
|
if (const TypedefType *Typedef = FileType->getAs<TypedefType>())
|
|
|
|
Context.setFILEDecl(Typedef->getDecl());
|
|
|
|
else {
|
|
|
|
const TagType *Tag = FileType->getAs<TagType>();
|
|
|
|
if (!Tag) {
|
|
|
|
Error("Invalid FILE type in AST file");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
Context.setFILEDecl(Tag->getDecl());
|
2011-08-06 07:07:10 +08:00
|
|
|
}
|
|
|
|
}
|
2010-03-18 08:56:54 +08:00
|
|
|
}
|
2011-09-10 08:30:18 +08:00
|
|
|
|
2011-11-12 03:13:12 +08:00
|
|
|
if (unsigned Jmp_buf = SpecialTypes[SPECIAL_TYPE_JMP_BUF]) {
|
2011-09-10 08:30:18 +08:00
|
|
|
QualType Jmp_bufType = GetType(Jmp_buf);
|
|
|
|
if (Jmp_bufType.isNull()) {
|
|
|
|
Error("jmp_buf type is NULL");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!Context.jmp_bufDecl) {
|
|
|
|
if (const TypedefType *Typedef = Jmp_bufType->getAs<TypedefType>())
|
|
|
|
Context.setjmp_bufDecl(Typedef->getDecl());
|
|
|
|
else {
|
|
|
|
const TagType *Tag = Jmp_bufType->getAs<TagType>();
|
|
|
|
if (!Tag) {
|
|
|
|
Error("Invalid jmp_buf type in AST file");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
Context.setjmp_bufDecl(Tag->getDecl());
|
|
|
|
}
|
|
|
|
}
|
2011-08-12 06:18:49 +08:00
|
|
|
}
|
|
|
|
|
2011-11-12 03:13:12 +08:00
|
|
|
if (unsigned Sigjmp_buf = SpecialTypes[SPECIAL_TYPE_SIGJMP_BUF]) {
|
2011-09-10 08:30:18 +08:00
|
|
|
QualType Sigjmp_bufType = GetType(Sigjmp_buf);
|
|
|
|
if (Sigjmp_bufType.isNull()) {
|
|
|
|
Error("sigjmp_buf type is NULL");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!Context.sigjmp_bufDecl) {
|
|
|
|
if (const TypedefType *Typedef = Sigjmp_bufType->getAs<TypedefType>())
|
|
|
|
Context.setsigjmp_bufDecl(Typedef->getDecl());
|
|
|
|
else {
|
|
|
|
const TagType *Tag = Sigjmp_bufType->getAs<TagType>();
|
|
|
|
assert(Tag && "Invalid sigjmp_buf type in AST file");
|
|
|
|
Context.setsigjmp_bufDecl(Tag->getDecl());
|
|
|
|
}
|
2011-08-06 07:07:10 +08:00
|
|
|
}
|
2009-07-28 10:25:19 +08:00
|
|
|
}
|
2011-08-12 06:18:49 +08:00
|
|
|
|
2011-09-10 08:30:18 +08:00
|
|
|
if (unsigned ObjCIdRedef
|
|
|
|
= SpecialTypes[SPECIAL_TYPE_OBJC_ID_REDEFINITION]) {
|
|
|
|
if (Context.ObjCIdRedefinitionType.isNull())
|
|
|
|
Context.ObjCIdRedefinitionType = GetType(ObjCIdRedef);
|
|
|
|
}
|
2011-08-12 06:18:49 +08:00
|
|
|
|
2011-09-10 08:30:18 +08:00
|
|
|
if (unsigned ObjCClassRedef
|
|
|
|
= SpecialTypes[SPECIAL_TYPE_OBJC_CLASS_REDEFINITION]) {
|
|
|
|
if (Context.ObjCClassRedefinitionType.isNull())
|
|
|
|
Context.ObjCClassRedefinitionType = GetType(ObjCClassRedef);
|
|
|
|
}
|
2011-08-06 07:07:10 +08:00
|
|
|
|
2011-09-10 08:30:18 +08:00
|
|
|
if (unsigned ObjCSelRedef
|
|
|
|
= SpecialTypes[SPECIAL_TYPE_OBJC_SEL_REDEFINITION]) {
|
|
|
|
if (Context.ObjCSelRedefinitionType.isNull())
|
|
|
|
Context.ObjCSelRedefinitionType = GetType(ObjCSelRedef);
|
|
|
|
}
|
2011-11-14 05:51:09 +08:00
|
|
|
|
|
|
|
if (unsigned Ucontext_t = SpecialTypes[SPECIAL_TYPE_UCONTEXT_T]) {
|
|
|
|
QualType Ucontext_tType = GetType(Ucontext_t);
|
|
|
|
if (Ucontext_tType.isNull()) {
|
|
|
|
Error("ucontext_t type is NULL");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!Context.ucontext_tDecl) {
|
|
|
|
if (const TypedefType *Typedef = Ucontext_tType->getAs<TypedefType>())
|
|
|
|
Context.setucontext_tDecl(Typedef->getDecl());
|
|
|
|
else {
|
|
|
|
const TagType *Tag = Ucontext_tType->getAs<TagType>();
|
|
|
|
assert(Tag && "Invalid ucontext_t type in AST file");
|
|
|
|
Context.setucontext_tDecl(Tag->getDecl());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2009-07-28 10:25:19 +08:00
|
|
|
}
|
2011-09-10 08:30:18 +08:00
|
|
|
|
2011-09-10 05:34:22 +08:00
|
|
|
ReadPragmaDiagnosticMappings(Context.getDiagnostics());
|
2011-02-10 05:04:32 +08:00
|
|
|
|
|
|
|
// If there were any CUDA special declarations, deserialize them.
|
|
|
|
if (!CUDASpecialDeclRefs.empty()) {
|
|
|
|
assert(CUDASpecialDeclRefs.size() == 1 && "More decl refs than expected!");
|
2011-09-10 05:34:22 +08:00
|
|
|
Context.setcudaConfigureCallDecl(
|
2011-02-10 05:04:32 +08:00
|
|
|
cast<FunctionDecl>(GetDecl(CUDASpecialDeclRefs[0])));
|
|
|
|
}
|
2011-12-03 08:59:55 +08:00
|
|
|
|
|
|
|
// Re-export any modules that were imported by a non-module AST file.
|
|
|
|
for (unsigned I = 0, N = ImportedModules.size(); I != N; ++I) {
|
|
|
|
if (Module *Imported = getSubmodule(ImportedModules[I]))
|
|
|
|
makeModuleVisible(Imported, Module::AllVisible);
|
|
|
|
}
|
|
|
|
ImportedModules.clear();
|
2009-06-19 08:03:23 +08:00
|
|
|
}
|
|
|
|
|
2011-12-02 06:20:10 +08:00
|
|
|
void ASTReader::finalizeForWriting() {
|
|
|
|
for (HiddenNamesMapType::iterator Hidden = HiddenNamesMap.begin(),
|
|
|
|
HiddenEnd = HiddenNamesMap.end();
|
|
|
|
Hidden != HiddenEnd; ++Hidden) {
|
|
|
|
makeNamesVisible(Hidden->second);
|
|
|
|
}
|
|
|
|
HiddenNamesMap.clear();
|
|
|
|
}
|
|
|
|
|
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,
|
2010-11-04 06:45:23 +08:00
|
|
|
FileManager &FileMgr,
|
2011-09-26 07:23:43 +08:00
|
|
|
DiagnosticsEngine &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-11-23 16:35:12 +08:00
|
|
|
Buffer.reset(FileMgr.getBufferForFile(ASTFileName, &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();
|
|
|
|
}
|
|
|
|
|
2011-12-01 07:21:26 +08:00
|
|
|
ASTReader::ASTReadResult ASTReader::ReadSubmoduleBlock(ModuleFile &F) {
|
2011-12-01 01:33:56 +08:00
|
|
|
// Enter the submodule block.
|
|
|
|
if (F.Stream.EnterSubBlock(SUBMODULE_BLOCK_ID)) {
|
|
|
|
Error("malformed submodule block record in AST file");
|
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
|
|
|
|
ModuleMap &ModMap = PP.getHeaderSearchInfo().getModuleMap();
|
2011-12-01 08:59:36 +08:00
|
|
|
bool First = true;
|
2011-12-01 07:21:26 +08:00
|
|
|
Module *CurrentModule = 0;
|
2011-12-01 01:33:56 +08:00
|
|
|
RecordData Record;
|
|
|
|
while (true) {
|
|
|
|
unsigned Code = F.Stream.ReadCode();
|
|
|
|
if (Code == llvm::bitc::END_BLOCK) {
|
|
|
|
if (F.Stream.ReadBlockEnd()) {
|
|
|
|
Error("error at end of submodule block in AST file");
|
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
return Success;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Code == llvm::bitc::ENTER_SUBBLOCK) {
|
|
|
|
// No known subblocks, always skip them.
|
|
|
|
F.Stream.ReadSubBlockID();
|
|
|
|
if (F.Stream.SkipBlock()) {
|
|
|
|
Error("malformed block record in AST file");
|
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Code == llvm::bitc::DEFINE_ABBREV) {
|
|
|
|
F.Stream.ReadAbbrevRecord();
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Read a record.
|
|
|
|
const char *BlobStart;
|
|
|
|
unsigned BlobLen;
|
|
|
|
Record.clear();
|
|
|
|
switch (F.Stream.ReadRecord(Code, Record, &BlobStart, &BlobLen)) {
|
|
|
|
default: // Default behavior: ignore.
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SUBMODULE_DEFINITION: {
|
2011-12-01 08:59:36 +08:00
|
|
|
if (First) {
|
|
|
|
Error("missing submodule metadata record at beginning of block");
|
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
|
2011-12-06 09:10:29 +08:00
|
|
|
if (Record.size() < 7) {
|
2011-12-06 06:27:44 +08:00
|
|
|
Error("malformed module definition");
|
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
|
2011-12-01 01:33:56 +08:00
|
|
|
StringRef Name(BlobStart, BlobLen);
|
2011-12-06 09:10:29 +08:00
|
|
|
SubmoduleID GlobalID = getGlobalSubmoduleID(F, Record[0]);
|
|
|
|
SubmoduleID Parent = getGlobalSubmoduleID(F, Record[1]);
|
|
|
|
bool IsFramework = Record[2];
|
|
|
|
bool IsExplicit = Record[3];
|
|
|
|
bool InferSubmodules = Record[4];
|
|
|
|
bool InferExplicitSubmodules = Record[5];
|
|
|
|
bool InferExportWildcard = Record[6];
|
2011-12-06 06:27:44 +08:00
|
|
|
|
2011-12-01 07:21:26 +08:00
|
|
|
Module *ParentModule = 0;
|
2011-12-01 08:59:36 +08:00
|
|
|
if (Parent)
|
|
|
|
ParentModule = getSubmodule(Parent);
|
2011-12-01 01:33:56 +08:00
|
|
|
|
|
|
|
// Retrieve this (sub)module from the module map, creating it if
|
|
|
|
// necessary.
|
|
|
|
CurrentModule = ModMap.findOrCreateModule(Name, ParentModule,
|
|
|
|
IsFramework,
|
|
|
|
IsExplicit).first;
|
2011-12-06 09:10:29 +08:00
|
|
|
SubmoduleID GlobalIndex = GlobalID - NUM_PREDEF_SUBMODULE_IDS;
|
|
|
|
if (GlobalIndex >= SubmodulesLoaded.size() ||
|
|
|
|
SubmodulesLoaded[GlobalIndex]) {
|
2011-12-01 08:59:36 +08:00
|
|
|
Error("too many submodules");
|
|
|
|
return Failure;
|
|
|
|
}
|
2011-12-03 01:30:13 +08:00
|
|
|
|
2011-12-06 06:27:44 +08:00
|
|
|
CurrentModule->InferSubmodules = InferSubmodules;
|
|
|
|
CurrentModule->InferExplicitSubmodules = InferExplicitSubmodules;
|
|
|
|
CurrentModule->InferExportWildcard = InferExportWildcard;
|
2011-12-03 01:30:13 +08:00
|
|
|
if (DeserializationListener)
|
2011-12-06 09:10:29 +08:00
|
|
|
DeserializationListener->ModuleRead(GlobalID, CurrentModule);
|
2011-12-03 01:30:13 +08:00
|
|
|
|
2011-12-06 09:10:29 +08:00
|
|
|
SubmodulesLoaded[GlobalIndex] = CurrentModule;
|
2011-12-01 01:33:56 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2011-12-09 03:11:24 +08:00
|
|
|
case SUBMODULE_UMBRELLA_HEADER: {
|
2011-12-01 08:59:36 +08:00
|
|
|
if (First) {
|
|
|
|
Error("missing submodule metadata record at beginning of block");
|
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
|
2011-12-01 01:33:56 +08:00
|
|
|
if (!CurrentModule)
|
|
|
|
break;
|
|
|
|
|
|
|
|
StringRef FileName(BlobStart, BlobLen);
|
|
|
|
if (const FileEntry *Umbrella = PP.getFileManager().getFile(FileName)) {
|
2011-12-09 01:39:04 +08:00
|
|
|
if (!CurrentModule->getUmbrellaHeader())
|
2011-12-06 09:10:29 +08:00
|
|
|
ModMap.setUmbrellaHeader(CurrentModule, Umbrella);
|
2011-12-09 01:39:04 +08:00
|
|
|
else if (CurrentModule->getUmbrellaHeader() != Umbrella) {
|
2011-12-01 01:33:56 +08:00
|
|
|
Error("mismatched umbrella headers in submodule");
|
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case SUBMODULE_HEADER: {
|
2011-12-01 08:59:36 +08:00
|
|
|
if (First) {
|
|
|
|
Error("missing submodule metadata record at beginning of block");
|
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
|
2011-12-01 01:33:56 +08:00
|
|
|
if (!CurrentModule)
|
|
|
|
break;
|
|
|
|
|
|
|
|
// FIXME: Be more lazy about this!
|
|
|
|
StringRef FileName(BlobStart, BlobLen);
|
|
|
|
if (const FileEntry *File = PP.getFileManager().getFile(FileName)) {
|
|
|
|
if (std::find(CurrentModule->Headers.begin(),
|
|
|
|
CurrentModule->Headers.end(),
|
|
|
|
File) == CurrentModule->Headers.end())
|
2011-12-06 09:10:29 +08:00
|
|
|
ModMap.addHeader(CurrentModule, File);
|
2011-12-01 01:33:56 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2011-12-01 08:59:36 +08:00
|
|
|
|
2011-12-09 03:11:24 +08:00
|
|
|
case SUBMODULE_UMBRELLA_DIR: {
|
|
|
|
if (First) {
|
|
|
|
Error("missing submodule metadata record at beginning of block");
|
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!CurrentModule)
|
|
|
|
break;
|
|
|
|
|
|
|
|
StringRef DirName(BlobStart, BlobLen);
|
|
|
|
if (const DirectoryEntry *Umbrella
|
|
|
|
= PP.getFileManager().getDirectory(DirName)) {
|
|
|
|
if (!CurrentModule->getUmbrellaDir())
|
|
|
|
ModMap.setUmbrellaDir(CurrentModule, Umbrella);
|
|
|
|
else if (CurrentModule->getUmbrellaDir() != Umbrella) {
|
|
|
|
Error("mismatched umbrella directories in submodule");
|
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2011-12-01 08:59:36 +08:00
|
|
|
case SUBMODULE_METADATA: {
|
|
|
|
if (!First) {
|
|
|
|
Error("submodule metadata record not at beginning of block");
|
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
First = false;
|
|
|
|
|
|
|
|
F.BaseSubmoduleID = getTotalNumSubmodules();
|
|
|
|
F.LocalNumSubmodules = Record[0];
|
|
|
|
unsigned LocalBaseSubmoduleID = Record[1];
|
|
|
|
if (F.LocalNumSubmodules > 0) {
|
|
|
|
// Introduce the global -> local mapping for submodules within this
|
|
|
|
// module.
|
|
|
|
GlobalSubmoduleMap.insert(std::make_pair(getTotalNumSubmodules()+1,&F));
|
|
|
|
|
|
|
|
// Introduce the local -> global mapping for submodules within this
|
|
|
|
// module.
|
2011-12-20 00:14:14 +08:00
|
|
|
F.SubmoduleRemap.insertOrReplace(
|
2011-12-01 08:59:36 +08:00
|
|
|
std::make_pair(LocalBaseSubmoduleID,
|
|
|
|
F.BaseSubmoduleID - LocalBaseSubmoduleID));
|
|
|
|
|
|
|
|
SubmodulesLoaded.resize(SubmodulesLoaded.size() + F.LocalNumSubmodules);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2011-12-03 02:58:38 +08:00
|
|
|
|
2011-12-06 00:33:54 +08:00
|
|
|
case SUBMODULE_IMPORTS: {
|
|
|
|
if (First) {
|
|
|
|
Error("missing submodule metadata record at beginning of block");
|
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!CurrentModule)
|
|
|
|
break;
|
|
|
|
|
|
|
|
for (unsigned Idx = 0; Idx != Record.size(); ++Idx) {
|
|
|
|
UnresolvedModuleImportExport Unresolved;
|
|
|
|
Unresolved.File = &F;
|
|
|
|
Unresolved.Mod = CurrentModule;
|
|
|
|
Unresolved.ID = Record[Idx];
|
|
|
|
Unresolved.IsImport = true;
|
|
|
|
Unresolved.IsWildcard = false;
|
|
|
|
UnresolvedModuleImportExports.push_back(Unresolved);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2011-12-03 02:58:38 +08:00
|
|
|
case SUBMODULE_EXPORTS: {
|
|
|
|
if (First) {
|
|
|
|
Error("missing submodule metadata record at beginning of block");
|
|
|
|
return Failure;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!CurrentModule)
|
|
|
|
break;
|
|
|
|
|
|
|
|
for (unsigned Idx = 0; Idx + 1 < Record.size(); Idx += 2) {
|
2011-12-06 00:33:54 +08:00
|
|
|
UnresolvedModuleImportExport Unresolved;
|
2011-12-03 02:58:38 +08:00
|
|
|
Unresolved.File = &F;
|
2011-12-06 00:33:54 +08:00
|
|
|
Unresolved.Mod = CurrentModule;
|
|
|
|
Unresolved.ID = Record[Idx];
|
|
|
|
Unresolved.IsImport = false;
|
|
|
|
Unresolved.IsWildcard = Record[Idx + 1];
|
|
|
|
UnresolvedModuleImportExports.push_back(Unresolved);
|
2011-12-03 02:58:38 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Once we've loaded the set of exports, there's no reason to keep
|
|
|
|
// the parsed, unresolved exports around.
|
|
|
|
CurrentModule->UnresolvedExports.clear();
|
|
|
|
break;
|
|
|
|
}
|
2011-12-01 01:33:56 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Success;
|
|
|
|
}
|
|
|
|
|
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(
|
2011-07-23 18:55:15 +08:00
|
|
|
const SmallVectorImpl<uint64_t> &Record) {
|
2009-06-19 08:03:23 +08:00
|
|
|
if (Listener) {
|
|
|
|
LangOptions LangOpts;
|
|
|
|
unsigned Idx = 0;
|
2011-09-14 02:26:39 +08:00
|
|
|
#define LANGOPT(Name, Bits, Default, Description) \
|
|
|
|
LangOpts.Name = Record[Idx++];
|
|
|
|
#define ENUM_LANGOPT(Name, Type, Bits, Default, Description) \
|
|
|
|
LangOpts.set##Name(static_cast<LangOptions::Type>(Record[Idx++]));
|
|
|
|
#include "clang/Basic/LangOptions.def"
|
|
|
|
|
2011-11-16 03:35:01 +08:00
|
|
|
unsigned Length = Record[Idx++];
|
|
|
|
LangOpts.CurrentModule.assign(Record.begin() + Idx,
|
|
|
|
Record.begin() + Idx + Length);
|
|
|
|
Idx += Length;
|
2009-06-19 08:03:23 +08:00
|
|
|
return Listener->ReadLanguageOptions(LangOpts);
|
2009-04-11 04:39:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2011-12-01 07:21:26 +08:00
|
|
|
std::pair<ModuleFile *, unsigned>
|
2011-10-25 08:29:50 +08:00
|
|
|
ASTReader::getModulePreprocessedEntity(unsigned GlobalIndex) {
|
2011-09-16 02:02:56 +08:00
|
|
|
GlobalPreprocessedEntityMapType::iterator
|
2011-10-25 08:29:50 +08:00
|
|
|
I = GlobalPreprocessedEntityMap.find(GlobalIndex);
|
2011-09-16 02:02:56 +08:00
|
|
|
assert(I != GlobalPreprocessedEntityMap.end() &&
|
|
|
|
"Corrupted global preprocessed entity map");
|
2011-12-01 07:21:26 +08:00
|
|
|
ModuleFile *M = I->second;
|
2011-10-25 08:29:50 +08:00
|
|
|
unsigned LocalIndex = GlobalIndex - M->BasePreprocessedEntityID;
|
|
|
|
return std::make_pair(M, LocalIndex);
|
|
|
|
}
|
|
|
|
|
|
|
|
PreprocessedEntity *ASTReader::ReadPreprocessedEntity(unsigned Index) {
|
|
|
|
PreprocessedEntityID PPID = Index+1;
|
2011-12-01 07:21:26 +08:00
|
|
|
std::pair<ModuleFile *, unsigned> PPInfo = getModulePreprocessedEntity(Index);
|
|
|
|
ModuleFile &M = *PPInfo.first;
|
2011-10-25 08:29:50 +08:00
|
|
|
unsigned LocalIndex = PPInfo.second;
|
2011-09-21 07:27:41 +08:00
|
|
|
const PPEntityOffset &PPOffs = M.PreprocessedEntityOffsets[LocalIndex];
|
2011-02-09 05:58:10 +08:00
|
|
|
|
2011-09-16 02:02:56 +08:00
|
|
|
SavedStreamPosition SavedPosition(M.PreprocessorDetailCursor);
|
2011-09-21 07:27:41 +08:00
|
|
|
M.PreprocessorDetailCursor.JumpToBit(PPOffs.BitOffset);
|
2011-09-21 07:27:38 +08:00
|
|
|
|
|
|
|
unsigned Code = M.PreprocessorDetailCursor.ReadCode();
|
|
|
|
switch (Code) {
|
|
|
|
case llvm::bitc::END_BLOCK:
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
case llvm::bitc::ENTER_SUBBLOCK:
|
|
|
|
Error("unexpected subblock record in preprocessor detail block");
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
case llvm::bitc::DEFINE_ABBREV:
|
|
|
|
Error("unexpected abbrevation record in preprocessor detail block");
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!PP.getPreprocessingRecord()) {
|
|
|
|
Error("no preprocessing record");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Read the record.
|
2011-09-21 07:27:41 +08:00
|
|
|
SourceRange Range(ReadSourceLocation(M, PPOffs.Begin),
|
|
|
|
ReadSourceLocation(M, PPOffs.End));
|
2011-09-21 07:27:38 +08:00
|
|
|
PreprocessingRecord &PPRec = *PP.getPreprocessingRecord();
|
|
|
|
const char *BlobStart = 0;
|
|
|
|
unsigned BlobLen = 0;
|
|
|
|
RecordData Record;
|
|
|
|
PreprocessorDetailRecordTypes RecType =
|
|
|
|
(PreprocessorDetailRecordTypes)M.PreprocessorDetailCursor.ReadRecord(
|
|
|
|
Code, Record, BlobStart, BlobLen);
|
|
|
|
switch (RecType) {
|
|
|
|
case PPD_MACRO_EXPANSION: {
|
2011-09-21 07:27:41 +08:00
|
|
|
bool isBuiltin = Record[0];
|
2011-09-21 07:27:38 +08:00
|
|
|
IdentifierInfo *Name = 0;
|
|
|
|
MacroDefinition *Def = 0;
|
|
|
|
if (isBuiltin)
|
2011-09-21 07:27:41 +08:00
|
|
|
Name = getLocalIdentifier(M, Record[1]);
|
2011-09-21 07:27:38 +08:00
|
|
|
else {
|
|
|
|
PreprocessedEntityID
|
2011-09-21 07:27:41 +08:00
|
|
|
GlobalID = getGlobalPreprocessedEntityID(M, Record[1]);
|
2011-09-21 07:27:38 +08:00
|
|
|
Def =cast<MacroDefinition>(PPRec.getLoadedPreprocessedEntity(GlobalID-1));
|
|
|
|
}
|
|
|
|
|
|
|
|
MacroExpansion *ME;
|
|
|
|
if (isBuiltin)
|
|
|
|
ME = new (PPRec) MacroExpansion(Name, Range);
|
|
|
|
else
|
|
|
|
ME = new (PPRec) MacroExpansion(Def, Range);
|
|
|
|
|
|
|
|
return ME;
|
|
|
|
}
|
|
|
|
|
|
|
|
case PPD_MACRO_DEFINITION: {
|
|
|
|
// Decode the identifier info and then check again; if the macro is
|
|
|
|
// still defined and associated with the identifier,
|
2011-09-21 07:27:41 +08:00
|
|
|
IdentifierInfo *II = getLocalIdentifier(M, Record[0]);
|
2011-09-21 07:27:38 +08:00
|
|
|
MacroDefinition *MD
|
2011-09-21 07:27:41 +08:00
|
|
|
= new (PPRec) MacroDefinition(II, Range);
|
2011-09-21 07:27:38 +08:00
|
|
|
|
|
|
|
if (DeserializationListener)
|
|
|
|
DeserializationListener->MacroDefinitionRead(PPID, MD);
|
|
|
|
|
|
|
|
return MD;
|
|
|
|
}
|
|
|
|
|
|
|
|
case PPD_INCLUSION_DIRECTIVE: {
|
2011-09-21 07:27:41 +08:00
|
|
|
const char *FullFileNameStart = BlobStart + Record[0];
|
2011-09-21 07:27:38 +08:00
|
|
|
const FileEntry *File
|
|
|
|
= PP.getFileManager().getFile(StringRef(FullFileNameStart,
|
2011-09-21 07:27:41 +08:00
|
|
|
BlobLen - Record[0]));
|
2011-09-21 07:27:38 +08:00
|
|
|
|
|
|
|
// FIXME: Stable encoding
|
|
|
|
InclusionDirective::InclusionKind Kind
|
2011-09-21 07:27:41 +08:00
|
|
|
= static_cast<InclusionDirective::InclusionKind>(Record[2]);
|
2011-09-21 07:27:38 +08:00
|
|
|
InclusionDirective *ID
|
|
|
|
= new (PPRec) InclusionDirective(PPRec, Kind,
|
2011-09-21 07:27:41 +08:00
|
|
|
StringRef(BlobStart, Record[0]),
|
|
|
|
Record[1],
|
2011-09-21 07:27:38 +08:00
|
|
|
File,
|
2011-09-21 07:27:41 +08:00
|
|
|
Range);
|
2011-09-21 07:27:38 +08:00
|
|
|
return ID;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Error("invalid offset in preprocessor detail block");
|
|
|
|
return 0;
|
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
|
|
|
}
|
|
|
|
|
2011-09-20 04:40:25 +08:00
|
|
|
/// \brief \arg SLocMapI points at a chunk of a module that contains no
|
|
|
|
/// preprocessed entities or the entities it contains are not the ones we are
|
|
|
|
/// looking for. Find the next module that contains entities and return the ID
|
|
|
|
/// of the first entry.
|
|
|
|
PreprocessedEntityID ASTReader::findNextPreprocessedEntity(
|
|
|
|
GlobalSLocOffsetMapType::const_iterator SLocMapI) const {
|
|
|
|
++SLocMapI;
|
|
|
|
for (GlobalSLocOffsetMapType::const_iterator
|
|
|
|
EndI = GlobalSLocOffsetMap.end(); SLocMapI != EndI; ++SLocMapI) {
|
2011-12-01 07:21:26 +08:00
|
|
|
ModuleFile &M = *SLocMapI->second;
|
2011-09-20 04:40:25 +08:00
|
|
|
if (M.NumPreprocessedEntities)
|
|
|
|
return getGlobalPreprocessedEntityID(M, M.BasePreprocessedEntityID);
|
|
|
|
}
|
|
|
|
|
|
|
|
return getTotalNumPreprocessedEntities();
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
template <unsigned PPEntityOffset::*PPLoc>
|
|
|
|
struct PPEntityComp {
|
|
|
|
const ASTReader &Reader;
|
2011-12-01 07:21:26 +08:00
|
|
|
ModuleFile &M;
|
2011-09-20 04:40:25 +08:00
|
|
|
|
2011-12-01 07:21:26 +08:00
|
|
|
PPEntityComp(const ASTReader &Reader, ModuleFile &M) : Reader(Reader), M(M) { }
|
2011-09-20 04:40:25 +08:00
|
|
|
|
2011-09-21 14:42:26 +08:00
|
|
|
bool operator()(const PPEntityOffset &L, const PPEntityOffset &R) const {
|
|
|
|
SourceLocation LHS = getLoc(L);
|
|
|
|
SourceLocation RHS = getLoc(R);
|
|
|
|
return Reader.getSourceManager().isBeforeInTranslationUnit(LHS, RHS);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool operator()(const PPEntityOffset &L, SourceLocation RHS) const {
|
2011-09-20 04:40:25 +08:00
|
|
|
SourceLocation LHS = getLoc(L);
|
|
|
|
return Reader.getSourceManager().isBeforeInTranslationUnit(LHS, RHS);
|
|
|
|
}
|
|
|
|
|
2011-09-21 14:42:26 +08:00
|
|
|
bool operator()(SourceLocation LHS, const PPEntityOffset &R) const {
|
2011-09-20 04:40:25 +08:00
|
|
|
SourceLocation RHS = getLoc(R);
|
|
|
|
return Reader.getSourceManager().isBeforeInTranslationUnit(LHS, RHS);
|
|
|
|
}
|
|
|
|
|
|
|
|
SourceLocation getLoc(const PPEntityOffset &PPE) const {
|
|
|
|
return Reader.ReadSourceLocation(M, PPE.*PPLoc);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief Returns the first preprocessed entity ID that ends after \arg BLoc.
|
|
|
|
PreprocessedEntityID
|
|
|
|
ASTReader::findBeginPreprocessedEntity(SourceLocation BLoc) const {
|
|
|
|
if (SourceMgr.isLocalSourceLocation(BLoc))
|
|
|
|
return getTotalNumPreprocessedEntities();
|
|
|
|
|
|
|
|
GlobalSLocOffsetMapType::const_iterator
|
|
|
|
SLocMapI = GlobalSLocOffsetMap.find(SourceManager::MaxLoadedOffset -
|
|
|
|
BLoc.getOffset());
|
|
|
|
assert(SLocMapI != GlobalSLocOffsetMap.end() &&
|
|
|
|
"Corrupted global sloc offset map");
|
|
|
|
|
|
|
|
if (SLocMapI->second->NumPreprocessedEntities == 0)
|
|
|
|
return findNextPreprocessedEntity(SLocMapI);
|
|
|
|
|
2011-12-01 07:21:26 +08:00
|
|
|
ModuleFile &M = *SLocMapI->second;
|
2011-09-20 04:40:25 +08:00
|
|
|
typedef const PPEntityOffset *pp_iterator;
|
|
|
|
pp_iterator pp_begin = M.PreprocessedEntityOffsets;
|
|
|
|
pp_iterator pp_end = pp_begin + M.NumPreprocessedEntities;
|
2011-09-23 05:17:02 +08:00
|
|
|
|
|
|
|
size_t Count = M.NumPreprocessedEntities;
|
|
|
|
size_t Half;
|
|
|
|
pp_iterator First = pp_begin;
|
|
|
|
pp_iterator PPI;
|
|
|
|
|
|
|
|
// Do a binary search manually instead of using std::lower_bound because
|
|
|
|
// The end locations of entities may be unordered (when a macro expansion
|
|
|
|
// is inside another macro argument), but for this case it is not important
|
|
|
|
// whether we get the first macro expansion or its containing macro.
|
|
|
|
while (Count > 0) {
|
|
|
|
Half = Count/2;
|
|
|
|
PPI = First;
|
|
|
|
std::advance(PPI, Half);
|
|
|
|
if (SourceMgr.isBeforeInTranslationUnit(ReadSourceLocation(M, PPI->End),
|
|
|
|
BLoc)){
|
|
|
|
First = PPI;
|
|
|
|
++First;
|
|
|
|
Count = Count - Half - 1;
|
|
|
|
} else
|
|
|
|
Count = Half;
|
|
|
|
}
|
2011-09-20 04:40:25 +08:00
|
|
|
|
|
|
|
if (PPI == pp_end)
|
|
|
|
return findNextPreprocessedEntity(SLocMapI);
|
|
|
|
|
|
|
|
return getGlobalPreprocessedEntityID(M,
|
|
|
|
M.BasePreprocessedEntityID + (PPI - pp_begin));
|
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief Returns the first preprocessed entity ID that begins after \arg ELoc.
|
|
|
|
PreprocessedEntityID
|
|
|
|
ASTReader::findEndPreprocessedEntity(SourceLocation ELoc) const {
|
|
|
|
if (SourceMgr.isLocalSourceLocation(ELoc))
|
|
|
|
return getTotalNumPreprocessedEntities();
|
|
|
|
|
|
|
|
GlobalSLocOffsetMapType::const_iterator
|
|
|
|
SLocMapI = GlobalSLocOffsetMap.find(SourceManager::MaxLoadedOffset -
|
|
|
|
ELoc.getOffset());
|
|
|
|
assert(SLocMapI != GlobalSLocOffsetMap.end() &&
|
|
|
|
"Corrupted global sloc offset map");
|
|
|
|
|
|
|
|
if (SLocMapI->second->NumPreprocessedEntities == 0)
|
|
|
|
return findNextPreprocessedEntity(SLocMapI);
|
|
|
|
|
2011-12-01 07:21:26 +08:00
|
|
|
ModuleFile &M = *SLocMapI->second;
|
2011-09-20 04:40:25 +08:00
|
|
|
typedef const PPEntityOffset *pp_iterator;
|
|
|
|
pp_iterator pp_begin = M.PreprocessedEntityOffsets;
|
|
|
|
pp_iterator pp_end = pp_begin + M.NumPreprocessedEntities;
|
|
|
|
pp_iterator PPI =
|
|
|
|
std::upper_bound(pp_begin, pp_end, ELoc,
|
|
|
|
PPEntityComp<&PPEntityOffset::Begin>(*this, M));
|
|
|
|
|
|
|
|
if (PPI == pp_end)
|
|
|
|
return findNextPreprocessedEntity(SLocMapI);
|
|
|
|
|
|
|
|
return getGlobalPreprocessedEntityID(M,
|
|
|
|
M.BasePreprocessedEntityID + (PPI - pp_begin));
|
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief Returns a pair of [Begin, End) indices of preallocated
|
|
|
|
/// preprocessed entities that \arg Range encompasses.
|
|
|
|
std::pair<unsigned, unsigned>
|
|
|
|
ASTReader::findPreprocessedEntitiesInRange(SourceRange Range) {
|
|
|
|
if (Range.isInvalid())
|
|
|
|
return std::make_pair(0,0);
|
|
|
|
assert(!SourceMgr.isBeforeInTranslationUnit(Range.getEnd(),Range.getBegin()));
|
|
|
|
|
|
|
|
PreprocessedEntityID BeginID = findBeginPreprocessedEntity(Range.getBegin());
|
|
|
|
PreprocessedEntityID EndID = findEndPreprocessedEntity(Range.getEnd());
|
|
|
|
return std::make_pair(BeginID, EndID);
|
|
|
|
}
|
|
|
|
|
2011-10-25 08:29:50 +08:00
|
|
|
/// \brief Optionally returns true or false if the preallocated preprocessed
|
|
|
|
/// entity with index \arg Index came from file \arg FID.
|
|
|
|
llvm::Optional<bool> ASTReader::isPreprocessedEntityInFileID(unsigned Index,
|
|
|
|
FileID FID) {
|
|
|
|
if (FID.isInvalid())
|
|
|
|
return false;
|
|
|
|
|
2011-12-01 07:21:26 +08:00
|
|
|
std::pair<ModuleFile *, unsigned> PPInfo = getModulePreprocessedEntity(Index);
|
|
|
|
ModuleFile &M = *PPInfo.first;
|
2011-10-25 08:29:50 +08:00
|
|
|
unsigned LocalIndex = PPInfo.second;
|
|
|
|
const PPEntityOffset &PPOffs = M.PreprocessedEntityOffsets[LocalIndex];
|
|
|
|
|
|
|
|
SourceLocation Loc = ReadSourceLocation(M, PPOffs.Begin);
|
|
|
|
if (Loc.isInvalid())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (SourceMgr.isInFileID(SourceMgr.getFileLoc(Loc), FID))
|
|
|
|
return true;
|
|
|
|
else
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2011-08-26 02:14:34 +08:00
|
|
|
namespace {
|
|
|
|
/// \brief Visitor used to search for information about a header file.
|
|
|
|
class HeaderFileInfoVisitor {
|
|
|
|
ASTReader &Reader;
|
|
|
|
const FileEntry *FE;
|
2011-07-28 12:50:02 +08:00
|
|
|
|
2011-08-26 02:14:34 +08:00
|
|
|
llvm::Optional<HeaderFileInfo> HFI;
|
Implement two related optimizations that make de-serialization of
AST/PCH files more lazy:
- Don't preload all of the file source-location entries when reading
the AST file. Instead, load them lazily, when needed.
- Only look up header-search information (whether a header was already
#import'd, how many times it's been included, etc.) when it's needed
by the preprocessor, rather than pre-populating it.
Previously, we would pre-load all of the file source-location entries,
which also populated the header-search information structure. This was
a relatively minor performance issue, since we would end up stat()'ing
all of the headers stored within a AST/PCH file when the AST/PCH file
was loaded. In the normal PCH use case, the stat()s were cached, so
the cost--of preloading ~860 source-location entries in the Cocoa.h
case---was relatively low.
However, the recent optimization that replaced stat+open with
open+fstat turned this into a major problem, since the preloading of
source-location entries would now end up opening those files. Worse,
those files wouldn't be closed until the file manager was destroyed,
so just opening a Cocoa.h PCH file would hold on to ~860 file
descriptors, and it was easy to blow through the process's limit on
the number of open file descriptors.
By eliminating the preloading of these files, we neither open nor stat
the headers stored in the PCH/AST file until they're actually needed
for something. Concretely, we went from
*** HeaderSearch Stats:
835 files tracked.
364 #import/#pragma once files.
823 included exactly once.
6 max times a file is included.
3 #include/#include_next/#import.
0 #includes skipped due to the multi-include optimization.
1 framework lookups.
0 subframework lookups.
*** Source Manager Stats:
835 files mapped, 3 mem buffers mapped.
37460 SLocEntry's allocated, 11215575B of Sloc address space used.
62 bytes of files mapped, 0 files with line #'s computed.
with a trivial program that uses a chained PCH including a Cocoa PCH
to
*** HeaderSearch Stats:
4 files tracked.
1 #import/#pragma once files.
3 included exactly once.
2 max times a file is included.
3 #include/#include_next/#import.
0 #includes skipped due to the multi-include optimization.
1 framework lookups.
0 subframework lookups.
*** Source Manager Stats:
3 files mapped, 3 mem buffers mapped.
37460 SLocEntry's allocated, 11215575B of Sloc address space used.
62 bytes of files mapped, 0 files with line #'s computed.
for the same program.
llvm-svn: 125286
2011-02-11 01:09:37 +08:00
|
|
|
|
2011-08-26 02:14:34 +08:00
|
|
|
public:
|
|
|
|
HeaderFileInfoVisitor(ASTReader &Reader, const FileEntry *FE)
|
|
|
|
: Reader(Reader), FE(FE) { }
|
|
|
|
|
2011-12-01 07:21:26 +08:00
|
|
|
static bool visit(ModuleFile &M, void *UserData) {
|
2011-08-26 02:14:34 +08:00
|
|
|
HeaderFileInfoVisitor *This
|
|
|
|
= static_cast<HeaderFileInfoVisitor *>(UserData);
|
|
|
|
|
|
|
|
HeaderFileInfoTrait Trait(This->Reader, M,
|
|
|
|
&This->Reader.getPreprocessor().getHeaderSearchInfo(),
|
|
|
|
M.HeaderFileFrameworkStrings,
|
|
|
|
This->FE->getName());
|
|
|
|
|
|
|
|
HeaderFileInfoLookupTable *Table
|
|
|
|
= static_cast<HeaderFileInfoLookupTable *>(M.HeaderFileInfoTable);
|
|
|
|
if (!Table)
|
|
|
|
return false;
|
Implement two related optimizations that make de-serialization of
AST/PCH files more lazy:
- Don't preload all of the file source-location entries when reading
the AST file. Instead, load them lazily, when needed.
- Only look up header-search information (whether a header was already
#import'd, how many times it's been included, etc.) when it's needed
by the preprocessor, rather than pre-populating it.
Previously, we would pre-load all of the file source-location entries,
which also populated the header-search information structure. This was
a relatively minor performance issue, since we would end up stat()'ing
all of the headers stored within a AST/PCH file when the AST/PCH file
was loaded. In the normal PCH use case, the stat()s were cached, so
the cost--of preloading ~860 source-location entries in the Cocoa.h
case---was relatively low.
However, the recent optimization that replaced stat+open with
open+fstat turned this into a major problem, since the preloading of
source-location entries would now end up opening those files. Worse,
those files wouldn't be closed until the file manager was destroyed,
so just opening a Cocoa.h PCH file would hold on to ~860 file
descriptors, and it was easy to blow through the process's limit on
the number of open file descriptors.
By eliminating the preloading of these files, we neither open nor stat
the headers stored in the PCH/AST file until they're actually needed
for something. Concretely, we went from
*** HeaderSearch Stats:
835 files tracked.
364 #import/#pragma once files.
823 included exactly once.
6 max times a file is included.
3 #include/#include_next/#import.
0 #includes skipped due to the multi-include optimization.
1 framework lookups.
0 subframework lookups.
*** Source Manager Stats:
835 files mapped, 3 mem buffers mapped.
37460 SLocEntry's allocated, 11215575B of Sloc address space used.
62 bytes of files mapped, 0 files with line #'s computed.
with a trivial program that uses a chained PCH including a Cocoa PCH
to
*** HeaderSearch Stats:
4 files tracked.
1 #import/#pragma once files.
3 included exactly once.
2 max times a file is included.
3 #include/#include_next/#import.
0 #includes skipped due to the multi-include optimization.
1 framework lookups.
0 subframework lookups.
*** Source Manager Stats:
3 files mapped, 3 mem buffers mapped.
37460 SLocEntry's allocated, 11215575B of Sloc address space used.
62 bytes of files mapped, 0 files with line #'s computed.
for the same program.
llvm-svn: 125286
2011-02-11 01:09:37 +08:00
|
|
|
|
2011-08-26 02:14:34 +08:00
|
|
|
// Look in the on-disk hash table for an entry for this file name.
|
|
|
|
HeaderFileInfoLookupTable::iterator Pos = Table->find(This->FE->getName(),
|
|
|
|
&Trait);
|
|
|
|
if (Pos == Table->end())
|
|
|
|
return false;
|
Implement two related optimizations that make de-serialization of
AST/PCH files more lazy:
- Don't preload all of the file source-location entries when reading
the AST file. Instead, load them lazily, when needed.
- Only look up header-search information (whether a header was already
#import'd, how many times it's been included, etc.) when it's needed
by the preprocessor, rather than pre-populating it.
Previously, we would pre-load all of the file source-location entries,
which also populated the header-search information structure. This was
a relatively minor performance issue, since we would end up stat()'ing
all of the headers stored within a AST/PCH file when the AST/PCH file
was loaded. In the normal PCH use case, the stat()s were cached, so
the cost--of preloading ~860 source-location entries in the Cocoa.h
case---was relatively low.
However, the recent optimization that replaced stat+open with
open+fstat turned this into a major problem, since the preloading of
source-location entries would now end up opening those files. Worse,
those files wouldn't be closed until the file manager was destroyed,
so just opening a Cocoa.h PCH file would hold on to ~860 file
descriptors, and it was easy to blow through the process's limit on
the number of open file descriptors.
By eliminating the preloading of these files, we neither open nor stat
the headers stored in the PCH/AST file until they're actually needed
for something. Concretely, we went from
*** HeaderSearch Stats:
835 files tracked.
364 #import/#pragma once files.
823 included exactly once.
6 max times a file is included.
3 #include/#include_next/#import.
0 #includes skipped due to the multi-include optimization.
1 framework lookups.
0 subframework lookups.
*** Source Manager Stats:
835 files mapped, 3 mem buffers mapped.
37460 SLocEntry's allocated, 11215575B of Sloc address space used.
62 bytes of files mapped, 0 files with line #'s computed.
with a trivial program that uses a chained PCH including a Cocoa PCH
to
*** HeaderSearch Stats:
4 files tracked.
1 #import/#pragma once files.
3 included exactly once.
2 max times a file is included.
3 #include/#include_next/#import.
0 #includes skipped due to the multi-include optimization.
1 framework lookups.
0 subframework lookups.
*** Source Manager Stats:
3 files mapped, 3 mem buffers mapped.
37460 SLocEntry's allocated, 11215575B of Sloc address space used.
62 bytes of files mapped, 0 files with line #'s computed.
for the same program.
llvm-svn: 125286
2011-02-11 01:09:37 +08:00
|
|
|
|
2011-08-26 02:14:34 +08:00
|
|
|
This->HFI = *Pos;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
llvm::Optional<HeaderFileInfo> getHeaderFileInfo() const { return HFI; }
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
HeaderFileInfo ASTReader::GetHeaderFileInfo(const FileEntry *FE) {
|
|
|
|
HeaderFileInfoVisitor Visitor(*this, FE);
|
|
|
|
ModuleMgr.visit(&HeaderFileInfoVisitor::visit, &Visitor);
|
|
|
|
if (llvm::Optional<HeaderFileInfo> HFI = Visitor.getHeaderFileInfo()) {
|
|
|
|
if (Listener)
|
|
|
|
Listener->ReadHeaderFileInfo(*HFI, FE->getUID());
|
|
|
|
return *HFI;
|
Implement two related optimizations that make de-serialization of
AST/PCH files more lazy:
- Don't preload all of the file source-location entries when reading
the AST file. Instead, load them lazily, when needed.
- Only look up header-search information (whether a header was already
#import'd, how many times it's been included, etc.) when it's needed
by the preprocessor, rather than pre-populating it.
Previously, we would pre-load all of the file source-location entries,
which also populated the header-search information structure. This was
a relatively minor performance issue, since we would end up stat()'ing
all of the headers stored within a AST/PCH file when the AST/PCH file
was loaded. In the normal PCH use case, the stat()s were cached, so
the cost--of preloading ~860 source-location entries in the Cocoa.h
case---was relatively low.
However, the recent optimization that replaced stat+open with
open+fstat turned this into a major problem, since the preloading of
source-location entries would now end up opening those files. Worse,
those files wouldn't be closed until the file manager was destroyed,
so just opening a Cocoa.h PCH file would hold on to ~860 file
descriptors, and it was easy to blow through the process's limit on
the number of open file descriptors.
By eliminating the preloading of these files, we neither open nor stat
the headers stored in the PCH/AST file until they're actually needed
for something. Concretely, we went from
*** HeaderSearch Stats:
835 files tracked.
364 #import/#pragma once files.
823 included exactly once.
6 max times a file is included.
3 #include/#include_next/#import.
0 #includes skipped due to the multi-include optimization.
1 framework lookups.
0 subframework lookups.
*** Source Manager Stats:
835 files mapped, 3 mem buffers mapped.
37460 SLocEntry's allocated, 11215575B of Sloc address space used.
62 bytes of files mapped, 0 files with line #'s computed.
with a trivial program that uses a chained PCH including a Cocoa PCH
to
*** HeaderSearch Stats:
4 files tracked.
1 #import/#pragma once files.
3 included exactly once.
2 max times a file is included.
3 #include/#include_next/#import.
0 #includes skipped due to the multi-include optimization.
1 framework lookups.
0 subframework lookups.
*** Source Manager Stats:
3 files mapped, 3 mem buffers mapped.
37460 SLocEntry's allocated, 11215575B of Sloc address space used.
62 bytes of files mapped, 0 files with line #'s computed.
for the same program.
llvm-svn: 125286
2011-02-11 01:09:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return HeaderFileInfo();
|
|
|
|
}
|
|
|
|
|
2011-09-26 07:23:43 +08:00
|
|
|
void ASTReader::ReadPragmaDiagnosticMappings(DiagnosticsEngine &Diag) {
|
2011-07-26 04:32:21 +08:00
|
|
|
for (ModuleIterator I = ModuleMgr.begin(), E = ModuleMgr.end(); I != E; ++I) {
|
2011-12-01 07:21:26 +08:00
|
|
|
ModuleFile &F = *(*I);
|
Revamp the SourceManager to separate the representation of parsed
source locations from source locations loaded from an AST/PCH file.
Previously, loading an AST/PCH file involved carefully pre-allocating
space at the beginning of the source manager for the source locations
and FileIDs that correspond to the prefix, and then appending the
source locations/FileIDs used for parsing the remaining translation
unit. This design forced us into loading PCH files early, as a prefix,
whic has become a rather significant limitation.
This patch splits the SourceManager space into two parts: for source
location "addresses", the lower values (growing upward) are used to
describe parsed code, while upper values (growing downward) are used
for source locations loaded from AST/PCH files. Similarly, positive
FileIDs are used to describe parsed code while negative FileIDs are
used to file/macro locations loaded from AST/PCH files. As a result,
we can load PCH/AST files even during parsing, making various
improvemnts in the future possible, e.g., teaching #include <foo.h> to
look for and load <foo.h.gch> if it happens to be already available.
This patch was originally written by Sebastian Redl, then brought
forward to the modern age by Jonathan Turner, and finally
polished/finished by me to be committed.
llvm-svn: 135484
2011-07-20 00:10:42 +08:00
|
|
|
unsigned Idx = 0;
|
|
|
|
while (Idx < F.PragmaDiagMappings.size()) {
|
|
|
|
SourceLocation Loc = ReadSourceLocation(F, F.PragmaDiagMappings[Idx++]);
|
2011-11-09 09:24:17 +08:00
|
|
|
Diag.DiagStates.push_back(*Diag.GetCurDiagState());
|
|
|
|
Diag.DiagStatePoints.push_back(
|
|
|
|
DiagnosticsEngine::DiagStatePoint(&Diag.DiagStates.back(),
|
|
|
|
FullSourceLoc(Loc, SourceMgr)));
|
Revamp the SourceManager to separate the representation of parsed
source locations from source locations loaded from an AST/PCH file.
Previously, loading an AST/PCH file involved carefully pre-allocating
space at the beginning of the source manager for the source locations
and FileIDs that correspond to the prefix, and then appending the
source locations/FileIDs used for parsing the remaining translation
unit. This design forced us into loading PCH files early, as a prefix,
whic has become a rather significant limitation.
This patch splits the SourceManager space into two parts: for source
location "addresses", the lower values (growing upward) are used to
describe parsed code, while upper values (growing downward) are used
for source locations loaded from AST/PCH files. Similarly, positive
FileIDs are used to describe parsed code while negative FileIDs are
used to file/macro locations loaded from AST/PCH files. As a result,
we can load PCH/AST files even during parsing, making various
improvemnts in the future possible, e.g., teaching #include <foo.h> to
look for and load <foo.h.gch> if it happens to be already available.
This patch was originally written by Sebastian Redl, then brought
forward to the modern age by Jonathan Turner, and finally
polished/finished by me to be committed.
llvm-svn: 135484
2011-07-20 00:10:42 +08:00
|
|
|
while (1) {
|
|
|
|
assert(Idx < F.PragmaDiagMappings.size() &&
|
|
|
|
"Invalid data, didn't find '-1' marking end of diag/map pairs");
|
|
|
|
if (Idx >= F.PragmaDiagMappings.size()) {
|
|
|
|
break; // Something is messed up but at least avoid infinite loop in
|
|
|
|
// release build.
|
|
|
|
}
|
|
|
|
unsigned DiagID = F.PragmaDiagMappings[Idx++];
|
|
|
|
if (DiagID == (unsigned)-1) {
|
|
|
|
break; // no more diag/map pairs for this location.
|
|
|
|
}
|
|
|
|
diag::Mapping Map = (diag::Mapping)F.PragmaDiagMappings[Idx++];
|
2011-11-09 09:24:17 +08:00
|
|
|
DiagnosticMappingInfo MappingInfo = Diag.makeMappingInfo(Map, Loc);
|
|
|
|
Diag.GetCurDiagState()->setMappingInfo(DiagID, MappingInfo);
|
Revamp the SourceManager to separate the representation of parsed
source locations from source locations loaded from an AST/PCH file.
Previously, loading an AST/PCH file involved carefully pre-allocating
space at the beginning of the source manager for the source locations
and FileIDs that correspond to the prefix, and then appending the
source locations/FileIDs used for parsing the remaining translation
unit. This design forced us into loading PCH files early, as a prefix,
whic has become a rather significant limitation.
This patch splits the SourceManager space into two parts: for source
location "addresses", the lower values (growing upward) are used to
describe parsed code, while upper values (growing downward) are used
for source locations loaded from AST/PCH files. Similarly, positive
FileIDs are used to describe parsed code while negative FileIDs are
used to file/macro locations loaded from AST/PCH files. As a result,
we can load PCH/AST files even during parsing, making various
improvemnts in the future possible, e.g., teaching #include <foo.h> to
look for and load <foo.h.gch> if it happens to be already available.
This patch was originally written by Sebastian Redl, then brought
forward to the modern age by Jonathan Turner, and finally
polished/finished by me to be committed.
llvm-svn: 135484
2011-07-20 00:10:42 +08:00
|
|
|
}
|
2011-01-15 04:54:07 +08:00
|
|
|
}
|
2010-11-06 06:10:18 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
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) {
|
2011-08-03 00:26:37 +08:00
|
|
|
GlobalTypeMapType::iterator I = GlobalTypeMap.find(Index);
|
2011-07-21 05:31:32 +08:00
|
|
|
assert(I != GlobalTypeMap.end() && "Corrupted global type map");
|
2011-12-01 07:21:26 +08:00
|
|
|
ModuleFile *M = I->second;
|
Change the hashing function for DeclContext lookup within an AST file
by eliminating the type ID from constructor, destructor, and
conversion function names. There are several reasons for this change:
- A given type (say, int*) isn't guaranteed to have a single, unique
type ID within a chain of PCH files. Hence, we could end up hashing
based on the wrong type ID, causing name lookup to fail.
- The mapping from types back to type IDs required one DenseMap
entry for every type that was ever deserialized, which was an
unacceptable cost to support just the name lookup of constructors,
destructors, and conversion functions. Plus, this mapping could
never actually work with chained or multiple PCH, based on the first
bullet.
Once we have eliminated the type from the hash function, these
problems go away, as does my horrible "reverse type remap" hack, which
was doomed from the start (see bullet #1 above) and far too
complicated.
However, note that removing the type from the hash function means that
all constructors, destructors, and conversion functions have the same
hash key, so I've updated the caller to double-check that the
declarations found have the appropriate name.
llvm-svn: 136708
2011-08-03 02:32:54 +08:00
|
|
|
return RecordLocation(M, M->TypeOffsets[Index - M->BaseTypeIndex]);
|
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.
|
2011-07-22 08:38:23 +08:00
|
|
|
QualType ASTReader::readTypeRecord(unsigned Index) {
|
2010-07-21 06:37:49 +08:00
|
|
|
RecordLocation Loc = TypeCursorForIndex(Index);
|
2010-10-05 23:59:54 +08:00
|
|
|
llvm::BitstreamCursor &DeclsCursor = Loc.F->DeclsCursor;
|
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
|
|
|
|
2011-07-22 08:38:23 +08:00
|
|
|
unsigned Idx = 0;
|
2010-10-05 23:59:54 +08:00
|
|
|
DeclsCursor.JumpToBit(Loc.Offset);
|
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();
|
|
|
|
}
|
2011-07-22 08:38:23 +08:00
|
|
|
QualType Base = readType(*Loc.F, Record, Idx);
|
|
|
|
Qualifiers Quals = Qualifiers::fromOpaqueValue(Record[Idx++]);
|
2011-09-10 05:34:22 +08:00
|
|
|
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();
|
|
|
|
}
|
2011-07-22 08:38:23 +08:00
|
|
|
QualType ElemType = readType(*Loc.F, Record, Idx);
|
2011-09-10 05:34:22 +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();
|
|
|
|
}
|
2011-07-22 08:38:23 +08:00
|
|
|
QualType PointeeType = readType(*Loc.F, Record, Idx);
|
2011-09-10 05:34:22 +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();
|
|
|
|
}
|
2011-07-22 08:38:23 +08:00
|
|
|
QualType PointeeType = readType(*Loc.F, Record, Idx);
|
2011-09-10 05:34:22 +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: {
|
2011-04-12 18:38:03 +08:00
|
|
|
if (Record.size() != 2) {
|
2010-03-18 08:56:54 +08:00
|
|
|
Error("Incorrect encoding of lvalue reference type");
|
|
|
|
return QualType();
|
|
|
|
}
|
2011-07-22 08:38:23 +08:00
|
|
|
QualType PointeeType = readType(*Loc.F, Record, Idx);
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.getLValueReferenceType(PointeeType, Record[1]);
|
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();
|
|
|
|
}
|
2011-07-22 08:38:23 +08:00
|
|
|
QualType PointeeType = readType(*Loc.F, Record, Idx);
|
2011-09-10 05:34:22 +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();
|
|
|
|
}
|
2011-07-22 08:38:23 +08:00
|
|
|
QualType PointeeType = readType(*Loc.F, Record, Idx);
|
|
|
|
QualType ClassType = readType(*Loc.F, Record, Idx);
|
2010-12-11 01:03:06 +08:00
|
|
|
if (PointeeType.isNull() || ClassType.isNull())
|
|
|
|
return QualType();
|
|
|
|
|
2011-09-10 05:34:22 +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: {
|
2011-07-22 08:38:23 +08:00
|
|
|
QualType ElementType = readType(*Loc.F, Record, Idx);
|
2009-04-14 04:46:52 +08:00
|
|
|
ArrayType::ArraySizeModifier ASM = (ArrayType::ArraySizeModifier)Record[1];
|
|
|
|
unsigned IndexTypeQuals = Record[2];
|
|
|
|
unsigned Idx = 3;
|
|
|
|
llvm::APInt Size = ReadAPInt(Record, Idx);
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.getConstantArrayType(ElementType, Size,
|
2009-07-06 23:59:29 +08:00
|
|
|
ASM, IndexTypeQuals);
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TYPE_INCOMPLETE_ARRAY: {
|
2011-07-22 08:38:23 +08:00
|
|
|
QualType ElementType = readType(*Loc.F, Record, Idx);
|
2009-04-14 04:46:52 +08:00
|
|
|
ArrayType::ArraySizeModifier ASM = (ArrayType::ArraySizeModifier)Record[1];
|
|
|
|
unsigned IndexTypeQuals = Record[2];
|
2011-09-10 05:34:22 +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: {
|
2011-07-22 08:38:23 +08:00
|
|
|
QualType ElementType = readType(*Loc.F, Record, Idx);
|
2009-04-15 05:18:50 +08:00
|
|
|
ArrayType::ArraySizeModifier ASM = (ArrayType::ArraySizeModifier)Record[1];
|
|
|
|
unsigned IndexTypeQuals = Record[2];
|
2010-10-05 23:59:54 +08:00
|
|
|
SourceLocation LBLoc = ReadSourceLocation(*Loc.F, Record[3]);
|
|
|
|
SourceLocation RBLoc = ReadSourceLocation(*Loc.F, Record[4]);
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.getVariableArrayType(ElementType, ReadExpr(*Loc.F),
|
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();
|
|
|
|
}
|
|
|
|
|
2011-07-22 08:38:23 +08:00
|
|
|
QualType ElementType = readType(*Loc.F, Record, Idx);
|
2009-04-14 04:46:52 +08:00
|
|
|
unsigned NumElements = Record[1];
|
2010-11-11 05:56:12 +08:00
|
|
|
unsigned VecKind = Record[2];
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.getVectorType(ElementType, NumElements,
|
2010-11-11 05:56:12 +08:00
|
|
|
(VectorType::VectorKind)VecKind);
|
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();
|
|
|
|
}
|
|
|
|
|
2011-07-22 08:38:23 +08:00
|
|
|
QualType ElementType = readType(*Loc.F, Record, Idx);
|
2009-04-14 04:46:52 +08:00
|
|
|
unsigned NumElements = Record[1];
|
2011-09-10 05:34:22 +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: {
|
2011-06-16 07:02:42 +08:00
|
|
|
if (Record.size() != 6) {
|
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();
|
|
|
|
}
|
2011-07-22 08:38:23 +08:00
|
|
|
QualType ResultType = readType(*Loc.F, Record, Idx);
|
2011-06-16 07:02:42 +08:00
|
|
|
FunctionType::ExtInfo Info(Record[1], Record[2], Record[3],
|
|
|
|
(CallingConv)Record[4], Record[5]);
|
2011-09-10 05:34:22 +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: {
|
2011-07-22 08:38:23 +08:00
|
|
|
QualType ResultType = readType(*Loc.F, Record, Idx);
|
2010-12-14 16:05:40 +08:00
|
|
|
|
|
|
|
FunctionProtoType::ExtProtoInfo EPI;
|
|
|
|
EPI.ExtInfo = FunctionType::ExtInfo(/*noreturn*/ Record[1],
|
2011-04-09 16:18:08 +08:00
|
|
|
/*hasregparm*/ Record[2],
|
|
|
|
/*regparm*/ Record[3],
|
2011-06-16 07:02:42 +08:00
|
|
|
static_cast<CallingConv>(Record[4]),
|
|
|
|
/*produces*/ Record[5]);
|
2010-12-14 16:05:40 +08:00
|
|
|
|
2011-06-16 07:02:42 +08:00
|
|
|
unsigned Idx = 6;
|
2009-04-14 04:46:52 +08:00
|
|
|
unsigned NumParams = Record[Idx++];
|
2011-07-23 18:55:15 +08:00
|
|
|
SmallVector<QualType, 16> ParamTypes;
|
2009-04-14 04:46:52 +08:00
|
|
|
for (unsigned I = 0; I != NumParams; ++I)
|
2011-07-22 08:38:23 +08:00
|
|
|
ParamTypes.push_back(readType(*Loc.F, Record, Idx));
|
2010-12-14 16:05:40 +08:00
|
|
|
|
|
|
|
EPI.Variadic = Record[Idx++];
|
|
|
|
EPI.TypeQuals = Record[Idx++];
|
2011-01-26 13:01:58 +08:00
|
|
|
EPI.RefQualifier = static_cast<RefQualifierKind>(Record[Idx++]);
|
2011-03-12 19:50:43 +08:00
|
|
|
ExceptionSpecificationType EST =
|
|
|
|
static_cast<ExceptionSpecificationType>(Record[Idx++]);
|
|
|
|
EPI.ExceptionSpecType = EST;
|
|
|
|
if (EST == EST_Dynamic) {
|
|
|
|
EPI.NumExceptions = Record[Idx++];
|
2011-07-23 18:55:15 +08:00
|
|
|
SmallVector<QualType, 2> Exceptions;
|
2011-03-12 19:50:43 +08:00
|
|
|
for (unsigned I = 0; I != EPI.NumExceptions; ++I)
|
2011-07-22 08:38:23 +08:00
|
|
|
Exceptions.push_back(readType(*Loc.F, Record, Idx));
|
2011-03-12 19:50:43 +08:00
|
|
|
EPI.Exceptions = Exceptions.data();
|
|
|
|
} else if (EST == EST_ComputedNoexcept) {
|
|
|
|
EPI.NoexceptExpr = ReadExpr(*Loc.F);
|
|
|
|
}
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.getFunctionType(ResultType, ParamTypes.data(), NumParams,
|
2010-12-14 16:05:40 +08:00
|
|
|
EPI);
|
2009-04-14 04:46:52 +08:00
|
|
|
}
|
|
|
|
|
2011-07-22 06:35:25 +08:00
|
|
|
case TYPE_UNRESOLVED_USING: {
|
|
|
|
unsigned Idx = 0;
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.getTypeDeclType(
|
2011-07-22 06:35:25 +08:00
|
|
|
ReadDeclAs<UnresolvedUsingTypenameDecl>(*Loc.F, Record, Idx));
|
|
|
|
}
|
|
|
|
|
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();
|
|
|
|
}
|
2011-07-22 06:35:25 +08:00
|
|
|
unsigned Idx = 0;
|
|
|
|
TypedefNameDecl *Decl = ReadDeclAs<TypedefNameDecl>(*Loc.F, Record, Idx);
|
2011-07-22 08:38:23 +08:00
|
|
|
QualType Canonical = readType(*Loc.F, Record, Idx);
|
2010-10-26 08:51:02 +08:00
|
|
|
if (!Canonical.isNull())
|
2011-09-10 05:34:22 +08:00
|
|
|
Canonical = Context.getCanonicalType(Canonical);
|
|
|
|
return Context.getTypedefType(Decl, Canonical);
|
2010-07-02 19:55:11 +08:00
|
|
|
}
|
2009-04-14 04:46:52 +08:00
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TYPE_TYPEOF_EXPR:
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.getTypeOfExprType(ReadExpr(*Loc.F));
|
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();
|
|
|
|
}
|
2011-07-22 08:38:23 +08:00
|
|
|
QualType UnderlyingType = readType(*Loc.F, Record, Idx);
|
2011-09-10 05:34:22 +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:
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.getDecltypeType(ReadExpr(*Loc.F));
|
2009-06-25 03:06:50 +08:00
|
|
|
|
2011-05-25 06:41:36 +08:00
|
|
|
case TYPE_UNARY_TRANSFORM: {
|
2011-07-22 08:38:23 +08:00
|
|
|
QualType BaseType = readType(*Loc.F, Record, Idx);
|
|
|
|
QualType UnderlyingType = readType(*Loc.F, Record, Idx);
|
2011-05-25 06:41:36 +08:00
|
|
|
UnaryTransformType::UTTKind UKind = (UnaryTransformType::UTTKind)Record[2];
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.getUnaryTransformType(BaseType, UnderlyingType, UKind);
|
2011-05-25 06:41:36 +08:00
|
|
|
}
|
|
|
|
|
2011-02-20 11:19:35 +08:00
|
|
|
case TYPE_AUTO:
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.getAutoType(readType(*Loc.F, Record, Idx));
|
2011-02-20 11:19:35 +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();
|
|
|
|
}
|
2011-07-22 06:35:25 +08:00
|
|
|
unsigned Idx = 0;
|
|
|
|
bool IsDependent = Record[Idx++];
|
|
|
|
QualType T
|
2011-09-10 05:34:22 +08:00
|
|
|
= Context.getRecordType(ReadDeclAs<RecordDecl>(*Loc.F, Record, Idx));
|
2011-01-19 14:33:43 +08:00
|
|
|
const_cast<Type*>(T.getTypePtr())->setDependent(IsDependent);
|
2010-07-08 21:09:53 +08:00
|
|
|
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();
|
|
|
|
}
|
2011-07-22 06:35:25 +08:00
|
|
|
unsigned Idx = 0;
|
|
|
|
bool IsDependent = Record[Idx++];
|
|
|
|
QualType T
|
2011-09-10 05:34:22 +08:00
|
|
|
= Context.getEnumType(ReadDeclAs<EnumDecl>(*Loc.F, Record, Idx));
|
2011-01-19 14:33:43 +08:00
|
|
|
const_cast<Type*>(T.getTypePtr())->setDependent(IsDependent);
|
2010-07-08 21:09:53 +08:00
|
|
|
return T;
|
|
|
|
}
|
2009-04-14 02:14:40 +08:00
|
|
|
|
2011-01-06 09:58:22 +08:00
|
|
|
case TYPE_ATTRIBUTED: {
|
|
|
|
if (Record.size() != 3) {
|
|
|
|
Error("incorrect encoding of attributed type");
|
|
|
|
return QualType();
|
|
|
|
}
|
2011-07-22 08:38:23 +08:00
|
|
|
QualType modifiedType = readType(*Loc.F, Record, Idx);
|
|
|
|
QualType equivalentType = readType(*Loc.F, Record, Idx);
|
2011-01-06 09:58:22 +08:00
|
|
|
AttributedType::Kind kind = static_cast<AttributedType::Kind>(Record[2]);
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.getAttributedType(kind, modifiedType, equivalentType);
|
2011-01-06 09:58:22 +08:00
|
|
|
}
|
|
|
|
|
2010-12-11 00:29:40 +08:00
|
|
|
case TYPE_PAREN: {
|
|
|
|
if (Record.size() != 1) {
|
|
|
|
Error("incorrect encoding of paren type");
|
|
|
|
return QualType();
|
|
|
|
}
|
2011-07-22 08:38:23 +08:00
|
|
|
QualType InnerType = readType(*Loc.F, Record, Idx);
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.getParenType(InnerType);
|
2010-12-11 00:29:40 +08:00
|
|
|
}
|
|
|
|
|
2010-12-20 10:24:11 +08:00
|
|
|
case TYPE_PACK_EXPANSION: {
|
2011-02-01 23:24:58 +08:00
|
|
|
if (Record.size() != 2) {
|
2010-12-20 10:24:11 +08:00
|
|
|
Error("incorrect encoding of pack expansion type");
|
|
|
|
return QualType();
|
|
|
|
}
|
2011-07-22 08:38:23 +08:00
|
|
|
QualType Pattern = readType(*Loc.F, Record, Idx);
|
2010-12-20 10:24:11 +08:00
|
|
|
if (Pattern.isNull())
|
|
|
|
return QualType();
|
2011-01-15 01:04:44 +08:00
|
|
|
llvm::Optional<unsigned> NumExpansions;
|
|
|
|
if (Record[1])
|
|
|
|
NumExpansions = Record[1] - 1;
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.getPackExpansionType(Pattern, NumExpansions);
|
2010-12-20 10:24:11 +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++];
|
2011-07-22 06:35:25 +08:00
|
|
|
NestedNameSpecifier *NNS = ReadNestedNameSpecifier(*Loc.F, Record, Idx);
|
2011-07-22 08:38:23 +08:00
|
|
|
QualType NamedType = readType(*Loc.F, Record, Idx);
|
2011-09-10 05:34:22 +08:00
|
|
|
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;
|
2011-07-22 06:35:25 +08:00
|
|
|
ObjCInterfaceDecl *ItfD
|
|
|
|
= ReadDeclAs<ObjCInterfaceDecl>(*Loc.F, Record, Idx);
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.getObjCInterfaceType(ItfD);
|
2010-05-15 19:32:37 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TYPE_OBJC_OBJECT: {
|
2010-05-15 19:32:37 +08:00
|
|
|
unsigned Idx = 0;
|
2011-07-22 08:38:23 +08:00
|
|
|
QualType Base = readType(*Loc.F, Record, Idx);
|
2009-04-22 14:45:28 +08:00
|
|
|
unsigned NumProtos = Record[Idx++];
|
2011-07-23 18:55:15 +08:00
|
|
|
SmallVector<ObjCProtocolDecl*, 4> Protos;
|
2009-04-22 14:45:28 +08:00
|
|
|
for (unsigned I = 0; I != NumProtos; ++I)
|
2011-07-22 06:35:25 +08:00
|
|
|
Protos.push_back(ReadDeclAs<ObjCProtocolDecl>(*Loc.F, Record, Idx));
|
2011-09-10 05:34:22 +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;
|
2011-07-22 08:38:23 +08:00
|
|
|
QualType Pointee = readType(*Loc.F, Record, Idx);
|
2011-09-10 05:34:22 +08:00
|
|
|
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;
|
2011-07-22 08:38:23 +08:00
|
|
|
QualType Parm = readType(*Loc.F, Record, Idx);
|
|
|
|
QualType Replacement = readType(*Loc.F, Record, Idx);
|
2009-10-18 17:09:24 +08:00
|
|
|
return
|
2011-09-10 05:34:22 +08:00
|
|
|
Context.getSubstTemplateTypeParmType(cast<TemplateTypeParmType>(Parm),
|
2009-10-18 17:09:24 +08:00
|
|
|
Replacement);
|
|
|
|
}
|
2010-03-10 11:28:59 +08:00
|
|
|
|
2011-01-14 10:55:32 +08:00
|
|
|
case TYPE_SUBST_TEMPLATE_TYPE_PARM_PACK: {
|
|
|
|
unsigned Idx = 0;
|
2011-07-22 08:38:23 +08:00
|
|
|
QualType Parm = readType(*Loc.F, Record, Idx);
|
2011-01-14 10:55:32 +08:00
|
|
|
TemplateArgument ArgPack = ReadTemplateArgument(*Loc.F, Record, Idx);
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.getSubstTemplateTypeParmPackType(
|
2011-01-14 10:55:32 +08:00
|
|
|
cast<TemplateTypeParmType>(Parm),
|
|
|
|
ArgPack);
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
case TYPE_INJECTED_CLASS_NAME: {
|
2011-07-22 06:35:25 +08:00
|
|
|
CXXRecordDecl *D = ReadDeclAs<CXXRecordDecl>(*Loc.F, Record, Idx);
|
2011-07-22 08:38:23 +08:00
|
|
|
QualType TST = readType(*Loc.F, Record, Idx); // 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
|
2011-09-10 05:34:22 +08:00
|
|
|
QualType(new (Context, TypeAlignment) InjectedClassNameType(D, TST), 0);
|
2010-03-10 11:28:59 +08:00
|
|
|
}
|
2010-10-21 11:16:25 +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++];
|
2011-07-22 06:35:25 +08:00
|
|
|
TemplateTypeParmDecl *D
|
|
|
|
= ReadDeclAs<TemplateTypeParmDecl>(*Loc.F, Record, Idx);
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.getTemplateTypeParmType(Depth, Index, Pack, D);
|
2010-06-20 03:29:09 +08:00
|
|
|
}
|
2010-10-21 11:16:25 +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++];
|
2011-07-22 06:35:25 +08:00
|
|
|
NestedNameSpecifier *NNS = ReadNestedNameSpecifier(*Loc.F, Record, Idx);
|
2011-07-29 04:55:49 +08:00
|
|
|
const IdentifierInfo *Name = this->GetIdentifierInfo(*Loc.F, Record, Idx);
|
2011-07-22 08:38:23 +08:00
|
|
|
QualType Canon = readType(*Loc.F, Record, Idx);
|
2010-10-26 08:51:02 +08:00
|
|
|
if (!Canon.isNull())
|
2011-09-10 05:34:22 +08:00
|
|
|
Canon = Context.getCanonicalType(Canon);
|
|
|
|
return Context.getDependentNameType(Keyword, NNS, Name, Canon);
|
2010-06-24 16:57:31 +08:00
|
|
|
}
|
2010-10-21 11:16:25 +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++];
|
2011-07-22 06:35:25 +08:00
|
|
|
NestedNameSpecifier *NNS = ReadNestedNameSpecifier(*Loc.F, Record, Idx);
|
2011-07-29 04:55:49 +08:00
|
|
|
const IdentifierInfo *Name = this->GetIdentifierInfo(*Loc.F, Record, Idx);
|
2010-06-26 00:24:58 +08:00
|
|
|
unsigned NumArgs = Record[Idx++];
|
2011-07-23 18:55:15 +08:00
|
|
|
SmallVector<TemplateArgument, 8> Args;
|
2010-06-26 00:24:58 +08:00
|
|
|
Args.reserve(NumArgs);
|
|
|
|
while (NumArgs--)
|
2010-10-05 23:59:54 +08:00
|
|
|
Args.push_back(ReadTemplateArgument(*Loc.F, Record, Idx));
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.getDependentTemplateSpecializationType(Keyword, NNS, Name,
|
2010-06-26 00:24:58 +08:00
|
|
|
Args.size(), Args.data());
|
|
|
|
}
|
2010-10-21 11:16: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
|
2011-07-22 08:38:23 +08:00
|
|
|
QualType ElementType = readType(*Loc.F, Record, Idx);
|
2010-06-30 16:49:25 +08:00
|
|
|
ArrayType::ArraySizeModifier ASM
|
|
|
|
= (ArrayType::ArraySizeModifier)Record[Idx++];
|
|
|
|
unsigned IndexTypeQuals = Record[Idx++];
|
|
|
|
|
|
|
|
// DependentSizedArrayType
|
2010-10-05 23:59:54 +08:00
|
|
|
Expr *NumElts = ReadExpr(*Loc.F);
|
|
|
|
SourceRange Brackets = ReadSourceRange(*Loc.F, Record, Idx);
|
2010-06-30 16:49:25 +08:00
|
|
|
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.getDependentSizedArrayType(ElementType, NumElts, ASM,
|
2010-06-30 16:49:25 +08:00
|
|
|
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++];
|
2011-01-15 14:45:20 +08:00
|
|
|
TemplateName Name = ReadTemplateName(*Loc.F, Record, Idx);
|
2011-07-23 18:55:15 +08:00
|
|
|
SmallVector<TemplateArgument, 8> Args;
|
2010-10-05 23:59:54 +08:00
|
|
|
ReadTemplateArgumentList(Args, *Loc.F, Record, Idx);
|
2011-07-22 08:38:23 +08:00
|
|
|
QualType Underlying = readType(*Loc.F, Record, Idx);
|
2010-07-08 21:09:53 +08:00
|
|
|
QualType T;
|
2011-05-06 05:57:07 +08:00
|
|
|
if (Underlying.isNull())
|
2011-09-10 05:34:22 +08:00
|
|
|
T = Context.getCanonicalTemplateSpecializationType(Name, Args.data(),
|
2010-07-08 21:09:53 +08:00
|
|
|
Args.size());
|
2010-07-02 19:55:11 +08:00
|
|
|
else
|
2011-09-10 05:34:22 +08:00
|
|
|
T = Context.getTemplateSpecializationType(Name, Args.data(),
|
2011-05-06 05:57:07 +08:00
|
|
|
Args.size(), Underlying);
|
2011-01-19 14:33:43 +08:00
|
|
|
const_cast<Type*>(T.getTypePtr())->setDependent(IsDependent);
|
2010-07-08 21:09:53 +08:00
|
|
|
return T;
|
2010-06-20 03:29:09 +08:00
|
|
|
}
|
2011-10-07 07:00:33 +08:00
|
|
|
|
|
|
|
case TYPE_ATOMIC: {
|
|
|
|
if (Record.size() != 1) {
|
|
|
|
Error("Incorrect encoding of atomic type");
|
|
|
|
return QualType();
|
|
|
|
}
|
|
|
|
QualType ValueType = readType(*Loc.F, Record, Idx);
|
|
|
|
return Context.getAtomicType(ValueType);
|
|
|
|
}
|
2009-04-10 06:27:44 +08:00
|
|
|
}
|
|
|
|
// Suppress a GCC warning
|
|
|
|
return QualType();
|
|
|
|
}
|
|
|
|
|
2010-10-05 23:59:54 +08:00
|
|
|
class clang::TypeLocReader : public TypeLocVisitor<TypeLocReader> {
|
2010-08-19 07:56:43 +08:00
|
|
|
ASTReader &Reader;
|
2011-12-01 07:21:26 +08:00
|
|
|
ModuleFile &F;
|
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;
|
|
|
|
|
2010-10-05 23:59:54 +08:00
|
|
|
SourceLocation ReadSourceLocation(const ASTReader::RecordData &R,
|
|
|
|
unsigned &I) {
|
|
|
|
return Reader.ReadSourceLocation(F, R, I);
|
|
|
|
}
|
|
|
|
|
2011-07-22 06:35:25 +08:00
|
|
|
template<typename T>
|
|
|
|
T *ReadDeclAs(const ASTReader::RecordData &Record, unsigned &Idx) {
|
|
|
|
return Reader.ReadDeclAs<T>(F, Record, Idx);
|
|
|
|
}
|
|
|
|
|
2009-10-17 05:56:05 +08:00
|
|
|
public:
|
2011-12-01 07:21:26 +08:00
|
|
|
TypeLocReader(ASTReader &Reader, ModuleFile &F,
|
2010-08-19 07:56:43 +08:00
|
|
|
const ASTReader::RecordData &Record, unsigned &Idx)
|
2010-10-05 23:59:54 +08:00
|
|
|
: Reader(Reader), F(F), DeclsCursor(F.DeclsCursor), 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-10-05 23:59:54 +08:00
|
|
|
TL.setBuiltinLoc(ReadSourceLocation(Record, Idx));
|
2010-01-19 02:04:31 +08:00
|
|
|
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) {
|
2010-10-05 23:59:54 +08:00
|
|
|
TL.setNameLoc(ReadSourceLocation(Record, Idx));
|
2009-10-17 05:56:05 +08:00
|
|
|
}
|
2009-10-18 09:05:36 +08:00
|
|
|
void TypeLocReader::VisitPointerTypeLoc(PointerTypeLoc TL) {
|
2010-10-05 23:59:54 +08:00
|
|
|
TL.setStarLoc(ReadSourceLocation(Record, Idx));
|
2009-10-17 05:56:05 +08:00
|
|
|
}
|
2009-10-18 09:05:36 +08:00
|
|
|
void TypeLocReader::VisitBlockPointerTypeLoc(BlockPointerTypeLoc TL) {
|
2010-10-05 23:59:54 +08:00
|
|
|
TL.setCaretLoc(ReadSourceLocation(Record, Idx));
|
2009-10-17 05:56:05 +08:00
|
|
|
}
|
2009-10-18 09:05:36 +08:00
|
|
|
void TypeLocReader::VisitLValueReferenceTypeLoc(LValueReferenceTypeLoc TL) {
|
2010-10-05 23:59:54 +08:00
|
|
|
TL.setAmpLoc(ReadSourceLocation(Record, Idx));
|
2009-10-17 05:56:05 +08:00
|
|
|
}
|
2009-10-18 09:05:36 +08:00
|
|
|
void TypeLocReader::VisitRValueReferenceTypeLoc(RValueReferenceTypeLoc TL) {
|
2010-10-05 23:59:54 +08:00
|
|
|
TL.setAmpAmpLoc(ReadSourceLocation(Record, Idx));
|
2009-10-17 05:56:05 +08:00
|
|
|
}
|
2009-10-18 09:05:36 +08:00
|
|
|
void TypeLocReader::VisitMemberPointerTypeLoc(MemberPointerTypeLoc TL) {
|
2010-10-05 23:59:54 +08:00
|
|
|
TL.setStarLoc(ReadSourceLocation(Record, Idx));
|
2011-03-05 22:42:21 +08:00
|
|
|
TL.setClassTInfo(Reader.GetTypeSourceInfo(F, Record, Idx));
|
2009-10-17 05:56:05 +08:00
|
|
|
}
|
2009-10-18 09:05:36 +08:00
|
|
|
void TypeLocReader::VisitArrayTypeLoc(ArrayTypeLoc TL) {
|
2010-10-05 23:59:54 +08:00
|
|
|
TL.setLBracketLoc(ReadSourceLocation(Record, Idx));
|
|
|
|
TL.setRBracketLoc(ReadSourceLocation(Record, Idx));
|
2009-10-17 05:56:05 +08:00
|
|
|
if (Record[Idx++])
|
2010-10-05 23:59:54 +08:00
|
|
|
TL.setSizeExpr(Reader.ReadExpr(F));
|
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) {
|
2010-10-05 23:59:54 +08:00
|
|
|
TL.setNameLoc(ReadSourceLocation(Record, Idx));
|
2009-10-18 09:05:36 +08:00
|
|
|
}
|
|
|
|
void TypeLocReader::VisitVectorTypeLoc(VectorTypeLoc TL) {
|
2010-10-05 23:59:54 +08:00
|
|
|
TL.setNameLoc(ReadSourceLocation(Record, Idx));
|
2009-10-18 09:05:36 +08:00
|
|
|
}
|
|
|
|
void TypeLocReader::VisitExtVectorTypeLoc(ExtVectorTypeLoc TL) {
|
2010-10-05 23:59:54 +08:00
|
|
|
TL.setNameLoc(ReadSourceLocation(Record, Idx));
|
2009-10-18 09:05:36 +08:00
|
|
|
}
|
|
|
|
void TypeLocReader::VisitFunctionTypeLoc(FunctionTypeLoc TL) {
|
2011-03-12 19:17:06 +08:00
|
|
|
TL.setLocalRangeBegin(ReadSourceLocation(Record, Idx));
|
|
|
|
TL.setLocalRangeEnd(ReadSourceLocation(Record, Idx));
|
2010-10-02 02:44:50 +08:00
|
|
|
TL.setTrailingReturn(Record[Idx++]);
|
2009-10-18 09:05:36 +08:00
|
|
|
for (unsigned i = 0, e = TL.getNumArgs(); i != e; ++i) {
|
2011-07-22 06:35:25 +08:00
|
|
|
TL.setArg(i, ReadDeclAs<ParmVarDecl>(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) {
|
2010-10-05 23:59:54 +08:00
|
|
|
TL.setNameLoc(ReadSourceLocation(Record, Idx));
|
2009-12-05 06:46:56 +08:00
|
|
|
}
|
2009-10-18 09:05:36 +08:00
|
|
|
void TypeLocReader::VisitTypedefTypeLoc(TypedefTypeLoc TL) {
|
2010-10-05 23:59:54 +08:00
|
|
|
TL.setNameLoc(ReadSourceLocation(Record, Idx));
|
2009-10-18 09:05:36 +08:00
|
|
|
}
|
|
|
|
void TypeLocReader::VisitTypeOfExprTypeLoc(TypeOfExprTypeLoc TL) {
|
2010-10-05 23:59:54 +08:00
|
|
|
TL.setTypeofLoc(ReadSourceLocation(Record, Idx));
|
|
|
|
TL.setLParenLoc(ReadSourceLocation(Record, Idx));
|
|
|
|
TL.setRParenLoc(ReadSourceLocation(Record, Idx));
|
2009-10-18 09:05:36 +08:00
|
|
|
}
|
|
|
|
void TypeLocReader::VisitTypeOfTypeLoc(TypeOfTypeLoc TL) {
|
2010-10-05 23:59:54 +08:00
|
|
|
TL.setTypeofLoc(ReadSourceLocation(Record, Idx));
|
|
|
|
TL.setLParenLoc(ReadSourceLocation(Record, Idx));
|
|
|
|
TL.setRParenLoc(ReadSourceLocation(Record, Idx));
|
|
|
|
TL.setUnderlyingTInfo(Reader.GetTypeSourceInfo(F, Record, Idx));
|
2009-10-18 09:05:36 +08:00
|
|
|
}
|
|
|
|
void TypeLocReader::VisitDecltypeTypeLoc(DecltypeTypeLoc TL) {
|
2010-10-05 23:59:54 +08:00
|
|
|
TL.setNameLoc(ReadSourceLocation(Record, Idx));
|
2009-10-18 09:05:36 +08:00
|
|
|
}
|
2011-05-25 06:41:36 +08:00
|
|
|
void TypeLocReader::VisitUnaryTransformTypeLoc(UnaryTransformTypeLoc TL) {
|
|
|
|
TL.setKWLoc(ReadSourceLocation(Record, Idx));
|
|
|
|
TL.setLParenLoc(ReadSourceLocation(Record, Idx));
|
|
|
|
TL.setRParenLoc(ReadSourceLocation(Record, Idx));
|
|
|
|
TL.setUnderlyingTInfo(Reader.GetTypeSourceInfo(F, Record, Idx));
|
|
|
|
}
|
2011-02-20 11:19:35 +08:00
|
|
|
void TypeLocReader::VisitAutoTypeLoc(AutoTypeLoc TL) {
|
|
|
|
TL.setNameLoc(ReadSourceLocation(Record, Idx));
|
|
|
|
}
|
2009-10-18 09:05:36 +08:00
|
|
|
void TypeLocReader::VisitRecordTypeLoc(RecordTypeLoc TL) {
|
2010-10-05 23:59:54 +08:00
|
|
|
TL.setNameLoc(ReadSourceLocation(Record, Idx));
|
2009-10-18 09:05:36 +08:00
|
|
|
}
|
|
|
|
void TypeLocReader::VisitEnumTypeLoc(EnumTypeLoc TL) {
|
2010-10-05 23:59:54 +08:00
|
|
|
TL.setNameLoc(ReadSourceLocation(Record, Idx));
|
2009-10-18 09:05:36 +08:00
|
|
|
}
|
2011-01-06 09:58:22 +08:00
|
|
|
void TypeLocReader::VisitAttributedTypeLoc(AttributedTypeLoc TL) {
|
|
|
|
TL.setAttrNameLoc(ReadSourceLocation(Record, Idx));
|
|
|
|
if (TL.hasAttrOperand()) {
|
|
|
|
SourceRange range;
|
|
|
|
range.setBegin(ReadSourceLocation(Record, Idx));
|
|
|
|
range.setEnd(ReadSourceLocation(Record, Idx));
|
|
|
|
TL.setAttrOperandParensRange(range);
|
|
|
|
}
|
|
|
|
if (TL.hasAttrExprOperand()) {
|
|
|
|
if (Record[Idx++])
|
|
|
|
TL.setAttrExprOperand(Reader.ReadExpr(F));
|
|
|
|
else
|
|
|
|
TL.setAttrExprOperand(0);
|
|
|
|
} else if (TL.hasAttrEnumOperand())
|
|
|
|
TL.setAttrEnumOperandLoc(ReadSourceLocation(Record, Idx));
|
|
|
|
}
|
2009-10-18 09:05:36 +08:00
|
|
|
void TypeLocReader::VisitTemplateTypeParmTypeLoc(TemplateTypeParmTypeLoc TL) {
|
2010-10-05 23:59:54 +08:00
|
|
|
TL.setNameLoc(ReadSourceLocation(Record, Idx));
|
2009-10-18 09:05:36 +08:00
|
|
|
}
|
2009-10-18 17:09:24 +08:00
|
|
|
void TypeLocReader::VisitSubstTemplateTypeParmTypeLoc(
|
|
|
|
SubstTemplateTypeParmTypeLoc TL) {
|
2010-10-05 23:59:54 +08:00
|
|
|
TL.setNameLoc(ReadSourceLocation(Record, Idx));
|
2009-10-18 17:09:24 +08:00
|
|
|
}
|
2011-01-14 10:55:32 +08:00
|
|
|
void TypeLocReader::VisitSubstTemplateTypeParmPackTypeLoc(
|
|
|
|
SubstTemplateTypeParmPackTypeLoc TL) {
|
|
|
|
TL.setNameLoc(ReadSourceLocation(Record, Idx));
|
|
|
|
}
|
2009-10-18 09:05:36 +08:00
|
|
|
void TypeLocReader::VisitTemplateSpecializationTypeLoc(
|
|
|
|
TemplateSpecializationTypeLoc TL) {
|
2010-10-05 23:59:54 +08:00
|
|
|
TL.setTemplateNameLoc(ReadSourceLocation(Record, Idx));
|
|
|
|
TL.setLAngleLoc(ReadSourceLocation(Record, Idx));
|
|
|
|
TL.setRAngleLoc(ReadSourceLocation(Record, Idx));
|
2009-10-29 16:12:44 +08:00
|
|
|
for (unsigned i = 0, e = TL.getNumArgs(); i != e; ++i)
|
|
|
|
TL.setArgLocInfo(i,
|
2010-10-05 23:59:54 +08:00
|
|
|
Reader.GetTemplateArgumentLocInfo(F,
|
|
|
|
TL.getTypePtr()->getArg(i).getKind(),
|
|
|
|
Record, Idx));
|
2009-10-18 09:05:36 +08:00
|
|
|
}
|
2010-12-11 00:29:40 +08:00
|
|
|
void TypeLocReader::VisitParenTypeLoc(ParenTypeLoc TL) {
|
|
|
|
TL.setLParenLoc(ReadSourceLocation(Record, Idx));
|
|
|
|
TL.setRParenLoc(ReadSourceLocation(Record, Idx));
|
|
|
|
}
|
2010-05-12 05:36:43 +08:00
|
|
|
void TypeLocReader::VisitElaboratedTypeLoc(ElaboratedTypeLoc TL) {
|
2010-10-05 23:59:54 +08:00
|
|
|
TL.setKeywordLoc(ReadSourceLocation(Record, Idx));
|
2011-03-02 02:12:44 +08:00
|
|
|
TL.setQualifierLoc(Reader.ReadNestedNameSpecifierLoc(F, Record, Idx));
|
2009-10-18 09:05:36 +08:00
|
|
|
}
|
2010-03-10 11:28:59 +08:00
|
|
|
void TypeLocReader::VisitInjectedClassNameTypeLoc(InjectedClassNameTypeLoc TL) {
|
2010-10-05 23:59:54 +08:00
|
|
|
TL.setNameLoc(ReadSourceLocation(Record, Idx));
|
2010-03-10 11:28:59 +08:00
|
|
|
}
|
2010-04-01 01:34:00 +08:00
|
|
|
void TypeLocReader::VisitDependentNameTypeLoc(DependentNameTypeLoc TL) {
|
2010-10-05 23:59:54 +08:00
|
|
|
TL.setKeywordLoc(ReadSourceLocation(Record, Idx));
|
2011-03-01 09:34:45 +08:00
|
|
|
TL.setQualifierLoc(Reader.ReadNestedNameSpecifierLoc(F, Record, Idx));
|
2010-10-05 23:59:54 +08:00
|
|
|
TL.setNameLoc(ReadSourceLocation(Record, Idx));
|
2009-10-18 09:05:36 +08:00
|
|
|
}
|
2010-06-11 08:33:02 +08:00
|
|
|
void TypeLocReader::VisitDependentTemplateSpecializationTypeLoc(
|
|
|
|
DependentTemplateSpecializationTypeLoc TL) {
|
2010-10-05 23:59:54 +08:00
|
|
|
TL.setKeywordLoc(ReadSourceLocation(Record, Idx));
|
2011-03-02 04:11:18 +08:00
|
|
|
TL.setQualifierLoc(Reader.ReadNestedNameSpecifierLoc(F, Record, Idx));
|
2010-10-05 23:59:54 +08:00
|
|
|
TL.setNameLoc(ReadSourceLocation(Record, Idx));
|
|
|
|
TL.setLAngleLoc(ReadSourceLocation(Record, Idx));
|
|
|
|
TL.setRAngleLoc(ReadSourceLocation(Record, Idx));
|
2010-06-11 08:33:02 +08:00
|
|
|
for (unsigned I = 0, E = TL.getNumArgs(); I != E; ++I)
|
|
|
|
TL.setArgLocInfo(I,
|
2010-10-05 23:59:54 +08:00
|
|
|
Reader.GetTemplateArgumentLocInfo(F,
|
|
|
|
TL.getTypePtr()->getArg(I).getKind(),
|
|
|
|
Record, Idx));
|
2010-06-11 08:33:02 +08:00
|
|
|
}
|
2010-12-20 10:24:11 +08:00
|
|
|
void TypeLocReader::VisitPackExpansionTypeLoc(PackExpansionTypeLoc TL) {
|
|
|
|
TL.setEllipsisLoc(ReadSourceLocation(Record, Idx));
|
|
|
|
}
|
2009-10-18 09:05:36 +08:00
|
|
|
void TypeLocReader::VisitObjCInterfaceTypeLoc(ObjCInterfaceTypeLoc TL) {
|
2010-10-05 23:59:54 +08:00
|
|
|
TL.setNameLoc(ReadSourceLocation(Record, Idx));
|
2010-05-15 19:32:37 +08:00
|
|
|
}
|
|
|
|
void TypeLocReader::VisitObjCObjectTypeLoc(ObjCObjectTypeLoc TL) {
|
|
|
|
TL.setHasBaseTypeAsWritten(Record[Idx++]);
|
2010-10-05 23:59:54 +08:00
|
|
|
TL.setLAngleLoc(ReadSourceLocation(Record, Idx));
|
|
|
|
TL.setRAngleLoc(ReadSourceLocation(Record, Idx));
|
2009-10-23 06:37:11 +08:00
|
|
|
for (unsigned i = 0, e = TL.getNumProtocols(); i != e; ++i)
|
2010-10-05 23:59:54 +08:00
|
|
|
TL.setProtocolLoc(i, ReadSourceLocation(Record, Idx));
|
2009-10-18 09:05:36 +08:00
|
|
|
}
|
|
|
|
void TypeLocReader::VisitObjCObjectPointerTypeLoc(ObjCObjectPointerTypeLoc TL) {
|
2010-10-05 23:59:54 +08:00
|
|
|
TL.setStarLoc(ReadSourceLocation(Record, Idx));
|
2009-10-17 05:56:05 +08:00
|
|
|
}
|
2011-10-07 07:00:33 +08:00
|
|
|
void TypeLocReader::VisitAtomicTypeLoc(AtomicTypeLoc TL) {
|
|
|
|
TL.setKWLoc(ReadSourceLocation(Record, Idx));
|
|
|
|
TL.setLParenLoc(ReadSourceLocation(Record, Idx));
|
|
|
|
TL.setRParenLoc(ReadSourceLocation(Record, Idx));
|
|
|
|
}
|
2009-10-17 05:56:05 +08:00
|
|
|
|
2011-12-01 07:21:26 +08:00
|
|
|
TypeSourceInfo *ASTReader::GetTypeSourceInfo(ModuleFile &F,
|
2010-07-23 06:43:28 +08:00
|
|
|
const RecordData &Record,
|
2009-10-17 05:56:05 +08:00
|
|
|
unsigned &Idx) {
|
2011-07-22 08:38:23 +08:00
|
|
|
QualType InfoTy = readType(F, Record, Idx);
|
2009-10-17 05:56:05 +08:00
|
|
|
if (InfoTy.isNull())
|
|
|
|
return 0;
|
|
|
|
|
2011-09-10 05:34:22 +08:00
|
|
|
TypeSourceInfo *TInfo = getContext().CreateTypeSourceInfo(InfoTy);
|
2010-10-05 23:59:54 +08:00
|
|
|
TypeLocReader TLR(*this, F, 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();
|
2011-09-10 05:34:22 +08:00
|
|
|
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!
|
2011-09-10 05:34:22 +08:00
|
|
|
T = Context.CharTy;
|
2009-04-10 06:27:44 +08:00
|
|
|
break;
|
|
|
|
|
2011-09-10 05:34:22 +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;
|
2011-10-15 07:23:15 +08:00
|
|
|
case PREDEF_TYPE_HALF_ID: T = Context.HalfTy; break;
|
2011-09-10 05:34:22 +08:00
|
|
|
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_BOUND_MEMBER: T = Context.BoundMemberTy; break;
|
2011-10-26 01:37:35 +08:00
|
|
|
case PREDEF_TYPE_PSEUDO_OBJECT: T = Context.PseudoObjectTy; break;
|
2011-09-10 05:34:22 +08:00
|
|
|
case PREDEF_TYPE_DEPENDENT_ID: T = Context.DependentTy; break;
|
|
|
|
case PREDEF_TYPE_UNKNOWN_ANY: T = Context.UnknownAnyTy; 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;
|
|
|
|
case PREDEF_TYPE_AUTO_DEDUCT: T = Context.getAutoDeductType(); break;
|
2011-08-09 23:13:55 +08:00
|
|
|
|
|
|
|
case PREDEF_TYPE_AUTO_RREF_DEDUCT:
|
2011-09-10 05:34:22 +08:00
|
|
|
T = Context.getAutoRRefDeductType();
|
2011-08-09 23:13:55 +08:00
|
|
|
break;
|
2011-10-18 02:09:15 +08:00
|
|
|
|
|
|
|
case PREDEF_TYPE_ARC_UNBRIDGED_CAST:
|
|
|
|
T = Context.ARCUnbridgedCastTy;
|
|
|
|
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()) {
|
2011-07-22 08:38:23 +08:00
|
|
|
TypesLoaded[Index] = readTypeRecord(Index);
|
2010-10-06 02:37:06 +08:00
|
|
|
if (TypesLoaded[Index].isNull())
|
|
|
|
return QualType();
|
|
|
|
|
2010-08-19 07:57:06 +08:00
|
|
|
TypesLoaded[Index]->setFromAST();
|
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
|
|
|
}
|
|
|
|
|
2011-12-01 07:21:26 +08:00
|
|
|
QualType ASTReader::getLocalType(ModuleFile &F, unsigned LocalID) {
|
2011-07-22 08:38:23 +08:00
|
|
|
return GetType(getGlobalTypeID(F, LocalID));
|
|
|
|
}
|
|
|
|
|
|
|
|
serialization::TypeID
|
2011-12-01 07:21:26 +08:00
|
|
|
ASTReader::getGlobalTypeID(ModuleFile &F, unsigned LocalID) const {
|
2011-08-03 00:26:37 +08:00
|
|
|
unsigned FastQuals = LocalID & Qualifiers::FastMask;
|
|
|
|
unsigned LocalIndex = LocalID >> Qualifiers::FastWidth;
|
|
|
|
|
|
|
|
if (LocalIndex < NUM_PREDEF_TYPE_IDS)
|
|
|
|
return LocalID;
|
|
|
|
|
|
|
|
ContinuousRangeMap<uint32_t, int, 2>::iterator I
|
|
|
|
= F.TypeRemap.find(LocalIndex - NUM_PREDEF_TYPE_IDS);
|
|
|
|
assert(I != F.TypeRemap.end() && "Invalid index into type index remap");
|
|
|
|
|
|
|
|
unsigned GlobalIndex = LocalIndex + I->second;
|
|
|
|
return (GlobalIndex << Qualifiers::FastWidth) | FastQuals;
|
|
|
|
}
|
|
|
|
|
2009-10-29 16:12:44 +08:00
|
|
|
TemplateArgumentLocInfo
|
2011-12-01 07:21:26 +08:00
|
|
|
ASTReader::GetTemplateArgumentLocInfo(ModuleFile &F,
|
2010-10-05 23:59:54 +08:00
|
|
|
TemplateArgument::ArgKind Kind,
|
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-10-05 23:59:54 +08:00
|
|
|
return ReadExpr(F);
|
2009-10-29 16:12:44 +08:00
|
|
|
case TemplateArgument::Type:
|
2010-10-05 23:59:54 +08:00
|
|
|
return GetTypeSourceInfo(F, Record, Index);
|
2009-11-11 09:00:40 +08:00
|
|
|
case TemplateArgument::Template: {
|
2011-03-03 01:09:35 +08:00
|
|
|
NestedNameSpecifierLoc QualifierLoc = ReadNestedNameSpecifierLoc(F, Record,
|
|
|
|
Index);
|
2011-01-06 02:58:31 +08:00
|
|
|
SourceLocation TemplateNameLoc = ReadSourceLocation(F, Record, Index);
|
2011-03-03 01:09:35 +08:00
|
|
|
return TemplateArgumentLocInfo(QualifierLoc, TemplateNameLoc,
|
2011-01-06 02:58:31 +08:00
|
|
|
SourceLocation());
|
|
|
|
}
|
|
|
|
case TemplateArgument::TemplateExpansion: {
|
2011-03-03 01:09:35 +08:00
|
|
|
NestedNameSpecifierLoc QualifierLoc = ReadNestedNameSpecifierLoc(F, Record,
|
|
|
|
Index);
|
2010-10-05 23:59:54 +08:00
|
|
|
SourceLocation TemplateNameLoc = ReadSourceLocation(F, Record, Index);
|
2011-01-06 01:40:24 +08:00
|
|
|
SourceLocation EllipsisLoc = ReadSourceLocation(F, Record, Index);
|
2011-03-03 01:09:35 +08:00
|
|
|
return TemplateArgumentLocInfo(QualifierLoc, TemplateNameLoc,
|
2011-01-06 01:40:24 +08:00
|
|
|
EllipsisLoc);
|
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
|
2011-12-01 07:21:26 +08:00
|
|
|
ASTReader::ReadTemplateArgumentLoc(ModuleFile &F,
|
2010-07-23 06:43:28 +08:00
|
|
|
const RecordData &Record, unsigned &Index) {
|
2010-10-05 23:59:54 +08:00
|
|
|
TemplateArgument Arg = ReadTemplateArgument(F, 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-10-05 23:59:54 +08:00
|
|
|
return TemplateArgumentLoc(Arg, GetTemplateArgumentLocInfo(F, Arg.getKind(),
|
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);
|
|
|
|
}
|
|
|
|
|
2011-12-01 07:21:26 +08:00
|
|
|
uint64_t ASTReader::readCXXBaseSpecifiers(ModuleFile &M, const RecordData &Record,
|
2011-08-04 08:01:48 +08:00
|
|
|
unsigned &Idx){
|
|
|
|
if (Idx >= Record.size())
|
2010-10-30 06:39:52 +08:00
|
|
|
return 0;
|
|
|
|
|
2011-08-04 08:01:48 +08:00
|
|
|
unsigned LocalID = Record[Idx++];
|
|
|
|
return getGlobalBitOffset(M, M.CXXBaseSpecifiersOffsets[LocalID - 1]);
|
2010-10-30 06:39:52 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
CXXBaseSpecifier *ASTReader::GetExternalCXXBaseSpecifiers(uint64_t Offset) {
|
Introduce a global bit-offset continuous range map into the ASTReader,
so that we have one, simple way to map from global bit offsets to
local bit offsets. Eliminates a number of loops over the chain, and
generalizes for more interesting bit remappings.
Also, as an amusing oddity, we were computing global bit offsets
*backwards* for preprocessed entities (e.g., the directly included PCH
file in the chain would start at offset zero, rather than the original
PCH that occurs first in translation unit). Even more amusingly, it
made precompiled preambles work, because we were forgetting to adjust
the local bit offset to a global bit offset when storing preprocessed
entity offsets in the ASTUnit. Two wrongs made a right, and now
they're both right.
llvm-svn: 135750
2011-07-22 14:10:01 +08:00
|
|
|
RecordLocation Loc = getLocalBitOffset(Offset);
|
|
|
|
llvm::BitstreamCursor &Cursor = Loc.F->DeclsCursor;
|
2010-10-30 06:39:52 +08:00
|
|
|
SavedStreamPosition SavedPosition(Cursor);
|
Introduce a global bit-offset continuous range map into the ASTReader,
so that we have one, simple way to map from global bit offsets to
local bit offsets. Eliminates a number of loops over the chain, and
generalizes for more interesting bit remappings.
Also, as an amusing oddity, we were computing global bit offsets
*backwards* for preprocessed entities (e.g., the directly included PCH
file in the chain would start at offset zero, rather than the original
PCH that occurs first in translation unit). Even more amusingly, it
made precompiled preambles work, because we were forgetting to adjust
the local bit offset to a global bit offset when storing preprocessed
entity offsets in the ASTUnit. Two wrongs made a right, and now
they're both right.
llvm-svn: 135750
2011-07-22 14:10:01 +08:00
|
|
|
Cursor.JumpToBit(Loc.Offset);
|
2010-10-30 06:39:52 +08:00
|
|
|
ReadingKindTracker ReadingKind(Read_Decl, *this);
|
|
|
|
RecordData Record;
|
|
|
|
unsigned Code = Cursor.ReadCode();
|
|
|
|
unsigned RecCode = Cursor.ReadRecord(Code, Record);
|
|
|
|
if (RecCode != DECL_CXX_BASE_SPECIFIERS) {
|
|
|
|
Error("Malformed AST file: missing C++ base specifiers");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned Idx = 0;
|
|
|
|
unsigned NumBases = Record[Idx++];
|
2011-09-10 05:34:22 +08:00
|
|
|
void *Mem = Context.Allocate(sizeof(CXXBaseSpecifier) * NumBases);
|
2010-10-30 06:39:52 +08:00
|
|
|
CXXBaseSpecifier *Bases = new (Mem) CXXBaseSpecifier [NumBases];
|
|
|
|
for (unsigned I = 0; I != NumBases; ++I)
|
Introduce a global bit-offset continuous range map into the ASTReader,
so that we have one, simple way to map from global bit offsets to
local bit offsets. Eliminates a number of loops over the chain, and
generalizes for more interesting bit remappings.
Also, as an amusing oddity, we were computing global bit offsets
*backwards* for preprocessed entities (e.g., the directly included PCH
file in the chain would start at offset zero, rather than the original
PCH that occurs first in translation unit). Even more amusingly, it
made precompiled preambles work, because we were forgetting to adjust
the local bit offset to a global bit offset when storing preprocessed
entity offsets in the ASTUnit. Two wrongs made a right, and now
they're both right.
llvm-svn: 135750
2011-07-22 14:10:01 +08:00
|
|
|
Bases[I] = ReadCXXBaseSpecifier(*Loc.F, Record, Idx);
|
2010-10-30 06:39:52 +08:00
|
|
|
return Bases;
|
|
|
|
}
|
|
|
|
|
2011-07-22 06:35:25 +08:00
|
|
|
serialization::DeclID
|
2011-12-01 07:21:26 +08:00
|
|
|
ASTReader::getGlobalDeclID(ModuleFile &F, unsigned LocalID) const {
|
2011-08-04 00:05:40 +08:00
|
|
|
if (LocalID < NUM_PREDEF_DECL_IDS)
|
2011-08-03 23:48:04 +08:00
|
|
|
return LocalID;
|
|
|
|
|
|
|
|
ContinuousRangeMap<uint32_t, int, 2>::iterator I
|
2011-08-04 00:05:40 +08:00
|
|
|
= F.DeclRemap.find(LocalID - NUM_PREDEF_DECL_IDS);
|
2011-08-03 23:48:04 +08:00
|
|
|
assert(I != F.DeclRemap.end() && "Invalid index into decl index remap");
|
|
|
|
|
|
|
|
return LocalID + I->second;
|
2011-07-22 06:35:25 +08:00
|
|
|
}
|
|
|
|
|
2011-09-01 08:58:55 +08:00
|
|
|
bool ASTReader::isDeclIDFromModule(serialization::GlobalDeclID ID,
|
2011-12-01 07:21:26 +08:00
|
|
|
ModuleFile &M) const {
|
2011-09-01 08:58:55 +08:00
|
|
|
GlobalDeclMapType::const_iterator I = GlobalDeclMap.find(ID);
|
|
|
|
assert(I != GlobalDeclMap.end() && "Corrupted global declaration map");
|
|
|
|
return &M == I->second;
|
|
|
|
}
|
|
|
|
|
2011-11-03 10:20:32 +08:00
|
|
|
SourceLocation ASTReader::getSourceLocationForDeclID(GlobalDeclID ID) {
|
|
|
|
if (ID < NUM_PREDEF_DECL_IDS)
|
|
|
|
return SourceLocation();
|
|
|
|
|
|
|
|
unsigned Index = ID - NUM_PREDEF_DECL_IDS;
|
|
|
|
|
|
|
|
if (Index > DeclsLoaded.size()) {
|
|
|
|
Error("declaration ID out-of-range for AST file");
|
|
|
|
return SourceLocation();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Decl *D = DeclsLoaded[Index])
|
|
|
|
return D->getLocation();
|
|
|
|
|
|
|
|
unsigned RawLocation = 0;
|
|
|
|
RecordLocation Rec = DeclCursorForID(ID, RawLocation);
|
|
|
|
return ReadSourceLocation(*Rec.F, RawLocation);
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:57:32 +08:00
|
|
|
Decl *ASTReader::GetDecl(DeclID ID) {
|
2011-08-04 00:05:40 +08:00
|
|
|
if (ID < NUM_PREDEF_DECL_IDS) {
|
|
|
|
switch ((PredefinedDeclIDs)ID) {
|
2011-08-12 08:15:20 +08:00
|
|
|
case PREDEF_DECL_NULL_ID:
|
2011-08-04 00:05:40 +08:00
|
|
|
return 0;
|
2011-08-12 08:15:20 +08:00
|
|
|
|
|
|
|
case PREDEF_DECL_TRANSLATION_UNIT_ID:
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.getTranslationUnitDecl();
|
2011-08-12 13:46:01 +08:00
|
|
|
|
|
|
|
case PREDEF_DECL_OBJC_ID_ID:
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.getObjCIdDecl();
|
2011-08-12 13:59:41 +08:00
|
|
|
|
2011-08-12 14:17:30 +08:00
|
|
|
case PREDEF_DECL_OBJC_SEL_ID:
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.getObjCSelDecl();
|
2011-08-12 14:17:30 +08:00
|
|
|
|
2011-08-12 13:59:41 +08:00
|
|
|
case PREDEF_DECL_OBJC_CLASS_ID:
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.getObjCClassDecl();
|
2011-08-12 14:49:56 +08:00
|
|
|
|
|
|
|
case PREDEF_DECL_INT_128_ID:
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.getInt128Decl();
|
2011-08-12 14:49:56 +08:00
|
|
|
|
|
|
|
case PREDEF_DECL_UNSIGNED_INT_128_ID:
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.getUInt128Decl();
|
2011-09-08 09:46:34 +08:00
|
|
|
|
|
|
|
case PREDEF_DECL_OBJC_INSTANCETYPE_ID:
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.getObjCInstanceTypeDecl();
|
2011-08-04 00:05:40 +08:00
|
|
|
}
|
|
|
|
|
2009-04-10 06:27:44 +08:00
|
|
|
return 0;
|
2011-08-04 00:05:40 +08:00
|
|
|
}
|
|
|
|
|
2011-08-12 08:15:20 +08:00
|
|
|
unsigned Index = ID - NUM_PREDEF_DECL_IDS;
|
|
|
|
|
|
|
|
if (Index > 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;
|
|
|
|
}
|
2011-08-12 08:15:20 +08:00
|
|
|
|
2011-12-17 06:37:11 +08:00
|
|
|
if (!DeclsLoaded[Index]) {
|
2011-08-03 23:48:04 +08:00
|
|
|
ReadDeclRecord(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
|
|
|
}
|
|
|
|
|
Completely re-implement (de-)serialization of declaration
chains. The previous implementation relied heavily on the declaration
chain being stored as a (circular) linked list on disk, as it is in
memory. However, when deserializing from multiple modules, the
different chains could get mixed up, leading to broken declaration chains.
The new solution keeps track of the first and last declarations in the
chain for each module file. When we load a declaration, we search all
of the module files for redeclarations of that declaration, then
splice together all of the lists into a coherent whole (along with any
redeclarations that were actually parsed).
As a drive-by fix, (de-)serialize the redeclaration chains of
TypedefNameDecls, which had somehow gotten missed previously. Add a
test of this serialization.
This new scheme creates a redeclaration table that is fairly large in
the PCH file (on the order of 400k for Cocoa.h's 12MB PCH file). The
table is mmap'd in and searched via a binary search, but it's still
quite large. A future tweak will eliminate entries for declarations
that have no redeclarations anywhere, and should
drastically reduce the size of this table.
llvm-svn: 146841
2011-12-18 07:38:30 +08:00
|
|
|
DeclID ASTReader::mapGlobalIDToModuleFileGlobalID(ModuleFile &M,
|
|
|
|
DeclID GlobalID) {
|
|
|
|
if (GlobalID < NUM_PREDEF_DECL_IDS)
|
|
|
|
return GlobalID;
|
|
|
|
|
|
|
|
GlobalDeclMapType::const_iterator I = GlobalDeclMap.find(GlobalID);
|
|
|
|
assert(I != GlobalDeclMap.end() && "Corrupted global declaration map");
|
|
|
|
ModuleFile *Owner = I->second;
|
|
|
|
|
|
|
|
llvm::DenseMap<ModuleFile *, serialization::DeclID>::iterator Pos
|
|
|
|
= M.GlobalToLocalDeclIDs.find(Owner);
|
|
|
|
if (Pos == M.GlobalToLocalDeclIDs.end())
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
return GlobalID - Owner->BaseDeclID + Pos->second;
|
|
|
|
}
|
|
|
|
|
2011-12-01 07:21:26 +08:00
|
|
|
serialization::DeclID ASTReader::ReadDeclID(ModuleFile &F,
|
2011-07-22 06:35:25 +08:00
|
|
|
const RecordData &Record,
|
|
|
|
unsigned &Idx) {
|
|
|
|
if (Idx >= Record.size()) {
|
|
|
|
Error("Corrupted AST file");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return getGlobalDeclID(F, Record[Idx++]);
|
|
|
|
}
|
|
|
|
|
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-10-28 17:29:32 +08:00
|
|
|
// Switch case IDs are per Decl.
|
|
|
|
ClearSwitchCaseIDs();
|
|
|
|
|
2010-07-23 01:01:13 +08:00
|
|
|
// Offset here is a global offset across the entire chain.
|
Introduce a global bit-offset continuous range map into the ASTReader,
so that we have one, simple way to map from global bit offsets to
local bit offsets. Eliminates a number of loops over the chain, and
generalizes for more interesting bit remappings.
Also, as an amusing oddity, we were computing global bit offsets
*backwards* for preprocessed entities (e.g., the directly included PCH
file in the chain would start at offset zero, rather than the original
PCH that occurs first in translation unit). Even more amusingly, it
made precompiled preambles work, because we were forgetting to adjust
the local bit offset to a global bit offset when storing preprocessed
entity offsets in the ASTUnit. Two wrongs made a right, and now
they're both right.
llvm-svn: 135750
2011-07-22 14:10:01 +08:00
|
|
|
RecordLocation Loc = getLocalBitOffset(Offset);
|
|
|
|
Loc.F->DeclsCursor.JumpToBit(Loc.Offset);
|
|
|
|
return ReadStmtFromStream(*Loc.F);
|
2009-04-18 08:07:54 +08:00
|
|
|
}
|
|
|
|
|
2011-08-25 05:27:34 +08:00
|
|
|
namespace {
|
|
|
|
class FindExternalLexicalDeclsVisitor {
|
|
|
|
ASTReader &Reader;
|
|
|
|
const DeclContext *DC;
|
|
|
|
bool (*isKindWeWant)(Decl::Kind);
|
2011-08-27 06:04:51 +08:00
|
|
|
|
2011-08-25 05:27:34 +08:00
|
|
|
SmallVectorImpl<Decl*> &Decls;
|
|
|
|
bool PredefsVisited[NUM_PREDEF_DECL_IDS];
|
|
|
|
|
|
|
|
public:
|
|
|
|
FindExternalLexicalDeclsVisitor(ASTReader &Reader, const DeclContext *DC,
|
|
|
|
bool (*isKindWeWant)(Decl::Kind),
|
|
|
|
SmallVectorImpl<Decl*> &Decls)
|
|
|
|
: Reader(Reader), DC(DC), isKindWeWant(isKindWeWant), Decls(Decls)
|
|
|
|
{
|
|
|
|
for (unsigned I = 0; I != NUM_PREDEF_DECL_IDS; ++I)
|
|
|
|
PredefsVisited[I] = false;
|
|
|
|
}
|
|
|
|
|
2011-12-01 07:21:26 +08:00
|
|
|
static bool visit(ModuleFile &M, bool Preorder, void *UserData) {
|
2011-08-25 05:27:34 +08:00
|
|
|
if (Preorder)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
FindExternalLexicalDeclsVisitor *This
|
|
|
|
= static_cast<FindExternalLexicalDeclsVisitor *>(UserData);
|
|
|
|
|
2011-12-01 07:21:26 +08:00
|
|
|
ModuleFile::DeclContextInfosMap::iterator Info
|
2011-08-25 05:27:34 +08:00
|
|
|
= M.DeclContextInfos.find(This->DC);
|
|
|
|
if (Info == M.DeclContextInfos.end() || !Info->second.LexicalDecls)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Load all of the declaration IDs
|
|
|
|
for (const KindDeclIDPair *ID = Info->second.LexicalDecls,
|
|
|
|
*IDE = ID + Info->second.NumLexicalDecls;
|
|
|
|
ID != IDE; ++ID) {
|
|
|
|
if (This->isKindWeWant && !This->isKindWeWant((Decl::Kind)ID->first))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// Don't add predefined declarations to the lexical context more
|
|
|
|
// than once.
|
|
|
|
if (ID->second < NUM_PREDEF_DECL_IDS) {
|
|
|
|
if (This->PredefsVisited[ID->second])
|
|
|
|
continue;
|
|
|
|
|
|
|
|
This->PredefsVisited[ID->second] = true;
|
|
|
|
}
|
|
|
|
|
2011-08-27 06:04:51 +08:00
|
|
|
if (Decl *D = This->Reader.GetLocalDecl(M, ID->second)) {
|
|
|
|
if (!This->DC->isDeclInLexicalTraversal(D))
|
|
|
|
This->Decls.push_back(D);
|
|
|
|
}
|
2011-08-25 05:27:34 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2011-07-16 05:46:17 +08:00
|
|
|
ExternalLoadResult ASTReader::FindExternalLexicalDecls(const DeclContext *DC,
|
2010-10-15 04:14:34 +08:00
|
|
|
bool (*isKindWeWant)(Decl::Kind),
|
2011-07-23 18:55:15 +08:00
|
|
|
SmallVectorImpl<Decl*> &Decls) {
|
2011-08-25 03:03:07 +08:00
|
|
|
// There might be lexical decls in multiple modules, for the TU at
|
2011-08-25 05:27:34 +08:00
|
|
|
// least. Walk all of the modules in the order they were loaded.
|
|
|
|
FindExternalLexicalDeclsVisitor Visitor(*this, DC, isKindWeWant, Decls);
|
|
|
|
ModuleMgr.visitDepthFirst(&FindExternalLexicalDeclsVisitor::visit, &Visitor);
|
2009-04-23 06:34:57 +08:00
|
|
|
++NumLexicalDeclContextsRead;
|
2011-07-16 05:46:17 +08:00
|
|
|
return ELR_Success;
|
2009-04-10 06:27:44 +08:00
|
|
|
}
|
|
|
|
|
2011-11-03 10:20:32 +08:00
|
|
|
namespace {
|
|
|
|
|
|
|
|
class DeclIDComp {
|
|
|
|
ASTReader &Reader;
|
2011-12-01 07:21:26 +08:00
|
|
|
ModuleFile &Mod;
|
2011-11-03 10:20:32 +08:00
|
|
|
|
|
|
|
public:
|
2011-12-01 07:21:26 +08:00
|
|
|
DeclIDComp(ASTReader &Reader, ModuleFile &M) : Reader(Reader), Mod(M) {}
|
2011-11-03 10:20:32 +08:00
|
|
|
|
|
|
|
bool operator()(LocalDeclID L, LocalDeclID R) const {
|
|
|
|
SourceLocation LHS = getLocation(L);
|
|
|
|
SourceLocation RHS = getLocation(R);
|
|
|
|
return Reader.getSourceManager().isBeforeInTranslationUnit(LHS, RHS);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool operator()(SourceLocation LHS, LocalDeclID R) const {
|
|
|
|
SourceLocation RHS = getLocation(R);
|
|
|
|
return Reader.getSourceManager().isBeforeInTranslationUnit(LHS, RHS);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool operator()(LocalDeclID L, SourceLocation RHS) const {
|
|
|
|
SourceLocation LHS = getLocation(L);
|
|
|
|
return Reader.getSourceManager().isBeforeInTranslationUnit(LHS, RHS);
|
|
|
|
}
|
|
|
|
|
|
|
|
SourceLocation getLocation(LocalDeclID ID) const {
|
|
|
|
return Reader.getSourceManager().getFileLoc(
|
|
|
|
Reader.getSourceLocationForDeclID(Reader.getGlobalDeclID(Mod, ID)));
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void ASTReader::FindFileRegionDecls(FileID File,
|
|
|
|
unsigned Offset, unsigned Length,
|
|
|
|
SmallVectorImpl<Decl *> &Decls) {
|
|
|
|
SourceManager &SM = getSourceManager();
|
|
|
|
|
|
|
|
llvm::DenseMap<FileID, FileDeclsInfo>::iterator I = FileDeclIDs.find(File);
|
|
|
|
if (I == FileDeclIDs.end())
|
|
|
|
return;
|
|
|
|
|
|
|
|
FileDeclsInfo &DInfo = I->second;
|
|
|
|
if (DInfo.Decls.empty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
SourceLocation
|
|
|
|
BeginLoc = SM.getLocForStartOfFile(File).getLocWithOffset(Offset);
|
|
|
|
SourceLocation EndLoc = BeginLoc.getLocWithOffset(Length);
|
|
|
|
|
|
|
|
DeclIDComp DIDComp(*this, *DInfo.Mod);
|
|
|
|
ArrayRef<serialization::LocalDeclID>::iterator
|
|
|
|
BeginIt = std::lower_bound(DInfo.Decls.begin(), DInfo.Decls.end(),
|
|
|
|
BeginLoc, DIDComp);
|
|
|
|
if (BeginIt != DInfo.Decls.begin())
|
|
|
|
--BeginIt;
|
|
|
|
|
2011-11-24 04:27:36 +08:00
|
|
|
// If we are pointing at a top-level decl inside an objc container, we need
|
|
|
|
// to backtrack until we find it otherwise we will fail to report that the
|
|
|
|
// region overlaps with an objc container.
|
|
|
|
while (BeginIt != DInfo.Decls.begin() &&
|
|
|
|
GetDecl(getGlobalDeclID(*DInfo.Mod, *BeginIt))
|
|
|
|
->isTopLevelDeclInObjCContainer())
|
|
|
|
--BeginIt;
|
|
|
|
|
2011-11-03 10:20:32 +08:00
|
|
|
ArrayRef<serialization::LocalDeclID>::iterator
|
|
|
|
EndIt = std::upper_bound(DInfo.Decls.begin(), DInfo.Decls.end(),
|
|
|
|
EndLoc, DIDComp);
|
|
|
|
if (EndIt != DInfo.Decls.end())
|
|
|
|
++EndIt;
|
|
|
|
|
|
|
|
for (ArrayRef<serialization::LocalDeclID>::iterator
|
|
|
|
DIt = BeginIt; DIt != EndIt; ++DIt)
|
|
|
|
Decls.push_back(GetDecl(getGlobalDeclID(*DInfo.Mod, *DIt)));
|
|
|
|
}
|
|
|
|
|
2011-08-25 03:03:07 +08:00
|
|
|
namespace {
|
2011-12-01 07:21:26 +08:00
|
|
|
/// \brief ModuleFile visitor used to perform name lookup into a
|
2011-08-25 03:03:07 +08:00
|
|
|
/// declaration context.
|
|
|
|
class DeclContextNameLookupVisitor {
|
|
|
|
ASTReader &Reader;
|
|
|
|
const DeclContext *DC;
|
|
|
|
DeclarationName Name;
|
|
|
|
SmallVectorImpl<NamedDecl *> &Decls;
|
|
|
|
|
|
|
|
public:
|
|
|
|
DeclContextNameLookupVisitor(ASTReader &Reader,
|
|
|
|
const DeclContext *DC, DeclarationName Name,
|
|
|
|
SmallVectorImpl<NamedDecl *> &Decls)
|
|
|
|
: Reader(Reader), DC(DC), Name(Name), Decls(Decls) { }
|
|
|
|
|
2011-12-01 07:21:26 +08:00
|
|
|
static bool visit(ModuleFile &M, void *UserData) {
|
2011-08-25 03:03:07 +08:00
|
|
|
DeclContextNameLookupVisitor *This
|
|
|
|
= static_cast<DeclContextNameLookupVisitor *>(UserData);
|
|
|
|
|
|
|
|
// Check whether we have any visible declaration information for
|
|
|
|
// this context in this module.
|
2011-12-01 07:21:26 +08:00
|
|
|
ModuleFile::DeclContextInfosMap::iterator Info
|
2011-08-25 03:03:07 +08:00
|
|
|
= M.DeclContextInfos.find(This->DC);
|
|
|
|
if (Info == M.DeclContextInfos.end() || !Info->second.NameLookupTableData)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Look for this name within this module.
|
|
|
|
ASTDeclContextNameLookupTable *LookupTable =
|
|
|
|
(ASTDeclContextNameLookupTable*)Info->second.NameLookupTableData;
|
|
|
|
ASTDeclContextNameLookupTable::iterator Pos
|
|
|
|
= LookupTable->find(This->Name);
|
|
|
|
if (Pos == LookupTable->end())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
bool FoundAnything = false;
|
|
|
|
ASTDeclContextNameLookupTrait::data_type Data = *Pos;
|
|
|
|
for (; Data.first != Data.second; ++Data.first) {
|
|
|
|
NamedDecl *ND = This->Reader.GetLocalDeclAs<NamedDecl>(M, *Data.first);
|
|
|
|
if (!ND)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (ND->getDeclName() != This->Name) {
|
|
|
|
assert(!This->Name.getCXXNameType().isNull() &&
|
|
|
|
"Name mismatch without a type");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Record this declaration.
|
|
|
|
FoundAnything = true;
|
|
|
|
This->Decls.push_back(ND);
|
|
|
|
}
|
|
|
|
|
|
|
|
return FoundAnything;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
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
|
|
|
|
2011-07-23 18:55:15 +08:00
|
|
|
SmallVector<NamedDecl *, 64> Decls;
|
2011-08-25 03:03:07 +08:00
|
|
|
DeclContextNameLookupVisitor Visitor(*this, DC, Name, Decls);
|
|
|
|
ModuleMgr.visit(&DeclContextNameLookupVisitor::visit, &Visitor);
|
2009-04-23 06:34:57 +08:00
|
|
|
++NumVisibleDeclContextsRead;
|
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
|
|
|
}
|
|
|
|
|
2011-09-14 05:35:00 +08:00
|
|
|
/// \brief Under non-PCH compilation the consumer receives the objc methods
|
|
|
|
/// before receiving the implementation, and codegen depends on this.
|
|
|
|
/// We simulate this by deserializing and passing to consumer the methods of the
|
|
|
|
/// implementation before passing the deserialized implementation decl.
|
|
|
|
static void PassObjCImplDeclToConsumer(ObjCImplDecl *ImplD,
|
|
|
|
ASTConsumer *Consumer) {
|
|
|
|
assert(ImplD && Consumer);
|
|
|
|
|
|
|
|
for (ObjCImplDecl::method_iterator
|
|
|
|
I = ImplD->meth_begin(), E = ImplD->meth_end(); I != E; ++I)
|
|
|
|
Consumer->HandleInterestingDecl(DeclGroupRef(*I));
|
|
|
|
|
|
|
|
Consumer->HandleInterestingDecl(DeclGroupRef(ImplD));
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:56:43 +08:00
|
|
|
void ASTReader::PassInterestingDeclsToConsumer() {
|
2010-07-07 23:46:26 +08:00
|
|
|
assert(Consumer);
|
|
|
|
while (!InterestingDecls.empty()) {
|
2011-09-14 05:35:00 +08:00
|
|
|
Decl *D = InterestingDecls.front();
|
2010-07-07 23:46:26 +08:00
|
|
|
InterestingDecls.pop_front();
|
2011-09-14 05:35:00 +08:00
|
|
|
|
2011-12-01 07:18:26 +08:00
|
|
|
PassInterestingDeclToConsumer(D);
|
2010-07-07 23:46:26 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-12-01 07:18:26 +08:00
|
|
|
void ASTReader::PassInterestingDeclToConsumer(Decl *D) {
|
|
|
|
if (ObjCImplDecl *ImplD = dyn_cast<ObjCImplDecl>(D))
|
|
|
|
PassObjCImplDeclToConsumer(ImplD, Consumer);
|
|
|
|
else
|
|
|
|
Consumer->HandleInterestingDecl(DeclGroupRef(D));
|
|
|
|
}
|
|
|
|
|
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
|
|
|
}
|
2011-09-16 02:47:32 +08:00
|
|
|
ExternalDefinitions.clear();
|
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);
|
2011-07-22 02:46:38 +08:00
|
|
|
if (unsigned TotalNumSLocEntries = getTotalNumSLocs())
|
2009-04-27 14:38:32 +08:00
|
|
|
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");
|
2011-07-22 03:50:14 +08:00
|
|
|
dump();
|
|
|
|
std::fprintf(stderr, "\n");
|
|
|
|
}
|
|
|
|
|
2011-12-01 07:21:26 +08:00
|
|
|
template<typename Key, typename ModuleFile, unsigned InitialCapacity>
|
2011-07-22 03:50:14 +08:00
|
|
|
static void
|
2011-07-23 18:55:15 +08:00
|
|
|
dumpModuleIDMap(StringRef Name,
|
2011-12-01 07:21:26 +08:00
|
|
|
const ContinuousRangeMap<Key, ModuleFile *,
|
2011-07-22 03:50:14 +08:00
|
|
|
InitialCapacity> &Map) {
|
|
|
|
if (Map.begin() == Map.end())
|
|
|
|
return;
|
|
|
|
|
2011-12-01 07:21:26 +08:00
|
|
|
typedef ContinuousRangeMap<Key, ModuleFile *, InitialCapacity> MapType;
|
2011-07-22 03:50:14 +08:00
|
|
|
llvm::errs() << Name << ":\n";
|
|
|
|
for (typename MapType::const_iterator I = Map.begin(), IEnd = Map.end();
|
|
|
|
I != IEnd; ++I) {
|
|
|
|
llvm::errs() << " " << I->first << " -> " << I->second->FileName
|
|
|
|
<< "\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ASTReader::dump() {
|
2011-12-01 07:21:26 +08:00
|
|
|
llvm::errs() << "*** PCH/ModuleFile Remappings:\n";
|
Introduce a global bit-offset continuous range map into the ASTReader,
so that we have one, simple way to map from global bit offsets to
local bit offsets. Eliminates a number of loops over the chain, and
generalizes for more interesting bit remappings.
Also, as an amusing oddity, we were computing global bit offsets
*backwards* for preprocessed entities (e.g., the directly included PCH
file in the chain would start at offset zero, rather than the original
PCH that occurs first in translation unit). Even more amusingly, it
made precompiled preambles work, because we were forgetting to adjust
the local bit offset to a global bit offset when storing preprocessed
entity offsets in the ASTUnit. Two wrongs made a right, and now
they're both right.
llvm-svn: 135750
2011-07-22 14:10:01 +08:00
|
|
|
dumpModuleIDMap("Global bit offset map", GlobalBitOffsetsMap);
|
2011-07-22 03:50:14 +08:00
|
|
|
dumpModuleIDMap("Global source location entry map", GlobalSLocEntryMap);
|
2011-07-29 08:21:44 +08:00
|
|
|
dumpModuleIDMap("Global type map", GlobalTypeMap);
|
2011-07-29 08:56:45 +08:00
|
|
|
dumpModuleIDMap("Global declaration map", GlobalDeclMap);
|
|
|
|
dumpModuleIDMap("Global identifier map", GlobalIdentifierMap);
|
2011-12-01 08:59:36 +08:00
|
|
|
dumpModuleIDMap("Global submodule map", GlobalSubmoduleMap);
|
2011-07-29 08:56:45 +08:00
|
|
|
dumpModuleIDMap("Global selector map", GlobalSelectorMap);
|
|
|
|
dumpModuleIDMap("Global preprocessed entity map",
|
|
|
|
GlobalPreprocessedEntityMap);
|
2011-08-02 19:12:41 +08:00
|
|
|
|
|
|
|
llvm::errs() << "\n*** PCH/Modules Loaded:";
|
|
|
|
for (ModuleManager::ModuleConstIterator M = ModuleMgr.begin(),
|
|
|
|
MEnd = ModuleMgr.end();
|
|
|
|
M != MEnd; ++M)
|
|
|
|
(*M)->dump();
|
2009-04-10 06:27:44 +08:00
|
|
|
}
|
|
|
|
|
2011-04-29 07:46:20 +08:00
|
|
|
/// Return the amount of memory used by memory buffers, breaking down
|
|
|
|
/// by heap-backed versus mmap'ed memory.
|
|
|
|
void ASTReader::getMemoryBufferSizes(MemoryBufferSizes &sizes) const {
|
2011-07-26 04:32:21 +08:00
|
|
|
for (ModuleConstIterator I = ModuleMgr.begin(),
|
|
|
|
E = ModuleMgr.end(); I != E; ++I) {
|
|
|
|
if (llvm::MemoryBuffer *buf = (*I)->Buffer.get()) {
|
2011-04-29 07:46:20 +08:00
|
|
|
size_t bytes = buf->getBufferSize();
|
|
|
|
switch (buf->getBufferKind()) {
|
|
|
|
case llvm::MemoryBuffer::MemoryBuffer_Malloc:
|
|
|
|
sizes.malloc_bytes += bytes;
|
|
|
|
break;
|
|
|
|
case llvm::MemoryBuffer::MemoryBuffer_MMap:
|
|
|
|
sizes.mmap_bytes += bytes;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2011-07-26 04:32:21 +08:00
|
|
|
}
|
2011-04-29 07:46:20 +08:00
|
|
|
}
|
|
|
|
|
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) {
|
Make the loading of information attached to an IdentifierInfo from an
AST file more lazy, so that we don't eagerly load that information for
all known identifiers each time a new AST file is loaded. The eager
reloading made some sense in the context of precompiled headers, since
very few identifiers were defined before PCH load time. With modules,
however, a huge amount of code can get parsed before we see an
@import, so laziness becomes important here.
The approach taken to make this information lazy is fairly simple:
when we load a new AST file, we mark all of the existing identifiers
as being out-of-date. Whenever we want to access information that may
come from an AST (e.g., whether the identifier has a macro definition,
or what top-level declarations have that name), we check the
out-of-date bit and, if it's set, ask the AST reader to update the
IdentifierInfo from the AST files. The update is a merge, and we now
take care to merge declarations before/after imports with declarations
from multiple imports.
The results of this optimization are fairly dramatic. On a small
application that brings in 14 non-trivial modules, this takes modules
from being > 3x slower than a "perfect" PCH file down to 30% slower
for a full rebuild. A partial rebuild (where the PCH file or modules
can be re-used) is down to 7% slower. Making the PCH file just a
little imperfect (e.g., adding two smallish modules used by a bunch of
.m files that aren't in the PCH file) tips the scales in favor of the
modules approach, with 24% faster partial rebuilds.
This is just a first step; the lazy scheme could possibly be improved
by adding versioning, so we don't search into modules we already
searched. Moreover, we'll need similar lazy schemes for all of the
other lookup data structures, such as DeclContexts.
llvm-svn: 143100
2011-10-27 17:33:13 +08:00
|
|
|
SemaObj->pushExternalDeclIntoScope(PreloadedDecls[I],
|
|
|
|
PreloadedDecls[I]->getDeclName());
|
2009-04-23 05:15:06 +08:00
|
|
|
}
|
|
|
|
PreloadedDecls.clear();
|
2009-04-23 06:02:47 +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!");
|
2011-07-28 08:57:24 +08:00
|
|
|
if (!SemaObj->StdNamespace)
|
|
|
|
SemaObj->StdNamespace = SemaDeclRefs[0];
|
|
|
|
if (!SemaObj->StdBadAlloc)
|
|
|
|
SemaObj->StdBadAlloc = SemaDeclRefs[1];
|
2010-08-02 15:14:54 +08:00
|
|
|
}
|
|
|
|
|
2011-02-16 03:46:30 +08:00
|
|
|
if (!FPPragmaOptions.empty()) {
|
|
|
|
assert(FPPragmaOptions.size() == 1 && "Wrong number of FP_PRAGMA_OPTIONS");
|
|
|
|
SemaObj->FPFeatures.fp_contract = FPPragmaOptions[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!OpenCLExtensions.empty()) {
|
|
|
|
unsigned I = 0;
|
|
|
|
#define OPENCLEXT(nm) SemaObj->OpenCLFeatures.nm = OpenCLExtensions[I++];
|
|
|
|
#include "clang/Basic/OpenCLExtensions.def"
|
|
|
|
|
|
|
|
assert(OpenCLExtensions.size() == I && "Wrong number of OPENCL_EXTENSIONS");
|
|
|
|
}
|
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
|
|
|
}
|
|
|
|
|
2011-08-20 12:39:52 +08:00
|
|
|
IdentifierInfo* ASTReader::get(const char *NameStart, const char *NameEnd) {
|
|
|
|
IdentifierLookupVisitor Visitor(StringRef(NameStart, NameEnd - NameStart));
|
|
|
|
ModuleMgr.visit(IdentifierLookupVisitor::visit, &Visitor);
|
Make the loading of information attached to an IdentifierInfo from an
AST file more lazy, so that we don't eagerly load that information for
all known identifiers each time a new AST file is loaded. The eager
reloading made some sense in the context of precompiled headers, since
very few identifiers were defined before PCH load time. With modules,
however, a huge amount of code can get parsed before we see an
@import, so laziness becomes important here.
The approach taken to make this information lazy is fairly simple:
when we load a new AST file, we mark all of the existing identifiers
as being out-of-date. Whenever we want to access information that may
come from an AST (e.g., whether the identifier has a macro definition,
or what top-level declarations have that name), we check the
out-of-date bit and, if it's set, ask the AST reader to update the
IdentifierInfo from the AST files. The update is a merge, and we now
take care to merge declarations before/after imports with declarations
from multiple imports.
The results of this optimization are fairly dramatic. On a small
application that brings in 14 non-trivial modules, this takes modules
from being > 3x slower than a "perfect" PCH file down to 30% slower
for a full rebuild. A partial rebuild (where the PCH file or modules
can be re-used) is down to 7% slower. Making the PCH file just a
little imperfect (e.g., adding two smallish modules used by a bunch of
.m files that aren't in the PCH file) tips the scales in favor of the
modules approach, with 24% faster partial rebuilds.
This is just a first step; the lazy scheme could possibly be improved
by adding versioning, so we don't search into modules we already
searched. Moreover, we'll need similar lazy schemes for all of the
other lookup data structures, such as DeclContexts.
llvm-svn: 143100
2011-10-27 17:33:13 +08:00
|
|
|
IdentifierInfo *II = Visitor.getIdentifierInfo();
|
|
|
|
if (II)
|
|
|
|
II->setOutOfDate(false);
|
|
|
|
return 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
|
|
|
}
|
|
|
|
|
2010-10-15 06:11:03 +08:00
|
|
|
namespace clang {
|
|
|
|
/// \brief An identifier-lookup iterator that enumerates all of the
|
|
|
|
/// identifiers stored within a set of AST files.
|
|
|
|
class ASTIdentifierIterator : public IdentifierIterator {
|
|
|
|
/// \brief The AST reader whose identifiers are being enumerated.
|
|
|
|
const ASTReader &Reader;
|
|
|
|
|
|
|
|
/// \brief The current index into the chain of AST files stored in
|
|
|
|
/// the AST reader.
|
|
|
|
unsigned Index;
|
|
|
|
|
|
|
|
/// \brief The current position within the identifier lookup table
|
|
|
|
/// of the current AST file.
|
|
|
|
ASTIdentifierLookupTable::key_iterator Current;
|
|
|
|
|
|
|
|
/// \brief The end position within the identifier lookup table of
|
|
|
|
/// the current AST file.
|
|
|
|
ASTIdentifierLookupTable::key_iterator End;
|
|
|
|
|
|
|
|
public:
|
|
|
|
explicit ASTIdentifierIterator(const ASTReader &Reader);
|
|
|
|
|
2011-07-23 18:55:15 +08:00
|
|
|
virtual StringRef Next();
|
2010-10-15 06:11:03 +08:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
ASTIdentifierIterator::ASTIdentifierIterator(const ASTReader &Reader)
|
2011-07-26 04:32:21 +08:00
|
|
|
: Reader(Reader), Index(Reader.ModuleMgr.size() - 1) {
|
2010-10-15 06:11:03 +08:00
|
|
|
ASTIdentifierLookupTable *IdTable
|
2011-07-26 04:32:21 +08:00
|
|
|
= (ASTIdentifierLookupTable *)Reader.ModuleMgr[Index].IdentifierLookupTable;
|
2010-10-15 06:11:03 +08:00
|
|
|
Current = IdTable->key_begin();
|
|
|
|
End = IdTable->key_end();
|
|
|
|
}
|
|
|
|
|
2011-07-23 18:55:15 +08:00
|
|
|
StringRef ASTIdentifierIterator::Next() {
|
2010-10-15 06:11:03 +08:00
|
|
|
while (Current == End) {
|
|
|
|
// If we have exhausted all of our AST files, we're done.
|
|
|
|
if (Index == 0)
|
2011-07-23 18:55:15 +08:00
|
|
|
return StringRef();
|
2010-10-15 06:11:03 +08:00
|
|
|
|
|
|
|
--Index;
|
|
|
|
ASTIdentifierLookupTable *IdTable
|
2011-07-26 04:32:21 +08:00
|
|
|
= (ASTIdentifierLookupTable *)Reader.ModuleMgr[Index].
|
|
|
|
IdentifierLookupTable;
|
2010-10-15 06:11:03 +08:00
|
|
|
Current = IdTable->key_begin();
|
|
|
|
End = IdTable->key_end();
|
|
|
|
}
|
|
|
|
|
|
|
|
// We have any identifiers remaining in the current AST file; return
|
|
|
|
// the next one.
|
|
|
|
std::pair<const char*, unsigned> Key = *Current;
|
|
|
|
++Current;
|
2011-07-23 18:55:15 +08:00
|
|
|
return StringRef(Key.first, Key.second);
|
2010-10-15 06:11:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
IdentifierIterator *ASTReader::getIdentifiers() const {
|
|
|
|
return new ASTIdentifierIterator(*this);
|
|
|
|
}
|
|
|
|
|
2011-08-25 22:51:20 +08:00
|
|
|
namespace clang { namespace serialization {
|
|
|
|
class ReadMethodPoolVisitor {
|
|
|
|
ASTReader &Reader;
|
|
|
|
Selector Sel;
|
|
|
|
llvm::SmallVector<ObjCMethodDecl *, 4> InstanceMethods;
|
|
|
|
llvm::SmallVector<ObjCMethodDecl *, 4> FactoryMethods;
|
|
|
|
|
|
|
|
/// \brief Build an ObjCMethodList from a vector of Objective-C method
|
|
|
|
/// declarations.
|
|
|
|
ObjCMethodList
|
|
|
|
buildObjCMethodList(const SmallVectorImpl<ObjCMethodDecl *> &Vec) const
|
|
|
|
{
|
|
|
|
ObjCMethodList List;
|
|
|
|
ObjCMethodList *Prev = 0;
|
|
|
|
for (unsigned I = 0, N = Vec.size(); I != N; ++I) {
|
|
|
|
if (!List.Method) {
|
|
|
|
// This is the first method, which is the easy case.
|
|
|
|
List.Method = Vec[I];
|
|
|
|
Prev = &List;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
ObjCMethodList *Mem =
|
|
|
|
Reader.getSema()->BumpAlloc.Allocate<ObjCMethodList>();
|
|
|
|
Prev->Next = new (Mem) ObjCMethodList(Vec[I], 0);
|
|
|
|
Prev = Prev->Next;
|
|
|
|
}
|
|
|
|
|
|
|
|
return List;
|
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
|
|
|
ReadMethodPoolVisitor(ASTReader &Reader, Selector Sel)
|
|
|
|
: Reader(Reader), Sel(Sel) { }
|
|
|
|
|
2011-12-01 07:21:26 +08:00
|
|
|
static bool visit(ModuleFile &M, void *UserData) {
|
2011-08-25 22:51:20 +08:00
|
|
|
ReadMethodPoolVisitor *This
|
|
|
|
= static_cast<ReadMethodPoolVisitor *>(UserData);
|
|
|
|
|
|
|
|
if (!M.SelectorLookupTable)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
ASTSelectorLookupTable *PoolTable
|
|
|
|
= (ASTSelectorLookupTable*)M.SelectorLookupTable;
|
|
|
|
ASTSelectorLookupTable::iterator Pos = PoolTable->find(This->Sel);
|
|
|
|
if (Pos == PoolTable->end())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
++This->Reader.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?
|
2011-08-25 22:51:20 +08:00
|
|
|
++This->Reader.NumMethodPoolEntriesRead;
|
2010-08-19 07:57:06 +08:00
|
|
|
ASTSelectorLookupTrait::data_type Data = *Pos;
|
2011-08-25 22:51:20 +08:00
|
|
|
if (This->Reader.DeserializationListener)
|
|
|
|
This->Reader.DeserializationListener->SelectorRead(Data.ID,
|
|
|
|
This->Sel);
|
|
|
|
|
|
|
|
This->InstanceMethods.append(Data.Instance.begin(), Data.Instance.end());
|
|
|
|
This->FactoryMethods.append(Data.Factory.begin(), Data.Factory.end());
|
|
|
|
return true;
|
2010-08-05 04:40:17 +08:00
|
|
|
}
|
2011-08-25 22:51:20 +08:00
|
|
|
|
|
|
|
/// \brief Retrieve the instance methods found by this visitor.
|
|
|
|
ObjCMethodList getInstanceMethods() const {
|
|
|
|
return buildObjCMethodList(InstanceMethods);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief Retrieve the instance methods found by this visitor.
|
|
|
|
ObjCMethodList getFactoryMethods() const {
|
|
|
|
return buildObjCMethodList(FactoryMethods);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
} } // end namespace clang::serialization
|
2009-04-25 05:10:55 +08:00
|
|
|
|
2011-08-25 22:51:20 +08:00
|
|
|
std::pair<ObjCMethodList, ObjCMethodList>
|
|
|
|
ASTReader::ReadMethodPool(Selector Sel) {
|
|
|
|
ReadMethodPoolVisitor Visitor(*this, Sel);
|
|
|
|
ModuleMgr.visit(&ReadMethodPoolVisitor::visit, &Visitor);
|
|
|
|
std::pair<ObjCMethodList, ObjCMethodList> Result;
|
|
|
|
Result.first = Visitor.getInstanceMethods();
|
|
|
|
Result.second = Visitor.getFactoryMethods();
|
|
|
|
|
|
|
|
if (!Result.first.Method && !Result.second.Method)
|
|
|
|
++NumMethodPoolMisses;
|
|
|
|
return Result;
|
2009-04-25 05:10:55 +08:00
|
|
|
}
|
|
|
|
|
2011-06-29 00:20:02 +08:00
|
|
|
void ASTReader::ReadKnownNamespaces(
|
2011-07-23 18:55:15 +08:00
|
|
|
SmallVectorImpl<NamespaceDecl *> &Namespaces) {
|
2011-06-29 00:20:02 +08:00
|
|
|
Namespaces.clear();
|
|
|
|
|
|
|
|
for (unsigned I = 0, N = KnownNamespaces.size(); I != N; ++I) {
|
|
|
|
if (NamespaceDecl *Namespace
|
|
|
|
= dyn_cast_or_null<NamespaceDecl>(GetDecl(KnownNamespaces[I])))
|
|
|
|
Namespaces.push_back(Namespace);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-07-28 04:58:46 +08:00
|
|
|
void ASTReader::ReadTentativeDefinitions(
|
|
|
|
SmallVectorImpl<VarDecl *> &TentativeDefs) {
|
|
|
|
for (unsigned I = 0, N = TentativeDefinitions.size(); I != N; ++I) {
|
|
|
|
VarDecl *Var = dyn_cast_or_null<VarDecl>(GetDecl(TentativeDefinitions[I]));
|
|
|
|
if (Var)
|
|
|
|
TentativeDefs.push_back(Var);
|
|
|
|
}
|
|
|
|
TentativeDefinitions.clear();
|
|
|
|
}
|
|
|
|
|
2011-07-28 05:45:57 +08:00
|
|
|
void ASTReader::ReadUnusedFileScopedDecls(
|
|
|
|
SmallVectorImpl<const DeclaratorDecl *> &Decls) {
|
|
|
|
for (unsigned I = 0, N = UnusedFileScopedDecls.size(); I != N; ++I) {
|
|
|
|
DeclaratorDecl *D
|
|
|
|
= dyn_cast_or_null<DeclaratorDecl>(GetDecl(UnusedFileScopedDecls[I]));
|
|
|
|
if (D)
|
|
|
|
Decls.push_back(D);
|
|
|
|
}
|
|
|
|
UnusedFileScopedDecls.clear();
|
|
|
|
}
|
|
|
|
|
2011-07-28 05:57:17 +08:00
|
|
|
void ASTReader::ReadDelegatingConstructors(
|
|
|
|
SmallVectorImpl<CXXConstructorDecl *> &Decls) {
|
|
|
|
for (unsigned I = 0, N = DelegatingCtorDecls.size(); I != N; ++I) {
|
|
|
|
CXXConstructorDecl *D
|
|
|
|
= dyn_cast_or_null<CXXConstructorDecl>(GetDecl(DelegatingCtorDecls[I]));
|
|
|
|
if (D)
|
|
|
|
Decls.push_back(D);
|
|
|
|
}
|
|
|
|
DelegatingCtorDecls.clear();
|
|
|
|
}
|
|
|
|
|
2011-07-28 08:39:29 +08:00
|
|
|
void ASTReader::ReadExtVectorDecls(SmallVectorImpl<TypedefNameDecl *> &Decls) {
|
|
|
|
for (unsigned I = 0, N = ExtVectorDecls.size(); I != N; ++I) {
|
|
|
|
TypedefNameDecl *D
|
|
|
|
= dyn_cast_or_null<TypedefNameDecl>(GetDecl(ExtVectorDecls[I]));
|
|
|
|
if (D)
|
|
|
|
Decls.push_back(D);
|
|
|
|
}
|
|
|
|
ExtVectorDecls.clear();
|
|
|
|
}
|
|
|
|
|
2011-07-28 08:53:40 +08:00
|
|
|
void ASTReader::ReadDynamicClasses(SmallVectorImpl<CXXRecordDecl *> &Decls) {
|
|
|
|
for (unsigned I = 0, N = DynamicClasses.size(); I != N; ++I) {
|
|
|
|
CXXRecordDecl *D
|
|
|
|
= dyn_cast_or_null<CXXRecordDecl>(GetDecl(DynamicClasses[I]));
|
|
|
|
if (D)
|
|
|
|
Decls.push_back(D);
|
|
|
|
}
|
|
|
|
DynamicClasses.clear();
|
|
|
|
}
|
|
|
|
|
2011-07-28 22:20:37 +08:00
|
|
|
void
|
|
|
|
ASTReader::ReadLocallyScopedExternalDecls(SmallVectorImpl<NamedDecl *> &Decls) {
|
|
|
|
for (unsigned I = 0, N = LocallyScopedExternalDecls.size(); I != N; ++I) {
|
|
|
|
NamedDecl *D
|
|
|
|
= dyn_cast_or_null<NamedDecl>(GetDecl(LocallyScopedExternalDecls[I]));
|
|
|
|
if (D)
|
|
|
|
Decls.push_back(D);
|
|
|
|
}
|
|
|
|
LocallyScopedExternalDecls.clear();
|
|
|
|
}
|
|
|
|
|
2011-07-28 22:54:22 +08:00
|
|
|
void ASTReader::ReadReferencedSelectors(
|
|
|
|
SmallVectorImpl<std::pair<Selector, SourceLocation> > &Sels) {
|
|
|
|
if (ReferencedSelectorsData.empty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
// If there are @selector references added them to its pool. This is for
|
|
|
|
// implementation of -Wselector.
|
|
|
|
unsigned int DataSize = ReferencedSelectorsData.size()-1;
|
|
|
|
unsigned I = 0;
|
|
|
|
while (I < DataSize) {
|
|
|
|
Selector Sel = DecodeSelector(ReferencedSelectorsData[I++]);
|
|
|
|
SourceLocation SelLoc
|
|
|
|
= SourceLocation::getFromRawEncoding(ReferencedSelectorsData[I++]);
|
|
|
|
Sels.push_back(std::make_pair(Sel, SelLoc));
|
|
|
|
}
|
|
|
|
ReferencedSelectorsData.clear();
|
|
|
|
}
|
|
|
|
|
2011-07-29 02:09:57 +08:00
|
|
|
void ASTReader::ReadWeakUndeclaredIdentifiers(
|
|
|
|
SmallVectorImpl<std::pair<IdentifierInfo *, WeakInfo> > &WeakIDs) {
|
|
|
|
if (WeakUndeclaredIdentifiers.empty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
for (unsigned I = 0, N = WeakUndeclaredIdentifiers.size(); I < N; /*none*/) {
|
|
|
|
IdentifierInfo *WeakId
|
|
|
|
= DecodeIdentifierInfo(WeakUndeclaredIdentifiers[I++]);
|
|
|
|
IdentifierInfo *AliasId
|
|
|
|
= DecodeIdentifierInfo(WeakUndeclaredIdentifiers[I++]);
|
|
|
|
SourceLocation Loc
|
|
|
|
= SourceLocation::getFromRawEncoding(WeakUndeclaredIdentifiers[I++]);
|
|
|
|
bool Used = WeakUndeclaredIdentifiers[I++];
|
|
|
|
WeakInfo WI(AliasId, Loc);
|
|
|
|
WI.setUsed(Used);
|
|
|
|
WeakIDs.push_back(std::make_pair(WeakId, WI));
|
|
|
|
}
|
|
|
|
WeakUndeclaredIdentifiers.clear();
|
|
|
|
}
|
|
|
|
|
2011-07-29 03:11:31 +08:00
|
|
|
void ASTReader::ReadUsedVTables(SmallVectorImpl<ExternalVTableUse> &VTables) {
|
|
|
|
for (unsigned Idx = 0, N = VTableUses.size(); Idx < N; /* In loop */) {
|
|
|
|
ExternalVTableUse VT;
|
|
|
|
VT.Record = dyn_cast_or_null<CXXRecordDecl>(GetDecl(VTableUses[Idx++]));
|
|
|
|
VT.Location = SourceLocation::getFromRawEncoding(VTableUses[Idx++]);
|
|
|
|
VT.DefinitionRequired = VTableUses[Idx++];
|
|
|
|
VTables.push_back(VT);
|
|
|
|
}
|
|
|
|
|
|
|
|
VTableUses.clear();
|
|
|
|
}
|
|
|
|
|
2011-07-29 03:49:54 +08:00
|
|
|
void ASTReader::ReadPendingInstantiations(
|
|
|
|
SmallVectorImpl<std::pair<ValueDecl *, SourceLocation> > &Pending) {
|
|
|
|
for (unsigned Idx = 0, N = PendingInstantiations.size(); Idx < N;) {
|
|
|
|
ValueDecl *D = cast<ValueDecl>(GetDecl(PendingInstantiations[Idx++]));
|
|
|
|
SourceLocation Loc
|
|
|
|
= SourceLocation::getFromRawEncoding(PendingInstantiations[Idx++]);
|
|
|
|
Pending.push_back(std::make_pair(D, Loc));
|
|
|
|
}
|
|
|
|
PendingInstantiations.clear();
|
|
|
|
}
|
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2011-07-29 04:55:49 +08:00
|
|
|
void ASTReader::SetIdentifierInfo(IdentifierID 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,
|
2011-07-23 18:55:15 +08:00
|
|
|
const SmallVectorImpl<uint32_t> &DeclIDs,
|
2009-07-07 02:54:52 +08:00
|
|
|
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) {
|
Make the loading of information attached to an IdentifierInfo from an
AST file more lazy, so that we don't eagerly load that information for
all known identifiers each time a new AST file is loaded. The eager
reloading made some sense in the context of precompiled headers, since
very few identifiers were defined before PCH load time. With modules,
however, a huge amount of code can get parsed before we see an
@import, so laziness becomes important here.
The approach taken to make this information lazy is fairly simple:
when we load a new AST file, we mark all of the existing identifiers
as being out-of-date. Whenever we want to access information that may
come from an AST (e.g., whether the identifier has a macro definition,
or what top-level declarations have that name), we check the
out-of-date bit and, if it's set, ask the AST reader to update the
IdentifierInfo from the AST files. The update is a merge, and we now
take care to merge declarations before/after imports with declarations
from multiple imports.
The results of this optimization are fairly dramatic. On a small
application that brings in 14 non-trivial modules, this takes modules
from being > 3x slower than a "perfect" PCH file down to 30% slower
for a full rebuild. A partial rebuild (where the PCH file or modules
can be re-used) is down to 7% slower. Making the PCH file just a
little imperfect (e.g., adding two smallish modules used by a bunch of
.m files that aren't in the PCH file) tips the scales in favor of the
modules approach, with 24% faster partial rebuilds.
This is just a first step; the lazy scheme could possibly be improved
by adding versioning, so we don't search into modules we already
searched. Moreover, we'll need similar lazy schemes for all of the
other lookup data structures, such as DeclContexts.
llvm-svn: 143100
2011-10-27 17:33:13 +08:00
|
|
|
// 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.
|
|
|
|
SemaObj->pushExternalDeclIntoScope(D, II);
|
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);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-07-29 04:55:49 +08:00
|
|
|
IdentifierInfo *ASTReader::DecodeIdentifierInfo(IdentifierID 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
|
|
|
|
2010-07-21 08:46:22 +08:00
|
|
|
ID -= 1;
|
|
|
|
if (!IdentifiersLoaded[ID]) {
|
2011-07-20 08:59:32 +08:00
|
|
|
GlobalIdentifierMapType::iterator I = GlobalIdentifierMap.find(ID + 1);
|
|
|
|
assert(I != GlobalIdentifierMap.end() && "Corrupted global identifier map");
|
2011-12-01 07:21:26 +08:00
|
|
|
ModuleFile *M = I->second;
|
2011-07-29 08:56:45 +08:00
|
|
|
unsigned Index = ID - M->BaseIdentifierID;
|
|
|
|
const char *Str = M->IdentifierTableData + M->IdentifierOffsets[Index];
|
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]
|
2011-09-10 06:02:16 +08:00
|
|
|
= &PP.getIdentifierTable().get(StringRef(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
|
|
|
}
|
|
|
|
|
2011-12-01 07:21:26 +08:00
|
|
|
IdentifierInfo *ASTReader::getLocalIdentifier(ModuleFile &M, unsigned LocalID) {
|
2011-07-29 04:55:49 +08:00
|
|
|
return DecodeIdentifierInfo(getGlobalIdentifierID(M, LocalID));
|
|
|
|
}
|
|
|
|
|
2011-12-01 07:21:26 +08:00
|
|
|
IdentifierID ASTReader::getGlobalIdentifierID(ModuleFile &M, unsigned LocalID) {
|
2011-08-04 05:49:18 +08:00
|
|
|
if (LocalID < NUM_PREDEF_IDENT_IDS)
|
|
|
|
return LocalID;
|
|
|
|
|
|
|
|
ContinuousRangeMap<uint32_t, int, 2>::iterator I
|
|
|
|
= M.IdentifierRemap.find(LocalID - NUM_PREDEF_IDENT_IDS);
|
|
|
|
assert(I != M.IdentifierRemap.end()
|
|
|
|
&& "Invalid index into identifier index remap");
|
|
|
|
|
|
|
|
return LocalID + I->second;
|
2011-07-29 04:55:49 +08:00
|
|
|
}
|
|
|
|
|
Revamp the SourceManager to separate the representation of parsed
source locations from source locations loaded from an AST/PCH file.
Previously, loading an AST/PCH file involved carefully pre-allocating
space at the beginning of the source manager for the source locations
and FileIDs that correspond to the prefix, and then appending the
source locations/FileIDs used for parsing the remaining translation
unit. This design forced us into loading PCH files early, as a prefix,
whic has become a rather significant limitation.
This patch splits the SourceManager space into two parts: for source
location "addresses", the lower values (growing upward) are used to
describe parsed code, while upper values (growing downward) are used
for source locations loaded from AST/PCH files. Similarly, positive
FileIDs are used to describe parsed code while negative FileIDs are
used to file/macro locations loaded from AST/PCH files. As a result,
we can load PCH/AST files even during parsing, making various
improvemnts in the future possible, e.g., teaching #include <foo.h> to
look for and load <foo.h.gch> if it happens to be already available.
This patch was originally written by Sebastian Redl, then brought
forward to the modern age by Jonathan Turner, and finally
polished/finished by me to be committed.
llvm-svn: 135484
2011-07-20 00:10:42 +08:00
|
|
|
bool ASTReader::ReadSLocEntry(int ID) {
|
2011-04-20 08:21:03 +08:00
|
|
|
return ReadSLocEntryRecord(ID) != Success;
|
2009-04-27 14:38:32 +08:00
|
|
|
}
|
|
|
|
|
2011-12-01 08:59:36 +08:00
|
|
|
serialization::SubmoduleID
|
|
|
|
ASTReader::getGlobalSubmoduleID(ModuleFile &M, unsigned LocalID) {
|
|
|
|
if (LocalID < NUM_PREDEF_SUBMODULE_IDS)
|
|
|
|
return LocalID;
|
|
|
|
|
|
|
|
ContinuousRangeMap<uint32_t, int, 2>::iterator I
|
|
|
|
= M.SubmoduleRemap.find(LocalID - NUM_PREDEF_SUBMODULE_IDS);
|
|
|
|
assert(I != M.SubmoduleRemap.end()
|
|
|
|
&& "Invalid index into identifier index remap");
|
|
|
|
|
|
|
|
return LocalID + I->second;
|
|
|
|
}
|
|
|
|
|
|
|
|
Module *ASTReader::getSubmodule(SubmoduleID GlobalID) {
|
|
|
|
if (GlobalID < NUM_PREDEF_SUBMODULE_IDS) {
|
|
|
|
assert(GlobalID == 0 && "Unhandled global submodule ID");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (GlobalID > SubmodulesLoaded.size()) {
|
|
|
|
Error("submodule ID out of range in AST file");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return SubmodulesLoaded[GlobalID - NUM_PREDEF_SUBMODULE_IDS];
|
|
|
|
}
|
|
|
|
|
2011-12-01 07:21:26 +08:00
|
|
|
Selector ASTReader::getLocalSelector(ModuleFile &M, unsigned LocalID) {
|
2011-07-29 05:16:51 +08:00
|
|
|
return DecodeSelector(getGlobalSelectorID(M, LocalID));
|
|
|
|
}
|
|
|
|
|
|
|
|
Selector ASTReader::DecodeSelector(serialization::SelectorID 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.
|
2011-07-20 09:10:58 +08:00
|
|
|
GlobalSelectorMapType::iterator I = GlobalSelectorMap.find(ID);
|
|
|
|
assert(I != GlobalSelectorMap.end() && "Corrupted global selector map");
|
2011-12-01 07:21:26 +08:00
|
|
|
ModuleFile &M = *I->second;
|
2011-07-29 08:56:45 +08:00
|
|
|
ASTSelectorLookupTrait Trait(*this, M);
|
2011-08-04 07:28:44 +08:00
|
|
|
unsigned Idx = ID - M.BaseSelectorID - NUM_PREDEF_SELECTOR_IDS;
|
2011-07-20 09:10:58 +08:00
|
|
|
SelectorsLoaded[ID - 1] =
|
2011-07-29 08:56:45 +08:00
|
|
|
Trait.ReadKey(M.SelectorLookupTableData + M.SelectorOffsets[Idx], 0);
|
2011-07-20 09:10:58 +08:00
|
|
|
if (DeserializationListener)
|
|
|
|
DeserializationListener->SelectorRead(ID, SelectorsLoaded[ID - 1]);
|
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
|
|
|
}
|
|
|
|
|
2011-07-28 22:41:43 +08:00
|
|
|
Selector ASTReader::GetExternalSelector(serialization::SelectorID 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
|
|
|
}
|
|
|
|
|
2011-08-04 07:28:44 +08:00
|
|
|
serialization::SelectorID
|
2011-12-01 07:21:26 +08:00
|
|
|
ASTReader::getGlobalSelectorID(ModuleFile &M, unsigned LocalID) const {
|
2011-08-04 07:28:44 +08:00
|
|
|
if (LocalID < NUM_PREDEF_SELECTOR_IDS)
|
|
|
|
return LocalID;
|
|
|
|
|
|
|
|
ContinuousRangeMap<uint32_t, int, 2>::iterator I
|
|
|
|
= M.SelectorRemap.find(LocalID - NUM_PREDEF_SELECTOR_IDS);
|
|
|
|
assert(I != M.SelectorRemap.end()
|
|
|
|
&& "Invalid index into identifier index remap");
|
|
|
|
|
|
|
|
return LocalID + I->second;
|
2011-07-28 22:41:43 +08:00
|
|
|
}
|
|
|
|
|
2009-09-09 23:08:12 +08:00
|
|
|
DeclarationName
|
2011-12-01 07:21:26 +08:00
|
|
|
ASTReader::ReadDeclarationName(ModuleFile &F,
|
2011-07-22 08:38:23 +08:00
|
|
|
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:
|
2011-07-29 04:55:49 +08:00
|
|
|
return DeclarationName(GetIdentifierInfo(F, Record, Idx));
|
2009-04-10 06:27:44 +08:00
|
|
|
|
|
|
|
case DeclarationName::ObjCZeroArgSelector:
|
|
|
|
case DeclarationName::ObjCOneArgSelector:
|
|
|
|
case DeclarationName::ObjCMultiArgSelector:
|
2011-07-29 05:16:51 +08:00
|
|
|
return DeclarationName(ReadSelector(F, Record, Idx));
|
2009-04-10 06:27:44 +08:00
|
|
|
|
|
|
|
case DeclarationName::CXXConstructorName:
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.DeclarationNames.getCXXConstructorName(
|
|
|
|
Context.getCanonicalType(readType(F, Record, Idx)));
|
2009-04-10 06:27:44 +08:00
|
|
|
|
|
|
|
case DeclarationName::CXXDestructorName:
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.DeclarationNames.getCXXDestructorName(
|
|
|
|
Context.getCanonicalType(readType(F, Record, Idx)));
|
2009-04-10 06:27:44 +08:00
|
|
|
|
|
|
|
case DeclarationName::CXXConversionFunctionName:
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.DeclarationNames.getCXXConversionFunctionName(
|
|
|
|
Context.getCanonicalType(readType(F, Record, Idx)));
|
2009-04-10 06:27:44 +08:00
|
|
|
|
|
|
|
case DeclarationName::CXXOperatorName:
|
2011-09-10 05:34:22 +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:
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.DeclarationNames.getCXXLiteralOperatorName(
|
2011-07-29 04:55:49 +08:00
|
|
|
GetIdentifierInfo(F, Record, Idx));
|
2009-11-29 15:34:05 +08:00
|
|
|
|
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
|
|
|
|
2011-12-01 07:21:26 +08:00
|
|
|
void ASTReader::ReadDeclarationNameLoc(ModuleFile &F,
|
2010-10-16 02:21:24 +08:00
|
|
|
DeclarationNameLoc &DNLoc,
|
|
|
|
DeclarationName Name,
|
|
|
|
const RecordData &Record, unsigned &Idx) {
|
|
|
|
switch (Name.getNameKind()) {
|
|
|
|
case DeclarationName::CXXConstructorName:
|
|
|
|
case DeclarationName::CXXDestructorName:
|
|
|
|
case DeclarationName::CXXConversionFunctionName:
|
|
|
|
DNLoc.NamedType.TInfo = GetTypeSourceInfo(F, Record, Idx);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case DeclarationName::CXXOperatorName:
|
|
|
|
DNLoc.CXXOperatorName.BeginOpNameLoc
|
|
|
|
= ReadSourceLocation(F, Record, Idx).getRawEncoding();
|
|
|
|
DNLoc.CXXOperatorName.EndOpNameLoc
|
|
|
|
= ReadSourceLocation(F, Record, Idx).getRawEncoding();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case DeclarationName::CXXLiteralOperatorName:
|
|
|
|
DNLoc.CXXLiteralOperatorName.OpNameLoc
|
|
|
|
= ReadSourceLocation(F, Record, Idx).getRawEncoding();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case DeclarationName::Identifier:
|
|
|
|
case DeclarationName::ObjCZeroArgSelector:
|
|
|
|
case DeclarationName::ObjCOneArgSelector:
|
|
|
|
case DeclarationName::ObjCMultiArgSelector:
|
|
|
|
case DeclarationName::CXXUsingDirective:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-12-01 07:21:26 +08:00
|
|
|
void ASTReader::ReadDeclarationNameInfo(ModuleFile &F,
|
2010-10-16 02:21:24 +08:00
|
|
|
DeclarationNameInfo &NameInfo,
|
|
|
|
const RecordData &Record, unsigned &Idx) {
|
2011-07-22 08:38:23 +08:00
|
|
|
NameInfo.setName(ReadDeclarationName(F, Record, Idx));
|
2010-10-16 02:21:24 +08:00
|
|
|
NameInfo.setLoc(ReadSourceLocation(F, Record, Idx));
|
|
|
|
DeclarationNameLoc DNLoc;
|
|
|
|
ReadDeclarationNameLoc(F, DNLoc, NameInfo.getName(), Record, Idx);
|
|
|
|
NameInfo.setInfo(DNLoc);
|
|
|
|
}
|
|
|
|
|
2011-12-01 07:21:26 +08:00
|
|
|
void ASTReader::ReadQualifierInfo(ModuleFile &F, QualifierInfo &Info,
|
2010-10-16 02:21:24 +08:00
|
|
|
const RecordData &Record, unsigned &Idx) {
|
2011-02-25 10:25:35 +08:00
|
|
|
Info.QualifierLoc = ReadNestedNameSpecifierLoc(F, Record, Idx);
|
2010-10-16 02:21:24 +08:00
|
|
|
unsigned NumTPLists = Record[Idx++];
|
|
|
|
Info.NumTemplParamLists = NumTPLists;
|
|
|
|
if (NumTPLists) {
|
2011-09-10 05:34:22 +08:00
|
|
|
Info.TemplParamLists = new (Context) TemplateParameterList*[NumTPLists];
|
2010-10-16 02:21:24 +08:00
|
|
|
for (unsigned i=0; i != NumTPLists; ++i)
|
|
|
|
Info.TemplParamLists[i] = ReadTemplateParameterList(F, Record, Idx);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-06-20 03:29:09 +08:00
|
|
|
TemplateName
|
2011-12-01 07:21:26 +08:00
|
|
|
ASTReader::ReadTemplateName(ModuleFile &F, const RecordData &Record,
|
2011-01-15 14:45:20 +08:00
|
|
|
unsigned &Idx) {
|
2010-10-21 11:16:25 +08:00
|
|
|
TemplateName::NameKind Kind = (TemplateName::NameKind)Record[Idx++];
|
2010-06-20 03:29:09 +08:00
|
|
|
switch (Kind) {
|
|
|
|
case TemplateName::Template:
|
2011-07-22 06:35:25 +08:00
|
|
|
return TemplateName(ReadDeclAs<TemplateDecl>(F, Record, Idx));
|
2010-06-20 03:29:09 +08:00
|
|
|
|
|
|
|
case TemplateName::OverloadedTemplate: {
|
|
|
|
unsigned size = Record[Idx++];
|
|
|
|
UnresolvedSet<8> Decls;
|
|
|
|
while (size--)
|
2011-07-22 06:35:25 +08:00
|
|
|
Decls.addDecl(ReadDeclAs<NamedDecl>(F, Record, Idx));
|
2010-06-20 03:29:09 +08:00
|
|
|
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.getOverloadedTemplateName(Decls.begin(), Decls.end());
|
2010-06-20 03:29:09 +08:00
|
|
|
}
|
2010-10-21 11:16:25 +08:00
|
|
|
|
2010-06-20 03:29:09 +08:00
|
|
|
case TemplateName::QualifiedTemplate: {
|
2011-07-22 06:35:25 +08:00
|
|
|
NestedNameSpecifier *NNS = ReadNestedNameSpecifier(F, Record, Idx);
|
2010-06-20 03:29:09 +08:00
|
|
|
bool hasTemplKeyword = Record[Idx++];
|
2011-07-22 06:35:25 +08:00
|
|
|
TemplateDecl *Template = ReadDeclAs<TemplateDecl>(F, Record, Idx);
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.getQualifiedTemplateName(NNS, hasTemplKeyword, Template);
|
2010-06-20 03:29:09 +08:00
|
|
|
}
|
2010-10-21 11:16:25 +08:00
|
|
|
|
2010-06-20 03:29:09 +08:00
|
|
|
case TemplateName::DependentTemplate: {
|
2011-07-22 06:35:25 +08:00
|
|
|
NestedNameSpecifier *NNS = ReadNestedNameSpecifier(F, Record, Idx);
|
2010-06-20 03:29:09 +08:00
|
|
|
if (Record[Idx++]) // isIdentifier
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.getDependentTemplateName(NNS,
|
2011-07-29 04:55:49 +08:00
|
|
|
GetIdentifierInfo(F, Record,
|
|
|
|
Idx));
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.getDependentTemplateName(NNS,
|
2010-06-28 17:31:42 +08:00
|
|
|
(OverloadedOperatorKind)Record[Idx++]);
|
2010-06-20 03:29:09 +08:00
|
|
|
}
|
2011-06-30 16:33:18 +08:00
|
|
|
|
|
|
|
case TemplateName::SubstTemplateTemplateParm: {
|
|
|
|
TemplateTemplateParmDecl *param
|
2011-07-22 06:35:25 +08:00
|
|
|
= ReadDeclAs<TemplateTemplateParmDecl>(F, Record, Idx);
|
2011-06-30 16:33:18 +08:00
|
|
|
if (!param) return TemplateName();
|
|
|
|
TemplateName replacement = ReadTemplateName(F, Record, Idx);
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.getSubstTemplateTemplateParm(param, replacement);
|
2011-06-30 16:33:18 +08:00
|
|
|
}
|
2011-01-15 14:45:20 +08:00
|
|
|
|
|
|
|
case TemplateName::SubstTemplateTemplateParmPack: {
|
|
|
|
TemplateTemplateParmDecl *Param
|
2011-07-22 06:35:25 +08:00
|
|
|
= ReadDeclAs<TemplateTemplateParmDecl>(F, Record, Idx);
|
2011-01-15 14:45:20 +08:00
|
|
|
if (!Param)
|
|
|
|
return TemplateName();
|
|
|
|
|
|
|
|
TemplateArgument ArgPack = ReadTemplateArgument(F, Record, Idx);
|
|
|
|
if (ArgPack.getKind() != TemplateArgument::Pack)
|
|
|
|
return TemplateName();
|
|
|
|
|
2011-09-10 05:34:22 +08:00
|
|
|
return Context.getSubstTemplateTemplateParmPack(Param, ArgPack);
|
2011-01-15 14:45:20 +08:00
|
|
|
}
|
2010-06-20 03:29:09 +08:00
|
|
|
}
|
2010-10-21 11:16:25 +08:00
|
|
|
|
2011-09-23 13:06:16 +08:00
|
|
|
llvm_unreachable("Unhandled template name kind!");
|
2010-06-20 03:29:09 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TemplateArgument
|
2011-12-01 07:21:26 +08:00
|
|
|
ASTReader::ReadTemplateArgument(ModuleFile &F,
|
2010-07-23 06:43:28 +08:00
|
|
|
const RecordData &Record, unsigned &Idx) {
|
2011-01-06 02:58:31 +08:00
|
|
|
TemplateArgument::ArgKind Kind = (TemplateArgument::ArgKind)Record[Idx++];
|
|
|
|
switch (Kind) {
|
2010-06-20 03:29:09 +08:00
|
|
|
case TemplateArgument::Null:
|
|
|
|
return TemplateArgument();
|
|
|
|
case TemplateArgument::Type:
|
2011-07-22 08:38:23 +08:00
|
|
|
return TemplateArgument(readType(F, Record, Idx));
|
2010-06-20 03:29:09 +08:00
|
|
|
case TemplateArgument::Declaration:
|
2011-07-22 06:35:25 +08:00
|
|
|
return TemplateArgument(ReadDecl(F, Record, Idx));
|
2010-06-28 17:31:34 +08:00
|
|
|
case TemplateArgument::Integral: {
|
|
|
|
llvm::APSInt Value = ReadAPSInt(Record, Idx);
|
2011-07-22 08:38:23 +08:00
|
|
|
QualType T = readType(F, Record, Idx);
|
2010-06-28 17:31:34 +08:00
|
|
|
return TemplateArgument(Value, T);
|
|
|
|
}
|
2011-01-06 02:58:31 +08:00
|
|
|
case TemplateArgument::Template:
|
2011-01-15 14:45:20 +08:00
|
|
|
return TemplateArgument(ReadTemplateName(F, Record, Idx));
|
2011-01-06 02:58:31 +08:00
|
|
|
case TemplateArgument::TemplateExpansion: {
|
2011-01-15 14:45:20 +08:00
|
|
|
TemplateName Name = ReadTemplateName(F, Record, Idx);
|
2011-01-15 07:41:42 +08:00
|
|
|
llvm::Optional<unsigned> NumTemplateExpansions;
|
|
|
|
if (unsigned NumExpansions = Record[Idx++])
|
|
|
|
NumTemplateExpansions = NumExpansions - 1;
|
|
|
|
return TemplateArgument(Name, NumTemplateExpansions);
|
2011-01-06 01:40:24 +08:00
|
|
|
}
|
2010-06-20 03:29:09 +08:00
|
|
|
case TemplateArgument::Expression:
|
2010-10-05 23:59:54 +08:00
|
|
|
return TemplateArgument(ReadExpr(F));
|
2010-06-20 03:29:09 +08:00
|
|
|
case TemplateArgument::Pack: {
|
|
|
|
unsigned NumArgs = Record[Idx++];
|
2011-09-10 05:34:22 +08:00
|
|
|
TemplateArgument *Args = new (Context) TemplateArgument[NumArgs];
|
2010-11-08 07:05:16 +08:00
|
|
|
for (unsigned I = 0; I != NumArgs; ++I)
|
|
|
|
Args[I] = ReadTemplateArgument(F, Record, Idx);
|
|
|
|
return TemplateArgument(Args, NumArgs);
|
2010-06-20 03:29:09 +08:00
|
|
|
}
|
|
|
|
}
|
2010-10-21 11:16:25 +08:00
|
|
|
|
2011-09-23 13:06:16 +08:00
|
|
|
llvm_unreachable("Unhandled template argument kind!");
|
2010-06-20 03:29:09 +08:00
|
|
|
}
|
|
|
|
|
2010-06-23 21:48:30 +08:00
|
|
|
TemplateParameterList *
|
2011-12-01 07:21:26 +08:00
|
|
|
ASTReader::ReadTemplateParameterList(ModuleFile &F,
|
2010-10-05 23:59:54 +08:00
|
|
|
const RecordData &Record, unsigned &Idx) {
|
|
|
|
SourceLocation TemplateLoc = ReadSourceLocation(F, Record, Idx);
|
|
|
|
SourceLocation LAngleLoc = ReadSourceLocation(F, Record, Idx);
|
|
|
|
SourceLocation RAngleLoc = ReadSourceLocation(F, Record, Idx);
|
2010-06-23 21:48:30 +08:00
|
|
|
|
|
|
|
unsigned NumParams = Record[Idx++];
|
2011-07-23 18:55:15 +08:00
|
|
|
SmallVector<NamedDecl *, 16> Params;
|
2010-06-23 21:48:30 +08:00
|
|
|
Params.reserve(NumParams);
|
|
|
|
while (NumParams--)
|
2011-07-22 06:35:25 +08:00
|
|
|
Params.push_back(ReadDeclAs<NamedDecl>(F, Record, Idx));
|
2010-10-21 11:16:25 +08:00
|
|
|
|
|
|
|
TemplateParameterList* TemplateParams =
|
2011-09-10 05:34:22 +08:00
|
|
|
TemplateParameterList::Create(Context, TemplateLoc, LAngleLoc,
|
2010-06-23 21:48:30 +08:00
|
|
|
Params.data(), Params.size(), RAngleLoc);
|
|
|
|
return TemplateParams;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2010-08-19 07:56:43 +08:00
|
|
|
ASTReader::
|
2011-07-23 18:55:15 +08:00
|
|
|
ReadTemplateArgumentList(SmallVector<TemplateArgument, 8> &TemplArgs,
|
2011-12-01 07:21:26 +08:00
|
|
|
ModuleFile &F, const RecordData &Record,
|
2010-10-05 23:59:54 +08:00
|
|
|
unsigned &Idx) {
|
2010-06-23 21:48:30 +08:00
|
|
|
unsigned NumTemplateArgs = Record[Idx++];
|
|
|
|
TemplArgs.reserve(NumTemplateArgs);
|
|
|
|
while (NumTemplateArgs--)
|
2010-10-05 23:59:54 +08:00
|
|
|
TemplArgs.push_back(ReadTemplateArgument(F, Record, Idx));
|
2010-06-23 21:48:30 +08:00
|
|
|
}
|
|
|
|
|
2010-07-02 19:55:32 +08:00
|
|
|
/// \brief Read a UnresolvedSet structure.
|
2011-12-01 07:21:26 +08:00
|
|
|
void ASTReader::ReadUnresolvedSet(ModuleFile &F, UnresolvedSetImpl &Set,
|
2010-07-02 19:55:32 +08:00
|
|
|
const RecordData &Record, unsigned &Idx) {
|
|
|
|
unsigned NumDecls = Record[Idx++];
|
|
|
|
while (NumDecls--) {
|
2011-07-22 06:35:25 +08:00
|
|
|
NamedDecl *D = ReadDeclAs<NamedDecl>(F, Record, Idx);
|
2010-07-02 19:55:32 +08:00
|
|
|
AccessSpecifier AS = (AccessSpecifier)Record[Idx++];
|
|
|
|
Set.addDecl(D, AS);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-07-03 07:30:27 +08:00
|
|
|
CXXBaseSpecifier
|
2011-12-01 07:21:26 +08:00
|
|
|
ASTReader::ReadCXXBaseSpecifier(ModuleFile &F,
|
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++]);
|
2011-02-06 03:23:19 +08:00
|
|
|
bool inheritConstructors = static_cast<bool>(Record[Idx++]);
|
2010-10-05 23:59:54 +08:00
|
|
|
TypeSourceInfo *TInfo = GetTypeSourceInfo(F, Record, Idx);
|
|
|
|
SourceRange Range = ReadSourceRange(F, Record, Idx);
|
2011-01-04 06:36:02 +08:00
|
|
|
SourceLocation EllipsisLoc = ReadSourceLocation(F, Record, Idx);
|
2011-02-06 03:23:19 +08:00
|
|
|
CXXBaseSpecifier Result(Range, isVirtual, isBaseOfClass, AS, TInfo,
|
2011-01-04 06:36:02 +08:00
|
|
|
EllipsisLoc);
|
2011-02-06 03:23:19 +08:00
|
|
|
Result.setInheritConstructors(inheritConstructors);
|
|
|
|
return Result;
|
2010-07-03 07:30:27 +08:00
|
|
|
}
|
|
|
|
|
2011-01-09 04:30:50 +08:00
|
|
|
std::pair<CXXCtorInitializer **, unsigned>
|
2011-12-01 07:21:26 +08:00
|
|
|
ASTReader::ReadCXXCtorInitializers(ModuleFile &F, const RecordData &Record,
|
2011-01-09 04:30:50 +08:00
|
|
|
unsigned &Idx) {
|
|
|
|
CXXCtorInitializer **CtorInitializers = 0;
|
2010-08-09 18:54:12 +08:00
|
|
|
unsigned NumInitializers = Record[Idx++];
|
|
|
|
if (NumInitializers) {
|
2011-01-09 04:30:50 +08:00
|
|
|
CtorInitializers
|
2011-09-10 05:34:22 +08:00
|
|
|
= new (Context) CXXCtorInitializer*[NumInitializers];
|
2010-08-09 18:54:12 +08:00
|
|
|
for (unsigned i=0; i != NumInitializers; ++i) {
|
2011-11-01 09:16:03 +08:00
|
|
|
TypeSourceInfo *TInfo = 0;
|
2010-08-09 18:54:12 +08:00
|
|
|
bool IsBaseVirtual = false;
|
|
|
|
FieldDecl *Member = 0;
|
2010-12-04 17:14:42 +08:00
|
|
|
IndirectFieldDecl *IndirectMember = 0;
|
2010-10-21 11:16:25 +08:00
|
|
|
|
2011-05-04 09:19:08 +08:00
|
|
|
CtorInitializerType Type = (CtorInitializerType)Record[Idx++];
|
|
|
|
switch (Type) {
|
2011-11-01 09:16:03 +08:00
|
|
|
case CTOR_INITIALIZER_BASE:
|
|
|
|
TInfo = GetTypeSourceInfo(F, Record, Idx);
|
2010-08-09 18:54:12 +08:00
|
|
|
IsBaseVirtual = Record[Idx++];
|
2011-05-04 09:19:08 +08:00
|
|
|
break;
|
2011-11-01 09:16:03 +08:00
|
|
|
|
|
|
|
case CTOR_INITIALIZER_DELEGATING:
|
|
|
|
TInfo = GetTypeSourceInfo(F, Record, Idx);
|
2011-05-04 09:19:08 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case CTOR_INITIALIZER_MEMBER:
|
2011-07-22 06:35:25 +08:00
|
|
|
Member = ReadDeclAs<FieldDecl>(F, Record, Idx);
|
2011-05-04 09:19:08 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case CTOR_INITIALIZER_INDIRECT_MEMBER:
|
2011-07-22 06:35:25 +08:00
|
|
|
IndirectMember = ReadDeclAs<IndirectFieldDecl>(F, Record, Idx);
|
2011-05-04 09:19:08 +08:00
|
|
|
break;
|
2010-08-09 18:54:12 +08:00
|
|
|
}
|
2011-05-04 09:19:08 +08:00
|
|
|
|
2011-01-04 08:32:56 +08:00
|
|
|
SourceLocation MemberOrEllipsisLoc = ReadSourceLocation(F, Record, Idx);
|
2010-10-05 23:59:54 +08:00
|
|
|
Expr *Init = ReadExpr(F);
|
|
|
|
SourceLocation LParenLoc = ReadSourceLocation(F, Record, Idx);
|
|
|
|
SourceLocation RParenLoc = ReadSourceLocation(F, Record, Idx);
|
2010-08-09 18:54:12 +08:00
|
|
|
bool IsWritten = Record[Idx++];
|
|
|
|
unsigned SourceOrderOrNumArrayIndices;
|
2011-07-23 18:55:15 +08:00
|
|
|
SmallVector<VarDecl *, 8> Indices;
|
2010-08-09 18:54:12 +08:00
|
|
|
if (IsWritten) {
|
|
|
|
SourceOrderOrNumArrayIndices = Record[Idx++];
|
|
|
|
} else {
|
|
|
|
SourceOrderOrNumArrayIndices = Record[Idx++];
|
|
|
|
Indices.reserve(SourceOrderOrNumArrayIndices);
|
|
|
|
for (unsigned i=0; i != SourceOrderOrNumArrayIndices; ++i)
|
2011-07-22 06:35:25 +08:00
|
|
|
Indices.push_back(ReadDeclAs<VarDecl>(F, Record, Idx));
|
2010-08-09 18:54:12 +08:00
|
|
|
}
|
2010-10-21 11:16:25 +08:00
|
|
|
|
2011-01-09 04:30:50 +08:00
|
|
|
CXXCtorInitializer *BOMInit;
|
2011-05-04 09:19:08 +08:00
|
|
|
if (Type == CTOR_INITIALIZER_BASE) {
|
2011-11-01 09:16:03 +08:00
|
|
|
BOMInit = new (Context) CXXCtorInitializer(Context, TInfo, IsBaseVirtual,
|
2011-01-09 04:30:50 +08:00
|
|
|
LParenLoc, Init, RParenLoc,
|
|
|
|
MemberOrEllipsisLoc);
|
2011-05-04 09:19:08 +08:00
|
|
|
} else if (Type == CTOR_INITIALIZER_DELEGATING) {
|
2011-11-01 09:16:03 +08:00
|
|
|
BOMInit = new (Context) CXXCtorInitializer(Context, TInfo, LParenLoc,
|
|
|
|
Init, RParenLoc);
|
2010-08-09 18:54:12 +08:00
|
|
|
} else if (IsWritten) {
|
2010-12-04 17:14:42 +08:00
|
|
|
if (Member)
|
2011-09-10 05:34:22 +08:00
|
|
|
BOMInit = new (Context) CXXCtorInitializer(Context, Member, MemberOrEllipsisLoc,
|
2011-01-09 04:30:50 +08:00
|
|
|
LParenLoc, Init, RParenLoc);
|
2010-12-04 17:14:42 +08:00
|
|
|
else
|
2011-09-10 05:34:22 +08:00
|
|
|
BOMInit = new (Context) CXXCtorInitializer(Context, IndirectMember,
|
2011-01-09 04:30:50 +08:00
|
|
|
MemberOrEllipsisLoc, LParenLoc,
|
|
|
|
Init, RParenLoc);
|
2010-08-09 18:54:12 +08:00
|
|
|
} else {
|
2011-09-10 05:34:22 +08:00
|
|
|
BOMInit = CXXCtorInitializer::Create(Context, Member, MemberOrEllipsisLoc,
|
2011-01-09 04:30:50 +08:00
|
|
|
LParenLoc, Init, RParenLoc,
|
|
|
|
Indices.data(), Indices.size());
|
2010-08-09 18:54:12 +08:00
|
|
|
}
|
|
|
|
|
2010-09-07 03:04:27 +08:00
|
|
|
if (IsWritten)
|
|
|
|
BOMInit->setSourceOrder(SourceOrderOrNumArrayIndices);
|
2011-01-09 04:30:50 +08:00
|
|
|
CtorInitializers[i] = BOMInit;
|
2010-08-09 18:54:12 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-01-09 04:30:50 +08:00
|
|
|
return std::make_pair(CtorInitializers, NumInitializers);
|
2010-08-09 18:54:12 +08:00
|
|
|
}
|
|
|
|
|
2010-05-08 05:43:38 +08:00
|
|
|
NestedNameSpecifier *
|
2011-12-01 07:21:26 +08:00
|
|
|
ASTReader::ReadNestedNameSpecifier(ModuleFile &F,
|
2011-07-22 06:35:25 +08:00
|
|
|
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: {
|
2011-07-29 04:55:49 +08:00
|
|
|
IdentifierInfo *II = GetIdentifierInfo(F, Record, Idx);
|
2011-09-10 05:34:22 +08:00
|
|
|
NNS = NestedNameSpecifier::Create(Context, Prev, II);
|
2010-05-08 05:43:38 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case NestedNameSpecifier::Namespace: {
|
2011-07-22 06:35:25 +08:00
|
|
|
NamespaceDecl *NS = ReadDeclAs<NamespaceDecl>(F, Record, Idx);
|
2011-09-10 05:34:22 +08:00
|
|
|
NNS = NestedNameSpecifier::Create(Context, Prev, NS);
|
2010-05-08 05:43:38 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2011-02-24 10:36:08 +08:00
|
|
|
case NestedNameSpecifier::NamespaceAlias: {
|
2011-07-22 06:35:25 +08:00
|
|
|
NamespaceAliasDecl *Alias =ReadDeclAs<NamespaceAliasDecl>(F, Record, Idx);
|
2011-09-10 05:34:22 +08:00
|
|
|
NNS = NestedNameSpecifier::Create(Context, Prev, Alias);
|
2011-02-24 10:36:08 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2010-05-08 05:43:38 +08:00
|
|
|
case NestedNameSpecifier::TypeSpec:
|
|
|
|
case NestedNameSpecifier::TypeSpecWithTemplate: {
|
2011-07-22 08:38:23 +08:00
|
|
|
const Type *T = readType(F, Record, Idx).getTypePtrOrNull();
|
2010-12-11 01:03:06 +08:00
|
|
|
if (!T)
|
|
|
|
return 0;
|
|
|
|
|
2010-05-08 05:43:38 +08:00
|
|
|
bool Template = Record[Idx++];
|
2011-09-10 05:34:22 +08:00
|
|
|
NNS = NestedNameSpecifier::Create(Context, Prev, Template, T);
|
2010-05-08 05:43:38 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case NestedNameSpecifier::Global: {
|
2011-09-10 05:34:22 +08:00
|
|
|
NNS = NestedNameSpecifier::GlobalSpecifier(Context);
|
2010-05-08 05:43:38 +08:00
|
|
|
// 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;
|
|
|
|
}
|
|
|
|
|
2011-02-25 08:36:19 +08:00
|
|
|
NestedNameSpecifierLoc
|
2011-12-01 07:21:26 +08:00
|
|
|
ASTReader::ReadNestedNameSpecifierLoc(ModuleFile &F, const RecordData &Record,
|
2011-02-25 08:36:19 +08:00
|
|
|
unsigned &Idx) {
|
|
|
|
unsigned N = Record[Idx++];
|
2011-03-01 07:58:31 +08:00
|
|
|
NestedNameSpecifierLocBuilder Builder;
|
2011-02-25 08:36:19 +08:00
|
|
|
for (unsigned I = 0; I != N; ++I) {
|
|
|
|
NestedNameSpecifier::SpecifierKind Kind
|
|
|
|
= (NestedNameSpecifier::SpecifierKind)Record[Idx++];
|
|
|
|
switch (Kind) {
|
|
|
|
case NestedNameSpecifier::Identifier: {
|
2011-07-29 04:55:49 +08:00
|
|
|
IdentifierInfo *II = GetIdentifierInfo(F, Record, Idx);
|
2011-02-25 08:36:19 +08:00
|
|
|
SourceRange Range = ReadSourceRange(F, Record, Idx);
|
2011-09-10 05:34:22 +08:00
|
|
|
Builder.Extend(Context, II, Range.getBegin(), Range.getEnd());
|
2011-02-25 08:36:19 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case NestedNameSpecifier::Namespace: {
|
2011-07-22 06:35:25 +08:00
|
|
|
NamespaceDecl *NS = ReadDeclAs<NamespaceDecl>(F, Record, Idx);
|
2011-02-25 08:36:19 +08:00
|
|
|
SourceRange Range = ReadSourceRange(F, Record, Idx);
|
2011-09-10 05:34:22 +08:00
|
|
|
Builder.Extend(Context, NS, Range.getBegin(), Range.getEnd());
|
2011-02-25 08:36:19 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case NestedNameSpecifier::NamespaceAlias: {
|
2011-07-22 06:35:25 +08:00
|
|
|
NamespaceAliasDecl *Alias =ReadDeclAs<NamespaceAliasDecl>(F, Record, Idx);
|
2011-02-25 08:36:19 +08:00
|
|
|
SourceRange Range = ReadSourceRange(F, Record, Idx);
|
2011-09-10 05:34:22 +08:00
|
|
|
Builder.Extend(Context, Alias, Range.getBegin(), Range.getEnd());
|
2011-02-25 08:36:19 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case NestedNameSpecifier::TypeSpec:
|
|
|
|
case NestedNameSpecifier::TypeSpecWithTemplate: {
|
|
|
|
bool Template = Record[Idx++];
|
|
|
|
TypeSourceInfo *T = GetTypeSourceInfo(F, Record, Idx);
|
|
|
|
if (!T)
|
|
|
|
return NestedNameSpecifierLoc();
|
|
|
|
SourceLocation ColonColonLoc = ReadSourceLocation(F, Record, Idx);
|
2011-03-01 07:58:31 +08:00
|
|
|
|
|
|
|
// FIXME: 'template' keyword location not saved anywhere, so we fake it.
|
2011-09-10 05:34:22 +08:00
|
|
|
Builder.Extend(Context,
|
2011-03-01 07:58:31 +08:00
|
|
|
Template? T->getTypeLoc().getBeginLoc() : SourceLocation(),
|
|
|
|
T->getTypeLoc(), ColonColonLoc);
|
2011-02-25 08:36:19 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case NestedNameSpecifier::Global: {
|
|
|
|
SourceLocation ColonColonLoc = ReadSourceLocation(F, Record, Idx);
|
2011-09-10 05:34:22 +08:00
|
|
|
Builder.MakeGlobal(Context, ColonColonLoc);
|
2011-02-25 08:36:19 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-10 05:34:22 +08:00
|
|
|
return Builder.getWithLocInContext(Context);
|
2011-02-25 08:36:19 +08:00
|
|
|
}
|
|
|
|
|
2010-05-08 05:43:38 +08:00
|
|
|
SourceRange
|
2011-12-01 07:21:26 +08:00
|
|
|
ASTReader::ReadSourceRange(ModuleFile &F, const RecordData &Record,
|
2010-10-05 23:59:54 +08:00
|
|
|
unsigned &Idx) {
|
|
|
|
SourceLocation beg = ReadSourceLocation(F, Record, Idx);
|
|
|
|
SourceLocation end = ReadSourceLocation(F, Record, Idx);
|
2010-06-02 23:47:10 +08:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
Implement a new 'availability' attribute, that allows one to specify
which versions of an OS provide a certain facility. For example,
void foo()
__attribute__((availability(macosx,introduced=10.2,deprecated=10.4,obsoleted=10.6)));
says that the function "foo" was introduced in 10.2, deprecated in
10.4, and completely obsoleted in 10.6. This attribute ties in with
the deployment targets (e.g., -mmacosx-version-min=10.1 specifies that
we want to deploy back to Mac OS X 10.1). There are several concrete
behaviors that this attribute enables, as illustrated with the
function foo() above:
- If we choose a deployment target >= Mac OS X 10.4, uses of "foo"
will result in a deprecation warning, as if we had placed
attribute((deprecated)) on it (but with a better diagnostic)
- If we choose a deployment target >= Mac OS X 10.6, uses of "foo"
will result in an "unavailable" warning (in C)/error (in C++), as
if we had placed attribute((unavailable)) on it
- If we choose a deployment target prior to 10.2, foo() is
weak-imported (if it is a kind of entity that can be weak
imported), as if we had placed the weak_import attribute on it.
Naturally, there can be multiple availability attributes on a
declaration, for different platforms; only the current platform
matters when checking availability attributes.
The only platforms this attribute currently works for are "ios" and
"macosx", since we already have -mxxxx-version-min flags for them and we
have experience there with macro tricks translating down to the
deprecated/unavailable/weak_import attributes. The end goal is to open
this up to other platforms, and even extension to other "platforms"
that are really libraries (say, through a #pragma clang
define_system), but that hasn't yet been designed and we may want to
shake out more issues with this narrower problem first.
Addresses <rdar://problem/6690412>.
As a drive-by bug-fix, if an entity is both deprecated and
unavailable, we only emit the "unavailable" diagnostic.
llvm-svn: 128127
2011-03-23 08:50:03 +08:00
|
|
|
VersionTuple ASTReader::ReadVersionTuple(const RecordData &Record,
|
|
|
|
unsigned &Idx) {
|
|
|
|
unsigned Major = Record[Idx++];
|
|
|
|
unsigned Minor = Record[Idx++];
|
|
|
|
unsigned Subminor = Record[Idx++];
|
|
|
|
if (Minor == 0)
|
|
|
|
return VersionTuple(Major);
|
|
|
|
if (Subminor == 0)
|
|
|
|
return VersionTuple(Major, Minor - 1);
|
|
|
|
return VersionTuple(Major, Minor - 1, Subminor - 1);
|
|
|
|
}
|
|
|
|
|
2011-12-01 07:21:26 +08:00
|
|
|
CXXTemporary *ASTReader::ReadCXXTemporary(ModuleFile &F,
|
2011-07-22 06:35:25 +08:00
|
|
|
const RecordData &Record,
|
2010-05-10 08:25:06 +08:00
|
|
|
unsigned &Idx) {
|
2011-07-22 06:35:25 +08:00
|
|
|
CXXDestructorDecl *Decl = ReadDeclAs<CXXDestructorDecl>(F, Record, Idx);
|
2011-09-10 05:34:22 +08:00
|
|
|
return CXXTemporary::Create(Context, Decl);
|
2010-05-10 08:25:06 +08:00
|
|
|
}
|
|
|
|
|
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) {
|
2010-11-19 04:06:41 +08:00
|
|
|
return Diags.Report(Loc, 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() {
|
2011-09-10 06:02:16 +08:00
|
|
|
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
|
|
|
|
2010-10-28 17:29:32 +08:00
|
|
|
void ASTReader::ClearSwitchCaseIDs() {
|
|
|
|
SwitchCaseStmts.clear();
|
|
|
|
}
|
|
|
|
|
2011-12-17 16:11:25 +08:00
|
|
|
void ASTReader::finishPendingActions() {
|
|
|
|
while (!PendingIdentifierInfos.empty() ||
|
|
|
|
!PendingPreviousDecls.empty() ||
|
Completely re-implement (de-)serialization of declaration
chains. The previous implementation relied heavily on the declaration
chain being stored as a (circular) linked list on disk, as it is in
memory. However, when deserializing from multiple modules, the
different chains could get mixed up, leading to broken declaration chains.
The new solution keeps track of the first and last declarations in the
chain for each module file. When we load a declaration, we search all
of the module files for redeclarations of that declaration, then
splice together all of the lists into a coherent whole (along with any
redeclarations that were actually parsed).
As a drive-by fix, (de-)serialize the redeclaration chains of
TypedefNameDecls, which had somehow gotten missed previously. Add a
test of this serialization.
This new scheme creates a redeclaration table that is fairly large in
the PCH file (on the order of 400k for Cocoa.h's 12MB PCH file). The
table is mmap'd in and searched via a binary search, but it's still
quite large. A future tweak will eliminate entries for declarations
that have no redeclarations anywhere, and should
drastically reduce the size of this table.
llvm-svn: 146841
2011-12-18 07:38:30 +08:00
|
|
|
!PendingDeclChains.empty() ||
|
2011-12-17 16:11:25 +08:00
|
|
|
!PendingChainedObjCCategories.empty()) {
|
|
|
|
|
|
|
|
// If any identifiers with corresponding top-level declarations have
|
|
|
|
// been loaded, load those declarations now.
|
|
|
|
while (!PendingIdentifierInfos.empty()) {
|
|
|
|
SetGloballyVisibleDecls(PendingIdentifierInfos.front().II,
|
|
|
|
PendingIdentifierInfos.front().DeclIDs, true);
|
|
|
|
PendingIdentifierInfos.pop_front();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ready to load previous declarations of Decls that were delayed.
|
|
|
|
while (!PendingPreviousDecls.empty()) {
|
|
|
|
loadAndAttachPreviousDecl(PendingPreviousDecls.front().first,
|
|
|
|
PendingPreviousDecls.front().second);
|
|
|
|
PendingPreviousDecls.pop_front();
|
|
|
|
}
|
|
|
|
|
Completely re-implement (de-)serialization of declaration
chains. The previous implementation relied heavily on the declaration
chain being stored as a (circular) linked list on disk, as it is in
memory. However, when deserializing from multiple modules, the
different chains could get mixed up, leading to broken declaration chains.
The new solution keeps track of the first and last declarations in the
chain for each module file. When we load a declaration, we search all
of the module files for redeclarations of that declaration, then
splice together all of the lists into a coherent whole (along with any
redeclarations that were actually parsed).
As a drive-by fix, (de-)serialize the redeclaration chains of
TypedefNameDecls, which had somehow gotten missed previously. Add a
test of this serialization.
This new scheme creates a redeclaration table that is fairly large in
the PCH file (on the order of 400k for Cocoa.h's 12MB PCH file). The
table is mmap'd in and searched via a binary search, but it's still
quite large. A future tweak will eliminate entries for declarations
that have no redeclarations anywhere, and should
drastically reduce the size of this table.
llvm-svn: 146841
2011-12-18 07:38:30 +08:00
|
|
|
// Load pending declaration chains.
|
|
|
|
for (unsigned I = 0; I != PendingDeclChains.size(); ++I) {
|
|
|
|
loadPendingDeclChain(PendingDeclChains[I]);
|
|
|
|
}
|
|
|
|
PendingDeclChains.clear();
|
|
|
|
|
2011-12-17 16:11:25 +08:00
|
|
|
for (std::vector<std::pair<ObjCInterfaceDecl *,
|
|
|
|
serialization::DeclID> >::iterator
|
|
|
|
I = PendingChainedObjCCategories.begin(),
|
|
|
|
E = PendingChainedObjCCategories.end(); I != E; ++I) {
|
|
|
|
loadObjCChainedCategories(I->second, I->first);
|
|
|
|
}
|
|
|
|
PendingChainedObjCCategories.clear();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
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) {
|
2011-12-17 12:13:28 +08:00
|
|
|
|
2011-12-17 16:11:25 +08:00
|
|
|
while (Consumer && !InterestingDecls.empty()) {
|
|
|
|
finishPendingActions();
|
|
|
|
|
2011-12-01 07:18:26 +08:00
|
|
|
// We are not in recursive loading, so it's safe to pass the "interesting"
|
|
|
|
// decls to the consumer.
|
2011-12-17 16:11:25 +08:00
|
|
|
Decl *D = InterestingDecls.front();
|
|
|
|
InterestingDecls.pop_front();
|
|
|
|
|
|
|
|
// Fully load the interesting decls, including deserializing their
|
|
|
|
// bodies, so that any other declarations that get referenced in the
|
|
|
|
// body will be fully deserialized by the time we pass them to the
|
|
|
|
// consumer.
|
|
|
|
if (FunctionDecl *FD = dyn_cast<FunctionDecl>(D)) {
|
|
|
|
if (FD->doesThisDeclarationHaveABody()) {
|
|
|
|
FD->getBody();
|
|
|
|
finishPendingActions();
|
|
|
|
}
|
2011-12-01 07:18:26 +08:00
|
|
|
}
|
2011-11-14 15:07:59 +08:00
|
|
|
|
2011-12-17 16:11:25 +08:00
|
|
|
PassInterestingDeclToConsumer(D);
|
|
|
|
}
|
|
|
|
|
|
|
|
finishPendingActions();
|
Completely re-implement (de-)serialization of declaration
chains. The previous implementation relied heavily on the declaration
chain being stored as a (circular) linked list on disk, as it is in
memory. However, when deserializing from multiple modules, the
different chains could get mixed up, leading to broken declaration chains.
The new solution keeps track of the first and last declarations in the
chain for each module file. When we load a declaration, we search all
of the module files for redeclarations of that declaration, then
splice together all of the lists into a coherent whole (along with any
redeclarations that were actually parsed).
As a drive-by fix, (de-)serialize the redeclaration chains of
TypedefNameDecls, which had somehow gotten missed previously. Add a
test of this serialization.
This new scheme creates a redeclaration table that is fairly large in
the PCH file (on the order of 400k for Cocoa.h's 12MB PCH file). The
table is mmap'd in and searched via a binary search, but it's still
quite large. A future tweak will eliminate entries for declarations
that have no redeclarations anywhere, and should
drastically reduce the size of this table.
llvm-svn: 146841
2011-12-18 07:38:30 +08:00
|
|
|
PendingDeclChainsKnown.clear();
|
2010-10-25 01:26:31 +08:00
|
|
|
|
|
|
|
assert(PendingForwardRefs.size() == 0 &&
|
|
|
|
"Some forward refs did not get linked to the definition!");
|
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
|
|
|
|
2011-09-02 08:26:20 +08:00
|
|
|
ASTReader::ASTReader(Preprocessor &PP, ASTContext &Context,
|
2011-07-23 00:35:34 +08:00
|
|
|
StringRef isysroot, bool DisableValidation,
|
2011-02-06 03:42:43 +08:00
|
|
|
bool DisableStatCache)
|
2010-08-24 08:50:04 +08:00
|
|
|
: Listener(new PCHValidator(PP, *this)), DeserializationListener(0),
|
|
|
|
SourceMgr(PP.getSourceManager()), FileMgr(PP.getFileManager()),
|
2011-09-10 06:02:16 +08:00
|
|
|
Diags(PP.getDiagnostics()), SemaObj(0), PP(PP), Context(Context),
|
2011-07-29 01:20:23 +08:00
|
|
|
Consumer(0), ModuleMgr(FileMgr.getFileSystemOptions()),
|
|
|
|
RelocatablePCH(false), isysroot(isysroot),
|
Revamp the SourceManager to separate the representation of parsed
source locations from source locations loaded from an AST/PCH file.
Previously, loading an AST/PCH file involved carefully pre-allocating
space at the beginning of the source manager for the source locations
and FileIDs that correspond to the prefix, and then appending the
source locations/FileIDs used for parsing the remaining translation
unit. This design forced us into loading PCH files early, as a prefix,
whic has become a rather significant limitation.
This patch splits the SourceManager space into two parts: for source
location "addresses", the lower values (growing upward) are used to
describe parsed code, while upper values (growing downward) are used
for source locations loaded from AST/PCH files. Similarly, positive
FileIDs are used to describe parsed code while negative FileIDs are
used to file/macro locations loaded from AST/PCH files. As a result,
we can load PCH/AST files even during parsing, making various
improvemnts in the future possible, e.g., teaching #include <foo.h> to
look for and load <foo.h.gch> if it happens to be already available.
This patch was originally written by Sebastian Redl, then brought
forward to the modern age by Jonathan Turner, and finally
polished/finished by me to be committed.
llvm-svn: 135484
2011-07-20 00:10:42 +08:00
|
|
|
DisableValidation(DisableValidation),
|
2011-02-06 03:42:43 +08:00
|
|
|
DisableStatCache(DisableStatCache), NumStatHits(0), NumStatMisses(0),
|
Revamp the SourceManager to separate the representation of parsed
source locations from source locations loaded from an AST/PCH file.
Previously, loading an AST/PCH file involved carefully pre-allocating
space at the beginning of the source manager for the source locations
and FileIDs that correspond to the prefix, and then appending the
source locations/FileIDs used for parsing the remaining translation
unit. This design forced us into loading PCH files early, as a prefix,
whic has become a rather significant limitation.
This patch splits the SourceManager space into two parts: for source
location "addresses", the lower values (growing upward) are used to
describe parsed code, while upper values (growing downward) are used
for source locations loaded from AST/PCH files. Similarly, positive
FileIDs are used to describe parsed code while negative FileIDs are
used to file/macro locations loaded from AST/PCH files. As a result,
we can load PCH/AST files even during parsing, making various
improvemnts in the future possible, e.g., teaching #include <foo.h> to
look for and load <foo.h.gch> if it happens to be already available.
This patch was originally written by Sebastian Redl, then brought
forward to the modern age by Jonathan Turner, and finally
polished/finished by me to be committed.
llvm-svn: 135484
2011-07-20 00:10:42 +08:00
|
|
|
NumSLocEntriesRead(0), TotalNumSLocEntries(0),
|
2011-02-06 03:42:43 +08:00
|
|
|
NumStatementsRead(0), TotalNumStatements(0), NumMacrosRead(0),
|
|
|
|
TotalNumMacros(0), NumSelectorsRead(0), NumMethodPoolEntriesRead(0),
|
|
|
|
NumMethodPoolMisses(0), TotalNumMethodPoolEntries(0),
|
|
|
|
NumLexicalDeclContextsRead(0), TotalLexicalDeclContexts(0),
|
2011-07-22 05:15:19 +08:00
|
|
|
NumVisibleDeclContextsRead(0), TotalVisibleDeclContexts(0),
|
|
|
|
TotalModulesSizeInBits(0), NumCurrentElementsDeserializing(0),
|
|
|
|
NumCXXBaseSpecifiersLoaded(0)
|
2011-02-06 03:42:43 +08:00
|
|
|
{
|
Revamp the SourceManager to separate the representation of parsed
source locations from source locations loaded from an AST/PCH file.
Previously, loading an AST/PCH file involved carefully pre-allocating
space at the beginning of the source manager for the source locations
and FileIDs that correspond to the prefix, and then appending the
source locations/FileIDs used for parsing the remaining translation
unit. This design forced us into loading PCH files early, as a prefix,
whic has become a rather significant limitation.
This patch splits the SourceManager space into two parts: for source
location "addresses", the lower values (growing upward) are used to
describe parsed code, while upper values (growing downward) are used
for source locations loaded from AST/PCH files. Similarly, positive
FileIDs are used to describe parsed code while negative FileIDs are
used to file/macro locations loaded from AST/PCH files. As a result,
we can load PCH/AST files even during parsing, making various
improvemnts in the future possible, e.g., teaching #include <foo.h> to
look for and load <foo.h.gch> if it happens to be already available.
This patch was originally written by Sebastian Redl, then brought
forward to the modern age by Jonathan Turner, and finally
polished/finished by me to be committed.
llvm-svn: 135484
2011-07-20 00:10:42 +08:00
|
|
|
SourceMgr.setExternalSLocEntrySource(this);
|
2010-08-24 08:50:04 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
ASTReader::~ASTReader() {
|
|
|
|
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)
|
2011-08-03 23:48:04 +08:00
|
|
|
delete static_cast<ASTDeclContextNameLookupTable*>(J->first);
|
2010-08-24 08:50:04 +08:00
|
|
|
}
|
|
|
|
}
|