Skip to content
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions clang/include/clang/Serialization/ASTWriter.h
Original file line number Diff line number Diff line change
Expand Up @@ -224,6 +224,7 @@ class ASTWriter : public ASTDeserializationListener,
/// discovery) and start at 2. 1 is reserved for the translation
/// unit, while 0 is reserved for NULL.
llvm::DenseMap<const Decl *, LocalDeclID> DeclIDs;
// TMP: ^ DeclIDs type for reference
Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I will remove all TMP comments before committing. Using this to simplify the review by pulling in the pointers to the relevant code.


/// Set of predefined decls. This is a helper data to determine if a decl
/// is predefined. It should be more clear and safer to query the set
Expand Down
11 changes: 8 additions & 3 deletions clang/lib/Serialization/ASTWriter.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -3925,6 +3925,7 @@ class ASTIdentifierTableTrait {
// Only emit declarations that aren't from a chained PCH, though.
SmallVector<NamedDecl *, 16> Decls(IdResolver->decls(II));
for (NamedDecl *D : llvm::reverse(Decls))
// TMP: corresponding `getDeclForLocalLookup` call
LE.write<DeclID>((DeclID)Writer.getDeclID(
getDeclForLocalLookup(PP.getLangOpts(), D)));
}
Expand Down Expand Up @@ -3969,6 +3970,7 @@ void ASTWriter::WriteIdentifierTable(Preprocessor &PP,
if (isLocalIdentifierID(ID) || II->hasChangedSinceDeserialization() ||
(Trait.needDecls() &&
II->hasFETokenInfoChangedSinceDeserialization()))
// TMP: ^ corresponding `hasFETokenInfoChangedSinceDeserialization` call
Generator.insert(II, ID, Trait);
}

Expand Down Expand Up @@ -5664,14 +5666,16 @@ void ASTWriter::PrepareWritingSpecialDecls(Sema &SemaRef) {
llvm::SmallVector<const IdentifierInfo*, 256> IIs;
for (const auto &ID : SemaRef.PP.getIdentifierTable()) {
const IdentifierInfo *II = ID.second;
if (!Chain || !II->isFromAST() || II->hasChangedSinceDeserialization())
if (!Chain || !II->isFromAST() || II->hasChangedSinceDeserialization() ||
II->hasFETokenInfoChangedSinceDeserialization())
IIs.push_back(II);
}
// Sort the identifiers to visit based on their name.
llvm::sort(IIs, llvm::deref<std::less<>>());
const LangOptions &LangOpts = getLangOpts();
for (const IdentifierInfo *II : IIs)
for (const Decl *D : SemaRef.IdResolver.decls(II))
GetDeclRef(D);
for (NamedDecl *D : SemaRef.IdResolver.decls(II))
GetDeclRef(getDeclForLocalLookup(LangOpts, D));
}

// Write all of the DeclsToCheckForDeferredDiags.
Expand Down Expand Up @@ -6845,6 +6849,7 @@ LocalDeclID ASTWriter::GetDeclRef(const Decl *D) {
}

assert(!(reinterpret_cast<uintptr_t>(D) & 0x01) && "Invalid decl pointer");
// TMP: adding D to DeclIDs
LocalDeclID &ID = DeclIDs[D];
if (ID.isInvalid()) {
if (DoneWritingDeclsAndTypes) {
Expand Down
100 changes: 100 additions & 0 deletions clang/test/Modules/non-modular-decl-use.c
Original file line number Diff line number Diff line change
@@ -0,0 +1,100 @@
// RUN: rm -rf %t
// RUN: split-file %s %t
// RUN: %clang_cc1 -fsyntax-only -I %t/include %t/test.c \
// RUN: -fmodules -fimplicit-module-maps -fmodules-cache-path=%t/modules.cache

// Test when a decl is present in multiple modules through an inclusion of
// a non-modular header. Make sure such decl is serialized correctly and can be
// used after deserialization.

//--- include/non-modular.h
#ifndef NON_MODULAR_H
#define NON_MODULAR_H

union TestUnion {
int x;
float y;
};

struct ReferenceUnion1 {
union TestUnion name;
unsigned versionMajor;
};
struct ReferenceUnion2 {
union TestUnion name;
unsigned versionMinor;
};

// Test another kind of RecordDecl.
struct TestStruct {
int p;
float q;
};

struct ReferenceStruct1 {
unsigned fieldA;
struct TestStruct fieldB;
};

struct ReferenceStruct2 {
unsigned field1;
struct TestStruct field2;
};

#endif

//--- include/piecewise1-empty.h
//--- include/piecewise1-initially-hidden.h
#include <non-modular.h>

//--- include/piecewise2-empty.h
//--- include/piecewise2-initially-hidden.h
#include <non-modular.h>

//--- include/with-multiple-decls.h
#include <piecewise1-empty.h>
// Include the non-modular header and resolve a name duplication between decl
// in non-modular.h and in piecewise1-initially-hidden.h, create a
// redeclaration chain.
#include <non-modular.h>
// Include a decl with a duplicate name again to add more to IdentifierResolver.
#include <piecewise2-empty.h>

//--- include/module.modulemap
module Piecewise1 {
module Empty {
header "piecewise1-empty.h"
}
module InitiallyHidden {
header "piecewise1-initially-hidden.h"
export *
}
}

module Piecewise2 {
module Empty {
header "piecewise2-empty.h"
}
module InitiallyHidden {
header "piecewise2-initially-hidden.h"
export *
}
}

module WithMultipleDecls {
header "with-multiple-decls.h"
export *
}

//--- test.c
#include <with-multiple-decls.h>

struct Test {
int x;
union TestUnion name;
};

struct Test2 {
struct TestStruct name;
float y;
};
Loading