Fix unloading from inline namespaces other than std.

This commit is contained in:
Frederich Munch 2017-02-08 19:32:39 -05:00 committed by sftnight
parent 87c97bab0d
commit 185b905525
2 changed files with 21 additions and 16 deletions

View File

@ -737,23 +737,17 @@ bool DeclUnloader::VisitRedeclarable(clang::Redeclarable<T>* R, DeclContext* DC)
bool DeclUnloader::VisitNamespaceDecl(NamespaceDecl* NSD) {
// NamespaceDecl: NamedDecl, DeclContext, Redeclarable
// When unloading a NamedDecl within an inline namespace within std the
// NamedDecl needs to be removed from the parent std as well.
// Not sure why, but test/CodeUnloading/NameSpaces.C demonstrates
// that this only affects the std namespace
StoredDeclsMap *removeFromSTD = nullptr;
// When unloading a NamedDecl within an inline namespace within the
// NamedDecl needs to be removed from the parent well.
StoredDeclsMap *removeFromParent = nullptr;
if (NSD->isInline()) {
if (NamespaceDecl *parent = dyn_cast<NamespaceDecl>(NSD->getParent())) {
if (parent->getNameAsString()=="std") {
if ((parent = m_Sema->getStdNamespace()))
removeFromSTD = m_Sema->getStdNamespace()->getLookupPtr();
}
}
if (NamespaceDecl *parent = dyn_cast<NamespaceDecl>(NSD->getParent()))
removeFromParent = parent->getFirstDecl()->getLookupPtr();
}
bool Successful = VisitRedeclarable(NSD, NSD->getDeclContext());
// Inlined version of VisitDeclContext so we can check against removeFromSTD
// Inlined version of VisitDeclContext so we can check against parent
llvm::SmallVector<Decl*, 64> declsToErase;
for (Decl *D : NSD->noload_decls())
declsToErase.push_back(D);
@ -762,11 +756,11 @@ bool DeclUnloader::VisitRedeclarable(clang::Redeclarable<T>* R, DeclContext* DC)
Successful = Visit(*I) & Successful;
assert(Successful);
if (removeFromSTD) {
if (removeFromParent) {
if (NamedDecl *ND = dyn_cast<NamedDecl>(*I)) {
eraseDeclFromMap(removeFromSTD, ND);
eraseDeclFromMap(removeFromParent, ND);
#ifndef NDEBUG
checkDeclIsGone(removeFromSTD, ND);
checkDeclIsGone(removeFromParent, ND);
#endif
}
}

View File

@ -24,6 +24,18 @@
#undef TEST_NAMESPACE
namespace A{}
namespace A { inline namespace __BBB { int f; } }
namespace A { inline namespace __BBB { int f1; } }
namespace A { inline namespace __BBB { int f2; } }
.undo
A::f2 // expected-error {{no member named 'f2' in namespace 'A'}}
.undo
A::f1 // expected-error {{no member named 'f1' in namespace 'A'}}
.undo
A::f // expected-error {{no member named 'f' in namespace 'A'}}
#include <stdexcept>
.undo
#include <stdexcept>
@ -32,5 +44,4 @@
101
// CHECK: (int) 101
// expected-no-diagnostics
.q