From: Paul Robinson Date: Thu, 21 Dec 2017 21:47:22 +0000 (+0000) Subject: [AST] Incorrectly qualified unscoped enumeration as template actual parameter. X-Git-Url: https://granicus.if.org/sourcecode?a=commitdiff_plain;h=2f5231150cfd6f2da1e23aa0a2880ad732998903;p=clang [AST] Incorrectly qualified unscoped enumeration as template actual parameter. An unscoped enumeration used as template argument, should not have any qualified information about its enclosing scope, as its visibility is global. In the case of scoped enumerations, they must include information about their enclosing scope. Patch by Carlos Alberto Enciso! Differential Revision: https://reviews.llvm.org/D39239 git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@321312 91177308-0d34-0410-b5e6-96231b3b80d8 --- diff --git a/lib/AST/Decl.cpp b/lib/AST/Decl.cpp index 2f51ec31a7..ee15a4d2b4 100644 --- a/lib/AST/Decl.cpp +++ b/lib/AST/Decl.cpp @@ -1548,7 +1548,10 @@ void NamedDecl::printQualifiedName(raw_ostream &OS, // enumerator is declared in the scope that immediately contains // the enum-specifier. Each scoped enumerator is declared in the // scope of the enumeration. - if (ED->isScoped() || ED->getIdentifier()) + // For the case of unscoped enumerator, do not include in the qualified + // name any information about its enum enclosing scope, as is visibility + // is global. + if (ED->isScoped()) OS << *ED; else continue; diff --git a/test/Modules/odr.cpp b/test/Modules/odr.cpp index 9cdbb4f080..c4700110af 100644 --- a/test/Modules/odr.cpp +++ b/test/Modules/odr.cpp @@ -18,6 +18,6 @@ int x = f() + g(); // expected-note@a.h:3 {{declaration of 'f' does not match}} // expected-note@a.h:1 {{definition has no member 'm'}} -// expected-error@b.h:5 {{'E::e2' from module 'b' is not present in definition of 'E' in module 'a'}} +// expected-error@b.h:5 {{'e2' from module 'b' is not present in definition of 'E' in module 'a'}} // expected-error@b.h:3 {{'Y::f' from module 'b' is not present in definition of 'Y' in module 'a'}} // expected-error@b.h:2 {{'Y::m' from module 'b' is not present in definition of 'Y' in module 'a'}} diff --git a/test/SemaCXX/return-noreturn.cpp b/test/SemaCXX/return-noreturn.cpp index cdd96e6a9e..cfeddffb66 100644 --- a/test/SemaCXX/return-noreturn.cpp +++ b/test/SemaCXX/return-noreturn.cpp @@ -143,7 +143,7 @@ template int PR9412_t() { } // expected-warning {{control reaches end of non-void function}} void PR9412_f() { - PR9412_t(); // expected-note {{in instantiation of function template specialization 'PR9412_t' requested here}} + PR9412_t(); // expected-note {{in instantiation of function template specialization 'PR9412_t' requested here}} } struct NoReturn { diff --git a/test/SemaTemplate/temp_arg_enum_printing.cpp b/test/SemaTemplate/temp_arg_enum_printing.cpp index bdf277d308..dbb4db81f6 100644 --- a/test/SemaTemplate/temp_arg_enum_printing.cpp +++ b/test/SemaTemplate/temp_arg_enum_printing.cpp @@ -13,9 +13,9 @@ template void foo(); void test() { - // CHECK: template<> void foo() + // CHECK: template<> void foo() NamedEnumNS::foo(); - // CHECK: template<> void foo() + // CHECK: template<> void foo() NamedEnumNS::foo<(NamedEnum)1>(); // CHECK: template<> void foo<2>() NamedEnumNS::foo<(NamedEnum)2>(); diff --git a/test/SemaTemplate/temp_arg_enum_printing_more.cpp b/test/SemaTemplate/temp_arg_enum_printing_more.cpp new file mode 100644 index 0000000000..a3a7158412 --- /dev/null +++ b/test/SemaTemplate/temp_arg_enum_printing_more.cpp @@ -0,0 +1,26 @@ +// RUN: %clang_cc1 -fsyntax-only -ast-print %s -std=c++11 | FileCheck %s + +// Make sure that for template value arguments that are unscoped enumerators, +// no qualified enum information is included in their name, as their visibility +// is global. In the case of scoped enumerators, they must include information +// about their enum enclosing scope. + +enum E1 { e1 }; +template struct tmpl_1 {}; +// CHECK: template<> struct tmpl_1 +tmpl_1 TMPL_1; // Name must be 'e1'. + +namespace nsp_1 { enum E2 { e2 }; } +template struct tmpl_2 {}; +// CHECK: template<> struct tmpl_2 +tmpl_2 TMPL_2; // Name must be 'nsp_1::e2'. + +enum class E3 { e3 }; +template struct tmpl_3 {}; +// CHECK: template<> struct tmpl_3 +tmpl_3 TMPL_3; // Name must be 'E3::e3'. + +namespace nsp_2 { enum class E4 { e4 }; } +template struct tmpl_4 {}; +// CHECK: template<> struct tmpl_4 +tmpl_4 TMPL_4; // Name must be 'nsp_2::E4::e4'. diff --git a/unittests/AST/NamedDeclPrinterTest.cpp b/unittests/AST/NamedDeclPrinterTest.cpp index 92df4577ac..002bb28f37 100644 --- a/unittests/AST/NamedDeclPrinterTest.cpp +++ b/unittests/AST/NamedDeclPrinterTest.cpp @@ -143,7 +143,7 @@ TEST(NamedDeclPrinter, TestNamedEnum) { ASSERT_TRUE(PrintedWrittenNamedDeclCXX11Matches( "enum X { A };", "A", - "X::A")); + "A")); } TEST(NamedDeclPrinter, TestScopedNamedEnum) { @@ -164,7 +164,7 @@ TEST(NamedDeclPrinter, TestClassWithUnscopedNamedEnum) { ASSERT_TRUE(PrintedWrittenNamedDeclCXX11Matches( "class X { enum Y { A }; };", "A", - "X::Y::A")); + "X::A")); } TEST(NamedDeclPrinter, TestClassWithScopedNamedEnum) {