Skip to content

Commit a5e6590

Browse files
author
Gabor Marton
committed
[ASTImporter] Support CXXDeductionGuideDecl with local typedef
CXXDeductionGuideDecl with a local typedef has its own copy of the TypedefDecl with the CXXDeductionGuideDecl as the DeclContext of that TypedefDecl. ``` template <typename T> struct A { typedef T U; A(U, T); }; A a{(int)0, (int)0}; ``` Related discussion on cfe-dev: http://lists.llvm.org/pipermail/cfe-dev/2020-November/067252.html Without this fix, when we import the CXXDeductionGuideDecl (via VisitFunctionDecl) then before creating the Decl we must import the FunctionType. However, the first parameter's type is the afore mentioned local typedef. So, we then start importing the TypedefDecl whose DeclContext is the CXXDeductionGuideDecl itself. The infinite loop is formed. ``` #0 clang::ASTNodeImporter::VisitCXXDeductionGuideDecl(clang::CXXDeductionGuideDecl*) clang/lib/AST/ASTImporter.cpp:3543:0 #1 clang::declvisitor::Base<std::add_pointer, clang::ASTNodeImporter, llvm::Expected<clang::Decl*> >::Visit(clang::Decl*) /home/egbomrt/WORK/llvm5/build/debug/tools/clang/include/clang/AST/DeclNodes.inc:405:0 #2 clang::ASTImporter::ImportImpl(clang::Decl*) clang/lib/AST/ASTImporter.cpp:8038:0 #3 clang::ASTImporter::Import(clang::Decl*) clang/lib/AST/ASTImporter.cpp:8200:0 #4 clang::ASTImporter::ImportContext(clang::DeclContext*) clang/lib/AST/ASTImporter.cpp:8297:0 #5 clang::ASTNodeImporter::ImportDeclContext(clang::Decl*, clang::DeclContext*&, clang::DeclContext*&) clang/lib/AST/ASTImporter.cpp:1852:0 #6 clang::ASTNodeImporter::ImportDeclParts(clang::NamedDecl*, clang::DeclContext*&, clang::DeclContext*&, clang::DeclarationName&, clang::NamedDecl*&, clang::SourceLocation&) clang/lib/AST/ASTImporter.cpp:1628:0 #7 clang::ASTNodeImporter::VisitTypedefNameDecl(clang::TypedefNameDecl*, bool) clang/lib/AST/ASTImporter.cpp:2419:0 #8 clang::ASTNodeImporter::VisitTypedefDecl(clang::TypedefDecl*) clang/lib/AST/ASTImporter.cpp:2500:0 #9 clang::declvisitor::Base<std::add_pointer, clang::ASTNodeImporter, llvm::Expected<clang::Decl*> >::Visit(clang::Decl*) /home/egbomrt/WORK/llvm5/build/debug/tools/clang/include/clang/AST/DeclNodes.inc:315:0 #10 clang::ASTImporter::ImportImpl(clang::Decl*) clang/lib/AST/ASTImporter.cpp:8038:0 #11 clang::ASTImporter::Import(clang::Decl*) clang/lib/AST/ASTImporter.cpp:8200:0 #12 llvm::Expected<clang::TypedefNameDecl*> clang::ASTNodeImporter::import<clang::TypedefNameDecl>(clang::TypedefNameDecl*) clang/lib/AST/ASTImporter.cpp:165:0 #13 clang::ASTNodeImporter::VisitTypedefType(clang::TypedefType const*) clang/lib/AST/ASTImporter.cpp:1304:0 #14 clang::TypeVisitor<clang::ASTNodeImporter, llvm::Expected<clang::QualType> >::Visit(clang::Type const*) /home/egbomrt/WORK/llvm5/build/debug/tools/clang/include/clang/AST/TypeNodes.inc:74:0 #15 clang::ASTImporter::Import(clang::QualType) clang/lib/AST/ASTImporter.cpp:8071:0 #16 llvm::Expected<clang::QualType> clang::ASTNodeImporter::import<clang::QualType>(clang::QualType const&) clang/lib/AST/ASTImporter.cpp:179:0 #17 clang::ASTNodeImporter::VisitFunctionProtoType(clang::FunctionProtoType const*) clang/lib/AST/ASTImporter.cpp:1244:0 #18 clang::TypeVisitor<clang::ASTNodeImporter, llvm::Expected<clang::QualType> >::Visit(clang::Type const*) /home/egbomrt/WORK/llvm5/build/debug/tools/clang/include/clang/AST/TypeNodes.inc:47:0 #19 clang::ASTImporter::Import(clang::QualType) clang/lib/AST/ASTImporter.cpp:8071:0 #20 llvm::Expected<clang::QualType> clang::ASTNodeImporter::import<clang::QualType>(clang::QualType const&) clang/lib/AST/ASTImporter.cpp:179:0 #21 clang::QualType clang::ASTNodeImporter::importChecked<clang::QualType>(llvm::Error&, clang::QualType const&) clang/lib/AST/ASTImporter.cpp:198:0 #22 clang::ASTNodeImporter::VisitFunctionDecl(clang::FunctionDecl*) clang/lib/AST/ASTImporter.cpp:3313:0 #23 clang::ASTNodeImporter::VisitCXXDeductionGuideDecl(clang::CXXDeductionGuideDecl*) clang/lib/AST/ASTImporter.cpp:3543:0 ``` The fix is to first create the TypedefDecl and only then start to import the DeclContext. Basically, we could do this during the import of all other Decls (not just for typedefs). But it seems, there is only one another AST construct that has a similar cycle: a struct defined as a function parameter: ``` int struct_in_proto(struct data_t{int a;int b;} *d); ``` In that case, however, we had decided to return simply with an error back then because that seemed to be a very rare construct. Differential Revision: https://reviews.llvm.org/D92209
1 parent 4ae8651 commit a5e6590

