The tests Modules/cxx-templates.cpp are hitting Modules/template-default-args.cpp are currently failing with the
assert from D84827 to prevent duplicate entries in the lookup results.
The specific problem seems to be that we have duplicate entries in the lookup results when instantiating a templated class
with a (non-scoped) enum declared inside of it. The EnumConstantDecl inside the EnumDecl ends up being twice
in the lookup result of the surrounding class.
The unintentional addition to the lookup result happens with the following backtrace:
frame #1: clang`clang::DeclContext::makeDeclVisibleInContextImpl(this=, D=, Internal=false) at DeclBase.cpp:1940:21 frame #2: clang`clang::DeclContext::buildLookupImpl(this=, DCtx=, Internal=false) at DeclBase.cpp:1647:9 frame #3: clang`clang::DeclContext::buildLookupImpl(this=, DCtx=0x000000012180f368, Internal=false) at DeclBase.cpp:1654:9 frame #4: clang`clang::DeclContext::buildLookup(this=) at DeclBase.cpp:1621:5 frame #5: clang`clang::DeclContext::lookup(this=, Name="E" const at DeclBase.cpp:1713:43 frame #6: clang`clang::ASTReader::CompleteRedeclChain(this=, D=) at ASTReader.cpp:7177:13 frame #7: clang`clang::LazyGenerationalUpdatePtr<clang::Decl const*, clang::Decl*, &(clang::ExternalASTSource::CompleteRedeclChain(clang::Decl const*))>::get(this=, O=) at ExternalASTSource.h:443:9 frame #8: clang`clang::Redeclarable<clang::TagDecl>::DeclLink::getPrevious(this=, D=) const at Redeclarable.h:134:62 frame #9: clang`clang::Redeclarable<clang::TagDecl>::getNextRedeclaration(this=) const at Redeclarable.h:190:23 frame #10: clang`clang::Redeclarable<clang::TagDecl>::redecl_iterator::operator++(this=) at Redeclarable.h:271:34 frame #11: clang`clang::TagDecl::getDefinition(this=) const at Decl.cpp:4210:15 frame #12: clang`clang::DeclContext::getPrimaryContext(this=) at DeclBase.cpp:1266:31 frame #13: clang`clang::DeclContext::addDecl(this=, D=) at DeclBase.cpp:1577:27 frame #14: clang`clang::TemplateDeclInstantiator::InstantiateEnumDefinition(this=, Enum=, Pattern=) at SemaTemplateInstantiateDecl.cpp:1311:13
Essentially what happens here are the following steps which all come from the addDecl function:
void DeclContext::addDecl(Decl *D) { addHiddenDecl(D); //<- step 1 if (auto *ND = dyn_cast<NamedDecl>(D)) ND->getDeclContext()->getPrimaryContext()->makeDeclVisibleInContextWithFlags(ND, false, true); // call to `getPrimaryContext()` and `makeDeclVisibleInContextWithFlags` }
- We add the EnumConstantDecl as hidden to the EnumDecl (as preparation for making it visible) in addDecl.
- We try to make the EnumConstantDecl visible now in the second part of addDecl.
- For this we calculate the PrimaryContext of the EnumDecl via getPrimaryContext()
- For the PrimaryContext we first try to find the Definition of the EnumDecl (as for every TagDecl).
- To find the definition, we walk the redecl chain of the EnumDecl.
- To walk the redecl chain, we do a lookup of the enum name in its containing DeclContext.
- This will cause us to build the lookup for the EnumDecls' DeclContext which is the template class instantiation.
- As the EnumDecl is a transparent DeclContext, building the lookup for its parent DeclContext will also build the lookup for our EnumDecl.
- DeclContext::buildLookupImpl for the template class instantiation will find the EnumConstantDecl added in step 1 and make it visible.
- Now we actually return from the getPrimaryContext() call from step 3 and call makeDeclVisibleInContextWithFlags on it to make our EnumConstantDecl visible.
As we already made it visible in Step 9, Step 10 just explicitly adds the EnumConstantDecl a second time to the lookup.
This patch moves the call to getPrimaryContext() up so that it is before we add the decl to the DeclContext. This way
when building the lookup we can never unintentionally add the decl we are about to add to any lookup.