Store the type ID for __builtin_va_list in the PCH file, so that the

AST context's __builtin_va_list type will be set when the PCH file is
loaded. This fixes the crash when CodeGen'ing a va_arg expression
pulled in from a PCH file.



git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@69421 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
Douglas Gregor 2009-04-18 05:55:16 +00:00
Родитель d92fa477d5
Коммит ad1de006ea
5 изменённых файлов: 41 добавлений и 5 удалений

Просмотреть файл

@ -142,9 +142,18 @@ namespace clang {
/// program (e.g., for code generation). /// program (e.g., for code generation).
EXTERNAL_DEFINITIONS = 7, EXTERNAL_DEFINITIONS = 7,
/// \brief Record code for the set of non-builtin, special
/// types.
///
/// This record contains the type IDs for the various type nodes
/// that are constructed during semantic analysis (e.g.,
/// __builtin_va_list). The SPECIAL_TYPE_* constants provide
/// offsets into this record.
SPECIAL_TYPES = 8,
/// \brief Record code for the block of extra statistics we /// \brief Record code for the block of extra statistics we
/// gather while generating a PCH file. /// gather while generating a PCH file.
STATISTICS = 8 STATISTICS = 9
}; };
/// \brief Record types used within a source manager block. /// \brief Record types used within a source manager block.
@ -314,6 +323,16 @@ namespace clang {
TYPE_OBJC_QUALIFIED_CLASS = 24 TYPE_OBJC_QUALIFIED_CLASS = 24
}; };
/// \brief The type IDs for special types constructed by semantic
/// analysis.
///
/// The constants in this enumeration are indices into the
/// SPECIAL_TYPES record.
enum SpecialTypeIDs {
/// \brief __builtin_va_list
SPECIAL_TYPE_BUILTIN_VA_LIST = 0
};
/// \brief Record codes for each kind of declaration. /// \brief Record codes for each kind of declaration.
/// ///
/// These constants describe the records that can occur within a /// These constants describe the records that can occur within a

Просмотреть файл

@ -157,6 +157,9 @@ private:
/// in the PCH file. /// in the PCH file.
unsigned TotalNumStatements; unsigned TotalNumStatements;
/// \brief
llvm::SmallVector<uint64_t, 4> SpecialTypes;
PCHReadResult ReadPCHBlock(); PCHReadResult ReadPCHBlock();
bool CheckPredefinesBuffer(const char *PCHPredef, bool CheckPredefinesBuffer(const char *PCHPredef,
unsigned PCHPredefLen, unsigned PCHPredefLen,

Просмотреть файл

@ -1316,6 +1316,10 @@ PCHReader::PCHReadResult PCHReader::ReadPCHBlock() {
ExternalDefinitions.swap(Record); ExternalDefinitions.swap(Record);
break; break;
case pch::SPECIAL_TYPES:
SpecialTypes.swap(Record);
break;
case pch::STATISTICS: case pch::STATISTICS:
TotalNumStatements = Record[0]; TotalNumStatements = Record[0];
break; break;
@ -1399,6 +1403,10 @@ PCHReader::PCHReadResult PCHReader::ReadPCH(const std::string &FileName) {
// Load the translation unit declaration // Load the translation unit declaration
ReadDeclRecord(DeclOffsets[0], 0); ReadDeclRecord(DeclOffsets[0], 0);
// Load the special types.
Context.setBuiltinVaListType(
GetType(SpecialTypes[pch::SPECIAL_TYPE_BUILTIN_VA_LIST]));
return Success; return Success;
} }

Просмотреть файл

@ -1739,6 +1739,7 @@ void PCHWriter::WritePCH(ASTContext &Context, const Preprocessor &PP) {
DeclsToEmit.push(Context.getTranslationUnitDecl()); DeclsToEmit.push(Context.getTranslationUnitDecl());
// Write the remaining PCH contents. // Write the remaining PCH contents.
RecordData Record;
Stream.EnterSubblock(pch::PCH_BLOCK_ID, 3); Stream.EnterSubblock(pch::PCH_BLOCK_ID, 3);
WriteTargetTriple(Context.Target); WriteTargetTriple(Context.Target);
WriteLanguageOptions(Context.getLangOptions()); WriteLanguageOptions(Context.getLangOptions());
@ -1749,11 +1750,17 @@ void PCHWriter::WritePCH(ASTContext &Context, const Preprocessor &PP) {
WriteIdentifierTable(); WriteIdentifierTable();
Stream.EmitRecord(pch::TYPE_OFFSET, TypeOffsets); Stream.EmitRecord(pch::TYPE_OFFSET, TypeOffsets);
Stream.EmitRecord(pch::DECL_OFFSET, DeclOffsets); Stream.EmitRecord(pch::DECL_OFFSET, DeclOffsets);
// Write the record of special types.
Record.clear();
AddTypeRef(Context.getBuiltinVaListType(), Record);
Stream.EmitRecord(pch::SPECIAL_TYPES, Record);
if (!ExternalDefinitions.empty()) if (!ExternalDefinitions.empty())
Stream.EmitRecord(pch::EXTERNAL_DEFINITIONS, ExternalDefinitions); Stream.EmitRecord(pch::EXTERNAL_DEFINITIONS, ExternalDefinitions);
// Some simple statistics // Some simple statistics
RecordData Record; Record.clear();
Record.push_back(NumStatements); Record.push_back(NumStatements);
Stream.EmitRecord(pch::STATISTICS, Record); Stream.EmitRecord(pch::STATISTICS, Record);
Stream.ExitBlock(); Stream.ExitBlock();

Просмотреть файл

@ -1,11 +1,10 @@
// Test this without pch. // Test this without pch.
// RUN: clang-cc -triple=x86_64-unknown-freebsd7.0 -include %S/va_arg.h %s // RUN: clang-cc -triple=x86_64-unknown-freebsd7.0 -include %S/va_arg.h %s -emit-llvm -o -
// Test with pch. // Test with pch.
// RUN: clang-cc -triple=x86_64-unknown-freebsd7.0 -o %t %S/va_arg.h && // RUN: clang-cc -triple=x86_64-unknown-freebsd7.0 -o %t %S/va_arg.h &&
// RUN: clang-cc -triple=x86_64-unknown-freebsd7.0 -include-pch %t %s // RUN: clang-cc -triple=x86_64-unknown-freebsd7.0 -include-pch %t %s -emit-llvm -o -
// FIXME: Crash when emitting LLVM bitcode using PCH!
char *g0(char** argv, int argc) { return argv[argc]; } char *g0(char** argv, int argc) { return argv[argc]; }
char *g(char **argv) { char *g(char **argv) {