File tree

2 files changed

+60
-5
lines changed

2 files changed

+60
-5
lines changed

clang/lib/AST/ASTImporter.cpp

+44-5
Original file line numberDiff line numberDiff line change
@@ -388,6 +388,8 @@ namespace clang {
388388
ExpectedType VisitObjCObjectPointerType(const ObjCObjectPointerType *T);
389389

390390
// Importing declarations
391+
Error ImportDeclParts(NamedDecl *D, DeclarationName &Name, NamedDecl *&ToD,
392+
SourceLocation &Loc);
391393
Error ImportDeclParts(
392394
NamedDecl *D, DeclContext *&DC, DeclContext *&LexicalDC,
393395
DeclarationName &Name, NamedDecl *&ToD, SourceLocation &Loc);
@@ -1647,6 +1649,25 @@ Error ASTNodeImporter::ImportDeclParts(
16471649
return Error::success();
16481650
}
16491651

1652+
Error ASTNodeImporter::ImportDeclParts(NamedDecl *D, DeclarationName &Name,
1653+
NamedDecl *&ToD, SourceLocation &Loc) {
1654+
1655+
// Import the name of this declaration.
1656+
if (Error Err = importInto(Name, D->getDeclName()))
1657+
return Err;
1658+
1659+
// Import the location of this declaration.
1660+
if (Error Err = importInto(Loc, D->getLocation()))
1661+
return Err;
1662+
1663+
ToD = cast_or_null<NamedDecl>(Importer.GetAlreadyImportedOrNull(D));
1664+
if (ToD)
1665+
if (Error Err = ASTNodeImporter(*this).ImportDefinitionIfNeeded(D, ToD))
1666+
return Err;
1667+
1668+
return Error::success();
1669+
}
1670+
16501671
Error ASTNodeImporter::ImportDefinitionIfNeeded(Decl *FromD, Decl *ToD) {
16511672
if (!FromD)
16521673
return Error::success();
@@ -2415,22 +2436,29 @@ ExpectedDecl ASTNodeImporter::VisitNamespaceAliasDecl(NamespaceAliasDecl *D) {
24152436
ExpectedDecl
24162437
ASTNodeImporter::VisitTypedefNameDecl(TypedefNameDecl *D, bool IsAlias) {
24172438
// Import the major distinguishing characteristics of this typedef.
2418-
DeclContext *DC, *LexicalDC;
24192439
DeclarationName Name;
24202440
SourceLocation Loc;
24212441
NamedDecl *ToD;
2422-
if (Error Err = ImportDeclParts(D, DC, LexicalDC, Name, ToD, Loc))
2442+
// Do not import the DeclContext, we will import it once the TypedefNameDecl
2443+
// is created.
2444+
if (Error Err = ImportDeclParts(D, Name, ToD, Loc))
24232445
return std::move(Err);
24242446
if (ToD)
24252447
return ToD;
24262448

2449+
DeclContext *DC = cast_or_null<DeclContext>(
2450+
Importer.GetAlreadyImportedOrNull(cast<Decl>(D->getDeclContext())));
2451+
DeclContext *LexicalDC =
2452+
cast_or_null<DeclContext>(Importer.GetAlreadyImportedOrNull(
2453+
cast<Decl>(D->getLexicalDeclContext())));
2454+
24272455
// If this typedef is not in block scope, determine whether we've
24282456
// seen a typedef with the same name (that we can merge with) or any
24292457
// other entity by that name (which name lookup could conflict with).
24302458
// Note: Repeated typedefs are not valid in C99:
24312459
// 'typedef int T; typedef int T;' is invalid
24322460
// We do not care about this now.
2433-
if (!DC->isFunctionOrMethod()) {
2461+
if (DC && !DC->isFunctionOrMethod()) {
24342462
SmallVector<NamedDecl *, 4> ConflictingDecls;
24352463
unsigned IDNS = Decl::IDNS_Ordinary;
24362464
auto FoundDecls = Importer.findDeclsInToCtx(DC, Name);
@@ -2487,8 +2515,15 @@ ASTNodeImporter::VisitTypedefNameDecl(TypedefNameDecl *D, bool IsAlias) {
24872515
Name.getAsIdentifierInfo(), ToTypeSourceInfo))
24882516
return ToTypedef;
24892517

2490-
ToTypedef->setAccess(D->getAccess());
2518+
// Import the DeclContext and set it to the Typedef.
2519+
if ((Err = ImportDeclContext(D, DC, LexicalDC)))
2520+
return std::move(Err);
2521+
ToTypedef->setDeclContext(DC);
24912522
ToTypedef->setLexicalDeclContext(LexicalDC);
2523+
// Add to the lookupTable because we could not do that in MapImported.
2524+
Importer.AddToLookupTable(ToTypedef);
2525+
2526+
ToTypedef->setAccess(D->getAccess());
24922527

24932528
// Templated declarations should not appear in DeclContext.
24942529
TypeAliasDecl *FromAlias = IsAlias ? cast<TypeAliasDecl>(D) : nullptr;
@@ -9198,7 +9233,11 @@ Decl *ASTImporter::MapImported(Decl *From, Decl *To) {
91989233
// This mapping should be maintained only in this function. Therefore do not
91999234
// check for additional consistency.
92009235
ImportedFromDecls[To] = From;
9201-
AddToLookupTable(To);
9236+
// In the case of TypedefNameDecl we create the Decl first and only then we
9237+
// import and set its DeclContext. So, the DC is still not set when we reach
9238+
// here from GetImportedOrCreateDecl.
9239+
if (To->getDeclContext())
9240+
AddToLookupTable(To);
92029241
return To;
92039242
}
92049243

clang/unittests/AST/ASTImporterTest.cpp

+16
Original file line numberDiff line numberDiff line change
@@ -5998,6 +5998,22 @@ TEST_P(ImportFunctions, CTADUserDefinedExplicit) {
59985998
EXPECT_TRUE(ToD->isExplicit());
59995999
}
60006000

6001+
TEST_P(ImportFunctions, CTADWithLocalTypedef) {
6002+
Decl *TU = getTuDecl(
6003+
R"(
6004+
template <typename T> struct A {
6005+
typedef T U;
6006+
A(U);
6007+
};
6008+
A a{(int)0};
6009+
)",
6010+
Lang_CXX17, "input.cc");
6011+
auto *FromD = FirstDeclMatcher<CXXDeductionGuideDecl>().match(
6012+
TU, cxxDeductionGuideDecl());
6013+
auto *ToD = Import(FromD, Lang_CXX17);
6014+
ASSERT_TRUE(ToD);
6015+
}
6016+
60016017
// FIXME Move these tests out of ASTImporterTest. For that we need to factor
60026018
// out the ASTImporter specific pars from ASTImporterOptionSpecificTestBase
60036019
// into a new test Fixture. Then we should lift up this Fixture to its own

0 commit comments

Comments
 (0)