[clang] [clang][NFC] Refactor expected directives in C++ DRs 400-499 (PR #74311)
Vlad Serebrennikov via cfe-commits
cfe-commits at lists.llvm.org
Mon Dec 4 05:20:25 PST 2023
https://github.com/Endilll updated https://github.com/llvm/llvm-project/pull/74311
>From fc5d07e9917e84b7927b4d3f6ea5058b008f120f Mon Sep 17 00:00:00 2001
From: Vlad Serebrennikov <serebrennikov.vladislav at gmail.com>
Date: Mon, 4 Dec 2023 15:15:08 +0300
Subject: [PATCH 1/3] [clang][NFC] Refactor expected directives in C++ DRs
400-499
This patch continues the work started with ea5b1ef016d020c37f903d6c7d4f623be975dab8. See that commit and its corresponding PR for details.
---
clang/test/CXX/drs/dr4xx.cpp | 659 ++++++++++++++++++++++-------------
1 file changed, 421 insertions(+), 238 deletions(-)
diff --git a/clang/test/CXX/drs/dr4xx.cpp b/clang/test/CXX/drs/dr4xx.cpp
index acb3cbfa52e7f..b7df9decdd4b8 100644
--- a/clang/test/CXX/drs/dr4xx.cpp
+++ b/clang/test/CXX/drs/dr4xx.cpp
@@ -1,9 +1,9 @@
-// RUN: env ASAN_OPTIONS=detect_stack_use_after_return=0 %clang_cc1 -std=c++98 %s -verify -fexceptions -fcxx-exceptions -pedantic-errors
-// RUN: env ASAN_OPTIONS=detect_stack_use_after_return=0 %clang_cc1 -std=c++11 %s -verify -fexceptions -fcxx-exceptions -pedantic-errors
-// RUN: env ASAN_OPTIONS=detect_stack_use_after_return=0 %clang_cc1 -std=c++14 %s -verify -fexceptions -fcxx-exceptions -pedantic-errors
-// RUN: env ASAN_OPTIONS=detect_stack_use_after_return=0 %clang_cc1 -std=c++17 %s -verify -fexceptions -fcxx-exceptions -pedantic-errors
-// RUN: env ASAN_OPTIONS=detect_stack_use_after_return=0 %clang_cc1 -std=c++20 %s -verify -fexceptions -fcxx-exceptions -pedantic-errors
-// RUN: env ASAN_OPTIONS=detect_stack_use_after_return=0 %clang_cc1 -std=c++23 %s -verify -fexceptions -fcxx-exceptions -pedantic-errors
+// RUN: env ASAN_OPTIONS=detect_stack_use_after_return=0 %clang_cc1 -std=c++98 %s -verify=expected,cxx98-14,cxx98-17,cxx98 -fexceptions -fcxx-exceptions -pedantic-errors
+// RUN: env ASAN_OPTIONS=detect_stack_use_after_return=0 %clang_cc1 -std=c++11 %s -verify=expected,cxx98-14,cxx98-17,since-cxx11 -fexceptions -fcxx-exceptions -pedantic-errors
+// RUN: env ASAN_OPTIONS=detect_stack_use_after_return=0 %clang_cc1 -std=c++14 %s -verify=expected,cxx98-14,cxx98-17,since-cxx11 -fexceptions -fcxx-exceptions -pedantic-errors
+// RUN: env ASAN_OPTIONS=detect_stack_use_after_return=0 %clang_cc1 -std=c++17 %s -verify=expected,since-cxx17,cxx98-17,since-cxx11 -fexceptions -fcxx-exceptions -pedantic-errors
+// RUN: env ASAN_OPTIONS=detect_stack_use_after_return=0 %clang_cc1 -std=c++20 %s -verify=expected,since-cxx17,since-cxx11 -fexceptions -fcxx-exceptions -pedantic-errors
+// RUN: env ASAN_OPTIONS=detect_stack_use_after_return=0 %clang_cc1 -std=c++23 %s -verify=expected,since-cxx17,since-cxx11 -fexceptions -fcxx-exceptions -pedantic-errors
// FIXME: __SIZE_TYPE__ expands to 'long long' on some targets.
__extension__ typedef __SIZE_TYPE__ size_t;
@@ -11,36 +11,55 @@ __extension__ typedef __SIZE_TYPE__ size_t;
namespace std { struct type_info; }
namespace dr400 { // dr400: yes
- struct A { int a; struct a {}; }; // expected-note 2{{conflicting}} expected-note {{ambiguous}}
- struct B { int a; struct a {}; }; // expected-note 2{{target}} expected-note {{ambiguous}}
+ struct A { int a; struct a {}; }; // #dr400-A
+ struct B { int a; struct a {}; }; // #dr400-B
struct C : A, B { using A::a; struct a b; };
- struct D : A, B { using A::a; using B::a; struct a b; }; // expected-error 2{{conflicts}}
- struct E : A, B { struct a b; }; // expected-error {{found in multiple base classes}}
+ struct D : A, B {
+ using A::a;
+ // FIXME: we should issue a single diagnostic
+ using B::a; // #dr400-using-B-a
+ // expected-error@#dr400-using-B-a {{target of using declaration conflicts with declaration already in scope}}
+ // expected-note@#dr400-B {{target of using declaration}}
+ // expected-note@#dr400-A {{conflicting declaration}}
+ // expected-error@#dr400-using-B-a {{target of using declaration conflicts with declaration already in scope}}
+ // expected-note@#dr400-B {{target of using declaration}}
+ // expected-note@#dr400-A {{conflicting declaration}}
+ struct a b;
+ };
+ struct E : A, B { struct a b; };
+ // expected-error at -1 {{member 'a' found in multiple base classes of different types}}
+ // expected-note@#dr400-A {{member type 'dr400::A::a' found by ambiguous name lookup}}
+ // expected-note@#dr400-B {{member type 'dr400::B::a' found by ambiguous name lookup}}
}
namespace dr401 { // dr401: 2.8
- template<class T, class U = typename T::type> class A : public T {}; // expected-error {{protected}} expected-error 2{{private}}
-
+ template<class T, class U = typename T::type> class A : public T {}; // #dr401-A
+ // expected-error@#dr401-A {{'type' is a private member of 'dr401::C'}}
+ // expected-note@#dr402-friend-A-C {{in instantiation of default argument for 'A<C>' required here}}
+ // expected-note@#dr402-C-type {{implicitly declared private here}}
+ // expected-error@#dr401-A {{'type' is a protected member of 'dr401::B'}}
+ // expected-note@#dr402-b {{in instantiation of default argument for 'A<B>' required here}}
+ // expected-note@#dr402-B-type {{declared protected here}}
+ // expected-error@#dr401-A {{'type' is a private member of 'dr401::D'}}
+ // expected-note@#dr402-d {{in instantiation of default argument for 'A<D>' required here}}
+ // expected-note@#dr402-D-type {{implicitly declared private here}}
class B {
protected:
- typedef int type; // expected-note {{protected}}
-#if __cplusplus == 199711L
- // expected-note at -2 {{protected}}
-#endif
+ typedef int type; // #dr402-B-type
};
class C {
- typedef int type; // expected-note {{private}}
- friend class A<C>; // expected-note {{default argument}}
+ typedef int type; // #dr402-C-type
+ friend class A<C>; // #dr402-friend-A-C
};
class D {
- typedef int type; // expected-note {{private}}
+ typedef int type; // #dr402-D-type
friend class A<D, int>;
};
- A<B> *b; // expected-note {{default argument}}
- A<D> *d; // expected-note {{in instantiation of default argument}}
+ A<B> *b; // #dr402-b
+ A<D> *d; // #dr402-d
struct E {
template<class T, class U = typename T::type> class A : public T {};
@@ -53,14 +72,15 @@ namespace dr401 { // dr401: 2.8
// FIXME: Why do we get different diagnostics in C++11 onwards here? We seem
// to not treat the default template argument as a SFINAE context in C++98.
- template<class T, class U = typename T::type> void f(T) {}
- void g(B b) { f(b); }
-#if __cplusplus < 201103L
- // expected-error at -3 0-1{{extension}} expected-error at -3 {{protected}} expected-note at -3 {{instantiation}}
- // expected-note at -3 {{substituting}}
-#else
- // expected-error at -5 {{no matching}} expected-note at -6 {{protected}}
-#endif
+ template<class T, class U = typename T::type> void f(T) {} // #dr402-f
+ // cxx98-error at -1 {{default template arguments for a function template are a C++11 extension}}
+ // cxx98-error at -2 {{'type' is a protected member of 'dr401::B'}}
+ // cxx98-note at -3 {{in instantiation of default argument for 'f<B>' required here}}
+ // cxx98-note@#dr402-f-b {{while substituting deduced template arguments into function template 'f' [with T = B, U = (no value)]}}
+ // cxx98-note@#dr402-B-type {{declared protected here}}
+ void g(B b) { f(b); } // #dr402-f-b
+ // since-cxx11-error at -1 {{no matching function for call to 'f'}}
+ // since-cxx11-note@#dr402-f {{candidate template ignored: substitution failure [with T = B, U = typename B::type]: 'type' is a protected member of 'dr401::B'}}
}
namespace dr403 { // dr403: yes
@@ -93,33 +113,45 @@ namespace dr405 { // dr405: yes
struct C {
int f;
- void test1(A::S as) { f(as); } // expected-error {{called object type 'int'}}
- void test2(A::S as) { void f(); f(as); } // expected-error {{too many arguments}} expected-note {{}}
+ void test1(A::S as) { f(as); }
+ // expected-error at -1 {{called object type 'int' is not a function or function pointer}}
+ void test2(A::S as) { void f(); f(as); }
+ // expected-error at -1 {{too many arguments to function call, expected 0, have 1}}
+ // expected-note at -2 {{'f' declared here}}
void test3(A::S as) { using A::f; f(as); } // ok
void test4(A::S as) { using B::f; f(as); } // ok
- void test5(A::S as) { int f; f(as); } // expected-error {{called object type 'int'}}
- void test6(A::S as) { struct f {}; (void) f(as); } // expected-error {{no matching conversion}} expected-note +{{}}
+ void test5(A::S as) { int f; f(as); }
+ // expected-error at -1 {{called object type 'int' is not a function or function pointer}}
+ void test6(A::S as) { struct f {}; (void) f(as); }
+ // expected-error at -1 {{no matching conversion for functional-style cast from 'A::S' to 'f'}}
+ // expected-note at -2 {{candidate constructor (the implicit copy constructor) not viable: no known conversion from 'A::S' to 'const f' for 1st argument}}
+ // since-cxx11-note at -3 {{candidate constructor (the implicit move constructor) not viable: no known conversion from 'A::S' to 'f' for 1st argument}}
+ // expected-note at -4 {{candidate constructor (the implicit default constructor) not viable: requires 0 arguments, but 1 was provided}}
};
namespace D {
struct S {};
struct X { void operator()(S); } f;
}
- void testD(D::S ds) { f(ds); } // expected-error {{undeclared identifier}}
+ void testD(D::S ds) { f(ds); }
+ // expected-error at -1 {{use of undeclared identifier 'f'}}
namespace E {
struct S {};
struct f { f(S); };
}
- void testE(E::S es) { f(es); } // expected-error {{undeclared identifier}}
+ void testE(E::S es) { f(es); }
+ // expected-error at -1 {{use of undeclared identifier 'f'}}
}
namespace dr406 { // dr406: 2.9
typedef struct {
- static int n; // expected-error {{static data member 'n' not allowed in anonymous struct}}
+ static int n;
+ // expected-error at -1 {{static data member 'n' not allowed in anonymous struct}}
} A;
typedef union {
- static int n; // expected-error {{static data member 'n' not allowed in anonymous union}}
+ static int n;
+ // expected-error at -1 {{static data member 'n' not allowed in anonymous union}}
} B;
}
@@ -130,23 +162,28 @@ namespace dr407 { // dr407: 3.8
void f() {
struct S *p;
{
- typedef struct S S; // expected-note {{here}}
- struct S *p; // expected-error {{typedef 'S' cannot be referenced with a struct specifier}}
+ typedef struct S S; // #dr407-typedef-S
+ struct S *p;
+ // expected-error at -1 {{typedef 'S' cannot be referenced with a struct specifier}}
+ // expected-note@#dr407-typedef-S {{declared here}}
}
}
struct S {};
namespace UsingDir {
namespace A {
- struct S {}; // expected-note {{found}}
+ struct S {}; // #dr407-A-S
}
namespace B {
- typedef int S; // expected-note {{found}}
+ typedef int S; // #dr407-B-S
}
namespace C {
using namespace A;
using namespace B;
- struct S s; // expected-error {{ambiguous}}
+ struct S s;
+ // expected-error at -1 {{ambiguous}}
+ // expected-note@#dr407-A-S {{candidate found by name lookup is 'dr407::UsingDir::A::S'}}
+ // expected-note@#dr407-B-S {{candidate found by name lookup is 'dr407::UsingDir::B::S'}}
}
namespace D {
using A::S;
@@ -214,9 +251,7 @@ namespace dr409 { // dr409: yes
A::B b2;
A<T>::B b3;
A<T*>::B b4;
-#if __cplusplus <= 201703L
- // expected-error at -2 {{implicit 'typename' is a C++20 extension}}
-#endif
+ // cxx98-17-error at -1 {{missing 'typename' prior to dependent type name A<T *>::B; implicit 'typename' is a C++20 extension}}
};
}
@@ -233,7 +268,7 @@ namespace dr410 { // dr410: no
template<class T> void i(T);
friend void i<>(int);
private:
- static void z(); // expected-note {{private}}
+ static void z(); // #dr410-z
};
template<> void h(int) { A::z(); }
@@ -242,7 +277,9 @@ namespace dr410 { // dr410: no
template<> void i(int) { A::z(); }
}
template<> void f(int) { M::A::z(); }
- void g(int) { M::A::z(); } // expected-error {{private}}
+ void g(int) { M::A::z(); }
+ // expected-error at -1 {{'z' is a private member of 'dr410::M::A'}}
+ // expected-note@#dr410-z {{declared private here}}
}
// dr412 is in its own file.
@@ -253,16 +290,19 @@ namespace dr413 { // dr413: yes
int : 17;
int b;
};
- S s = { 1, 2, 3 }; // expected-error {{excess elements}}
+ S s = { 1, 2, 3 };
+ // expected-error at -1 {{excess elements in struct initializer}}
struct E {};
- struct T { // expected-note {{here}}
+ struct T { // #dr413-T
int a;
E e;
int b;
};
T t1 = { 1, {}, 2 };
- T t2 = { 1, 2 }; // expected-error {{aggregate with no elements requires explicit braces}}
+ T t2 = { 1, 2 };
+ // expected-error at -1 {{initializer for aggregate with no elements requires explicit braces}}
+ // expected-note@#dr413-T {{'dr413::T' declared here}}
}
namespace dr414 { // dr414: dup 305
@@ -290,10 +330,12 @@ namespace dr416 { // dr416: yes
namespace dr417 { // dr417: no
struct A;
- struct dr417::A {}; // expected-warning {{extra qualification}}
+ struct dr417::A {};
+ // expected-warning at -1 {{extra qualification on member 'A'}}
struct B { struct X; };
struct C : B {};
- struct C::X {}; // expected-error {{no struct named 'X' in 'dr417::C'}}
+ struct C::X {};
+ // expected-error at -1 {{no struct named 'X' in 'dr417::C'}}
struct B::X { struct Y; };
struct C::X::Y {}; // ok!
namespace N {
@@ -304,15 +346,20 @@ namespace dr417 { // dr417: no
}
// FIXME: This is ill-formed.
using N::D;
- struct dr417::D {}; // expected-warning {{extra qualification}}
+ struct dr417::D {};
+ // expected-warning at -1 {{extra qualification on member 'D'}}
using namespace N;
- struct dr417::E {}; // expected-warning {{extra qualification}} expected-error {{no struct named 'E'}}
+ struct dr417::E {};
+ // expected-error at -1 {{no struct named 'E' in namespace 'dr417'}}
+ // expected-warning at -2 {{extra qualification on member 'E'}}
struct N::F {};
struct G;
using N::H;
namespace M {
- struct dr417::G {}; // expected-error {{namespace 'M' does not enclose}}
- struct dr417::H {}; // expected-error {{namespace 'M' does not enclose}}
+ struct dr417::G {};
+ // expected-error at -1 {{cannot define or redeclare 'G' here because namespace 'M' does not enclose namespace 'dr417'}}
+ struct dr417::H {};
+ // expected-error at -1 {{cannot define or redeclare 'H' here because namespace 'M' does not enclose namespace 'dr417'}}
}
}
@@ -326,7 +373,7 @@ void g() { f1(); }
namespace example2 {
namespace A {
-void f2(int); // #dr418-f2-decl
+void f2(int); // #dr418-f2
}
namespace B {
using A::f2;
@@ -336,8 +383,9 @@ void f2(int = 3);
}
void g2() {
using B::f2;
- f2(); // expected-error {{no matching function}}
- // expected-note@#dr418-f2-decl {{requires 1 argument}}
+ f2();
+ // expected-error at -1 {{no matching function for call to 'f2'}}
+ // expected-note@#dr418-f2 {{candidate function not viable: requires 1 argument, but 0 were provided}}
}
} // namespace example2
@@ -381,7 +429,8 @@ namespace dr420 { // dr420: 9
void test2(T p) {
p->template Y<int>::~Y<int>();
p->~Y<int>();
- p->template ~Y<int>(); // expected-error {{'template' keyword not permitted in destructor name}}
+ p->template ~Y<int>();
+ // expected-error at -1 {{'template' keyword not permitted in destructor name}}
}
template<typename T> struct Y {};
template void test2(Y<int>*);
@@ -413,17 +462,22 @@ namespace dr420 { // dr420: 9
namespace dr421 { // dr421: yes
struct X { X(); int n; int &r; };
- int *p = &X().n; // expected-error-re {{{{taking the address of a temporary|cannot take the address of an rvalue}}}}
+ int *p = &X().n;
+ // cxx98-error at -1 {{taking the address of a temporary object of type 'int'}}
+ // since-cxx11-error at -2 {{cannot take the address of an rvalue of type 'int'}}
int *q = &X().r;
}
namespace dr422 { // dr422: yes
template<typename T, typename U> void f() {
- typedef T type; // expected-note {{prev}}
- typedef U type; // expected-error {{redef}}
+ typedef T type; // #dr422-typedef-T
+ typedef U type;
+ // expected-error at -1 {{typedef redefinition with different types ('char' vs 'int')}}
+ // expected-note@#dr422-f-int-char {{in instantiation of function template specialization 'dr422::f<int, char>' requested here}}
+ // expected-note@#dr422-typedef-T {{previous definition is here}}
}
template void f<int, int>();
- template void f<int, char>(); // expected-note {{instantiation}}
+ template void f<int, char>(); // #dr422-f-int-char
}
namespace dr423 { // dr423: yes
@@ -433,37 +487,46 @@ namespace dr423 { // dr423: yes
namespace dr424 { // dr424: yes
struct A {
- typedef int N; // expected-note {{previous}}
- typedef int N; // expected-error {{redefinition}}
+ typedef int N; // #dr424-N
+ typedef int N;
+ // expected-error at -1 {{redefinition of 'N'}}
+ // expected-note@#dr424-N {{previous definition is here}}
struct X;
- typedef X X; // expected-note {{previous}}
+ typedef X X; // #dr424-X
struct X {};
struct X *p;
struct A::X *q;
X *r;
- typedef X X; // expected-error {{redefinition}}
+ typedef X X;
+ // expected-error at -1 {{redefinition of 'X'}}
+ // expected-note@#dr424-X {{previous definition is here}}
};
struct B {
- typedef int N;
+ typedef int M;
};
struct C : B {
- typedef int N; // expected-note {{previous}}
- typedef int N; // expected-error {{redefinition}}
+ typedef int M; // #dr424-M
+ typedef int M;
+ // expected-error at -1 {{redefinition of 'M'}}
+ // expected-note@#dr424-M {{previous definition is here}}
};
}
namespace dr425 { // dr425: yes
struct A { template<typename T> operator T() const; } a;
- float f = 1.0f * a; // expected-error {{ambiguous}} expected-note 5+{{built-in candidate}}
+ float f = 1.0f * a;
+ // expected-error at -1 {{use of overloaded operator '*' is ambiguous (with operand types 'float' and 'struct A')}}
+ // expected-note at -2 +{{built-in candidate}}
template<typename T> struct is_float;
template<> struct is_float<float> { typedef void type; };
struct B {
- template<typename T, typename U = typename is_float<T>::type> operator T() const; // expected-error 0-1{{extension}}
+ template<typename T, typename U = typename is_float<T>::type> operator T() const;
+ // cxx98-error at -1 {{default template arguments for a function template are a C++11 extension}}
} b;
float g = 1.0f * b; // ok
}
@@ -471,28 +534,42 @@ namespace dr425 { // dr425: yes
namespace dr427 { // dr427: yes
struct B {};
struct D : public B {
- D(B &) = delete; // expected-error 0-1{{extension}} expected-note {{deleted}}
+ D(B &) = delete; // #dr427-D
+ // cxx98-error at -1 {{deleted function definitions are a C++11 extension}}
};
extern D d1;
B &b = d1;
const D &d2 = static_cast<const D&>(b);
const D &d3 = (const D&)b;
- const D &d4(b); // expected-error {{deleted}}
+ const D &d4(b);
+ // expected-error at -1 {{conversion function from 'B' to 'const D' invokes a deleted function}}
+ // expected-note@#dr427-D {{'D' has been explicitly marked deleted here}}
}
namespace dr428 { // dr428: yes
template<typename T> T make();
- extern struct X x; // expected-note 5{{forward declaration}}
+ extern struct X x; // #dr428-X
void f() {
- throw void(); // expected-error {{cannot throw}}
+ throw void();
+ // expected-error at -1 {{cannot throw}}
throw make<void*>();
throw make<const volatile void*>();
- throw x; // expected-error {{cannot throw}}
- throw make<X&>(); // expected-error {{cannot throw}}
- throw make<X*>(); // expected-error {{cannot throw}}
- throw make<const volatile X&>(); // expected-error {{cannot throw}}
- throw make<const volatile X*>(); // expected-error {{cannot throw}}
+ throw x;
+ // expected-error at -1 {{cannot throw}}
+ // expected-note@#dr428-X {{forward declaration of 'dr428::X'}}
+ throw make<X&>();
+ // expected-error at -1 {{cannot throw}}
+ // expected-note@#dr428-X {{forward declaration of 'dr428::X'}}
+ throw make<X*>();
+ // expected-error at -1 {{cannot throw}}
+ // expected-note@#dr428-X {{forward declaration of 'dr428::X'}}
+ throw make<const volatile X&>();
+ // expected-error at -1 {{cannot throw}}
+ // expected-note@#dr428-X {{forward declaration of 'dr428::X'}}
+ throw make<const volatile X*>();
+ // expected-error at -1 {{cannot throw}}
+ // expected-note@#dr428-X {{forward declaration of 'dr428::X'}}
}
}
@@ -500,12 +577,10 @@ namespace dr429 { // dr429: 2.8 c++11
// FIXME: This rule is obviously intended to apply to C++98 as well.
struct A {
static void *operator new(size_t, size_t);
- static void operator delete(void*, size_t);
+ static void operator delete(void*, size_t); // #dr429-delete
} *a = new (0) A;
-#if __cplusplus >= 201103L
- // expected-error at -2 {{'new' expression with placement arguments refers to non-placement 'operator delete'}}
- // expected-note at -4 {{here}}
-#endif
+ // since-cxx11-error at -1 {{'new' expression with placement arguments refers to non-placement 'operator delete'}}
+ // since-cxx11-note@#dr429-delete {{here}}
struct B {
static void *operator new(size_t, size_t);
static void operator delete(void*);
@@ -518,9 +593,7 @@ namespace dr430 { // dr430: yes c++11
// FIXME: This should apply in C++98 too.
void f(int n) {
int a[] = { n++, n++, n++ };
-#if __cplusplus < 201103L
- // expected-warning at -2 {{multiple unsequenced modifications to 'n'}}
-#endif
+ // cxx98-warning at -1 {{multiple unsequenced modifications to 'n'}}
}
}
@@ -535,29 +608,37 @@ namespace dr431 { // dr431: yes
template<typename T> void f(A a) {
a.get<A>()->get<T>();
a.get<T>()
- ->get<T>(); // expected-error {{use 'template'}}
+ ->get<T>();
+ // expected-error at -1 {{use 'template' keyword to treat 'get' as a dependent template name}}
a.get<T>()->template get<T>();
a.A::get<T>();
A::B<int> *b = a.get<A::B<int> >();
b->get<int>();
b->A::B<int>::get<int>();
b->A::B<int>::get<T>();
- b->A::B<T>::get<int>(); // expected-error {{use 'template'}}
+ b->A::B<T>::get<int>();
+ // expected-error at -1 {{use 'template' keyword to treat 'get' as a dependent template name}}
b->A::B<T>::template get<int>();
- b->A::B<T>::get<T>(); // expected-error {{use 'template'}}
+ b->A::B<T>::get<T>();
+ // expected-error at -1 {{use 'template' keyword to treat 'get' as a dependent template name}}
b->A::B<T>::template get<T>();
A::B<T> *c = a.get<A::B<T> >();
- c->get<int>(); // expected-error {{use 'template'}}
+ c->get<int>();
+ // expected-error at -1 {{use 'template' keyword to treat 'get' as a dependent template name}}
c->template get<int>();
}
}
namespace dr432 { // dr432: 3.0
template<typename T> struct A {};
- template<typename T> struct B : A<B> {}; // expected-error {{requires template arguments}} expected-note {{declared}}
+ template<typename T> struct B : A<B> {};
+ // expected-error at -1 {{use of class template 'B' requires template arguments}}
+ // expected-note at -2 {{template is declared here}}
template<typename T> struct C : A<C<T> > {};
#if __cplusplus >= 201103L
- template<typename T> struct D : decltype(A<D>()) {}; // expected-error {{requires template arguments}} expected-note {{declared}}
+ template<typename T> struct D : decltype(A<D>()) {};
+ // since-cxx11-error at -1 {{use of class template 'D' requires template arguments}}
+ // since-cxx11-note at -2 {{template is declared here}}
#endif
}
@@ -575,7 +656,8 @@ namespace dr434 { // dr434: sup 2352
void f() {
const int ci = 0;
int *pi = 0;
- const int *&rpci = pi; // expected-error {{incompatible qualifiers}}
+ const int *&rpci = pi;
+ // expected-error at -1 {{binding reference of type 'const int *' to value of type 'int *' not permitted due to incompatible qualifiers}}
const int * const &rcpci = pi; // OK
rpci = &ci;
*pi = 1;
@@ -591,8 +673,10 @@ namespace dr434 { // dr434: sup 2352
// dr435: na
namespace dr436 { // dr436: yes
- enum E { f }; // expected-note {{previous}}
- void f(); // expected-error {{redefinition}}
+ enum E { f }; // #dr436-f
+ void f();
+ // expected-error at -1 {{redefinition of 'f' as different kind of symbol}}
+ // expected-note@#dr436-f {{previous definition is here}}
}
namespace dr437 { // dr437: sup 1308
@@ -601,18 +685,15 @@ namespace dr437 { // dr437: sup 1308
template<typename U> struct T : U {};
struct S {
void f() throw(S);
-#if __cplusplus > 201402L
- // expected-error at -2 {{ISO C++17 does not allow}} expected-note at -2 {{use 'noexcept}}
-#endif
+ // since-cxx17-error at -1 {{ISO C++17 does not allow dynamic exception specifications}}
+ // since-cxx17-note at -2 {{use 'noexcept(false)' instead}}
void g() throw(T<S>);
-#if __cplusplus > 201402L
- // expected-error at -2 {{ISO C++17 does not allow}} expected-note at -2 {{use 'noexcept}}
-#endif
+ // since-cxx17-error at -1 {{ISO C++17 does not allow dynamic exception specifications}}
+ // since-cxx17-note at -2 {{use 'noexcept(false)' instead}}
struct U;
void h() throw(U);
-#if __cplusplus > 201402L
- // expected-error at -2 {{ISO C++17 does not allow}} expected-note at -2 {{use 'noexcept}}
-#endif
+ // since-cxx17-error at -1 {{ISO C++17 does not allow dynamic exception specifications}}
+ // since-cxx17-note at -2 {{use 'noexcept(false)' instead}}
struct U {};
};
}
@@ -625,21 +706,30 @@ namespace dr437 { // dr437: sup 1308
namespace dr444 { // dr444: yes
struct D;
- struct B { // expected-note {{candidate function (the implicit copy}} expected-note 0-1 {{implicit move}}
- D &operator=(D &) = delete; // expected-error 0-1{{extension}} expected-note {{deleted}}
+ struct B { // #dr444-B
+ D &operator=(D &) = delete; // #dr444-deleted
+ // cxx98-error at -1 {{deleted function definitions are a C++11 extension}}
};
- struct D : B { // expected-note {{candidate function (the implicit copy}} expected-note 0-1 {{implicit move}}
+ struct D : B { // #dr444-D
using B::operator=;
} extern d;
void f() {
- d = d; // expected-error {{deleted}}
+ d = d;
+ // expected-error at -1 {{overload resolution selected deleted operator '='}}
+ // expected-note@#dr444-deleted {{candidate function has been explicitly deleted}}
+ // expected-note@#dr444-D {{candidate function (the implicit copy assignment operator)}}
+ // expected-note@#dr444-B {{candidate function (the implicit copy assignment operator)}}
+ // since-cxx11-note@#dr444-B {{candidate function (the implicit move assignment operator) not viable: expects an rvalue for 1st argument}}
+ // since-cxx11-note@#dr444-D {{candidate function (the implicit move assignment operator) not viable: expects an rvalue for 1st argument}}
}
}
namespace dr445 { // dr445: 3.2
- class A { void f(); }; // expected-note {{private}}
+ class A { void f(); }; // #dr445-f
struct B {
- friend void A::f(); // expected-error {{private}}
+ friend void A::f();
+ // expected-error at -1 {{friend function 'f' is a private member of 'dr445::A'}}
+ // expected-note@#dr445-f {{implicitly declared private here}}
};
}
@@ -647,29 +737,33 @@ namespace dr446 { // dr446: 2.8
struct C;
struct A {
A();
- A(const A&) = delete; // expected-error 0-1{{extension}} expected-note +{{deleted}}
+ A(const A&) = delete; // #dr446-deleted
+ // cxx98-error at -1 {{deleted function definitions are a C++11 extension}}
A(const C&);
};
struct C : A {};
void f(A a, bool b, C c) {
void(b ? a : a);
- b ? A() : a; // expected-error {{deleted}}
- b ? a : A(); // expected-error {{deleted}}
+ b ? A() : a;
+ // expected-error at -1 {{call to deleted constructor of 'A'}}
+ // expected-note@#dr446-deleted {{'A' has been explicitly marked deleted here}}
+ b ? a : A();
+ // expected-error at -1 {{call to deleted constructor of 'A'}}
+ // expected-note@#dr446-deleted {{'A' has been explicitly marked deleted here}}
b ? A() : A();
-#if __cplusplus <= 201402L
- // expected-error at -2 {{deleted}}
-#endif
+ // cxx98-14-error at -1 {{call to deleted constructor of 'A'}}
+ // expected-note@#dr446-deleted {{'A' has been explicitly marked deleted here}}
void(b ? a : c);
- b ? a : C(); // expected-error {{deleted}}
+ b ? a : C();
+ // expected-error at -1 {{call to deleted constructor of 'A'}}
+ // cxx98-14-note@#dr446-deleted {{'A' has been explicitly marked deleted here}}
b ? c : A();
-#if __cplusplus <= 201402L
- // expected-error at -2 {{deleted}}
-#endif
+ // cxx98-14-error at -1 {{call to deleted constructor of 'A'}}
+ // cxx98-14-note@#dr446-deleted {{'A' has been explicitly marked deleted here}}
b ? A() : C();
-#if __cplusplus <= 201402L
- // expected-error at -2 {{deleted}}
-#endif
+ // cxx98-14-error at -1 {{call to deleted constructor of 'A'}}
+ // cxx98-14-note@#dr446-deleted {{'A' has been explicitly marked deleted here}}
}
}
@@ -686,25 +780,37 @@ namespace dr447 { // dr447: yes
g(__builtin_offsetof(T, n)).h<int>();
// value dependent if first argument is a dependent type
U<__builtin_offsetof(A, n)>::type a;
- U<__builtin_offsetof(T, n)>::type b; // expected-error +{{}} expected-warning 0+{{}}
+ // FIXME: instead of ";", we should be suggesting "typename"
+ U<__builtin_offsetof(T, n)>::type b;
+ // expected-error at -1 {{expected ';' after expression}}
+ // expected-error at -2 {{use of undeclared identifier 'b'}}
// as an extension, we allow the member-designator to include array indices
g(__builtin_offsetof(A, a[0])).h<int>();
g(__builtin_offsetof(A, a[N])).h<int>();
U<__builtin_offsetof(A, a[0])>::type c;
- U<__builtin_offsetof(A, a[N])>::type d; // expected-error +{{}} expected-warning 0+{{}}
+ // FIXME: instead of ";", we should be suggesting "typename"
+ U<__builtin_offsetof(A, a[N])>::type d;
+ // expected-error at -1 {{expected ';' after expression}}
+ // expected-error at -2 {{use of undeclared identifier 'd'}}
}
}
namespace dr448 { // dr448: 2.8
- template<typename T = int> void f(int); // expected-error 0-1{{extension}} expected-note {{no known conversion}}
+ template<typename T = int> void f(int); // #dr448-f-int
+ // cxx98-error at -1 {{default template arguments for a function template are a C++11 extension}}
template<typename T> void g(T t) {
- f<T>(t); // expected-error {{neither visible in the template definition nor found by argument-dependent lookup}}
- dr448::f(t); // expected-error {{no matching function}}
+ f<T>(t);
+ // expected-error at -1 {{call to function 'f' that is neither visible in the template definition nor found by argument-dependent lookup}}
+ // expected-note@#dr448-g {{in instantiation of function template specialization 'dr448::g<dr448::HideFromADL::X>' requested here}}
+ // expected-note@#dr448-f-T {{'f' should be declared prior to the call site or in namespace 'dr448::HideFromADL'}}
+ dr448::f(t);
+ // expected-error at -1 {{no matching function for call to 'f'}}
+ // expected-note@#dr448-f-int {{candidate function template not viable: no known conversion from 'dr448::HideFromADL::X' to 'int' for 1st argument}}
}
- template<typename T> void f(T); // expected-note {{should be declared prior to the call site}}
+ template<typename T> void f(T); // #dr448-f-T
namespace HideFromADL { struct X {}; }
template void g(int); // ok
- template void g(HideFromADL::X); // expected-note {{instantiation of}}
+ template void g(HideFromADL::X); // #dr448-g
}
// dr449: na
@@ -712,24 +818,34 @@ namespace dr448 { // dr448: 2.8
namespace dr450 { // dr450: yes
typedef int A[3];
void f1(const A &);
- void f2(A &); // expected-note +{{not viable}}
+ void f2(A &); // #dr450-f2
struct S { A n; };
void g() {
f1(S().n);
- f2(S().n); // expected-error {{no match}}}
+ f2(S().n);
+ // expected-error at -1 {{no matching function for call to 'f2'}}}
+ // expected-note@#dr450-f2 {{candidate function not viable: expects an lvalue for 1st argument}}
}
#if __cplusplus >= 201103L
void h() {
f1(A{});
- f2(A{}); // expected-error {{no match}}
+ f2(A{});
+ // expected-error at -1 {{no matching function for call to 'f2'}}}
+ // expected-note@#dr450-f2 {{candidate function not viable: expects an lvalue for 1st argument}}
}
#endif
}
namespace dr451 { // dr451: yes
- const int a = 1 / 0; // expected-warning {{undefined}}
- const int b = 1 / 0; // expected-warning {{undefined}} expected-note {{here}} expected-note 0-1{{division by zero}}
- int arr[b]; // expected-error +{{variable length arr}} expected-note {{initializer of 'b' is not a constant}}
+ const int a = 1 / 0;
+ // expected-warning at -1 {{division by zero is undefined}}
+ const int b = 1 / 0; // #dr451-b
+ // expected-warning at -1 {{division by zero is undefined}}
+ int arr[b]; // #dr451-arr
+ // expected-error at -1 {{variable length arrays in C++ are a Clang extension}}
+ // expected-note at -2 {{initializer of 'b' is not a constant expression}}
+ // expected-note@#dr451-b {{declared here}}
+ // expected-error@#dr451-arr {{variable length array declaration not allowed at file scope}}
}
namespace dr452 { // dr452: yes
@@ -747,30 +863,29 @@ namespace dr456 { // dr456: yes
// sup 903 c++11
const int null = 0;
void *p = null;
-#if __cplusplus >= 201103L
- // expected-error at -2 {{cannot initialize}}
-#else
- // expected-warning at -4 {{null}}
-#endif
+ // cxx98-warning at -1 {{expression which evaluates to zero treated as a null pointer constant of type 'void *'}}
+ // since-cxx11-error at -2 {{cannot initialize a variable of type 'void *' with an lvalue of type 'const int'}}
const bool f = false;
void *q = f;
-#if __cplusplus >= 201103L
- // expected-error at -2 {{cannot initialize}}
-#else
- // expected-warning at -4 {{null}}
-#endif
+ // cxx98-warning at -1 {{initialization of pointer of type 'void *' to null from a constant boolean}}
+ // since-cxx11-error at -2 {{cannot initialize a variable of type 'void *' with an lvalue of type 'const bool'}}
}
namespace dr457 { // dr457: yes
const int a = 1;
const volatile int b = 1;
int ax[a];
- int bx[b]; // expected-error +{{variable length array}} expected-note {{read of volatile}}
+ int bx[b];
+ // expected-error at -1 {{variable length arrays in C++ are a Clang extension}}
+ // expected-note at -2 {{read of volatile-qualified type 'const volatile int' is not allowed in a constant expression}}
+ // expected-error at -3 {{variable length array declaration not allowed at file scope}}
enum E {
ea = a,
- eb = b // expected-error {{constant}} expected-note {{read of volatile-qualified}}
+ eb = b
+ // expected-error at -1 {{expression is not an integral constant expression}}
+ // expected-note at -2 {{read of volatile-qualified type 'const volatile int' is not allowed in a constant expression}}
};
}
@@ -790,31 +905,40 @@ namespace dr458 { // dr458: 11
int A::f() {
return T;
}
- template<typename T> // expected-note {{declared here}}
+ template<typename T> // #dr458-g-T
int A::g() {
- return T; // expected-error {{'T' does not refer to a value}}
+ return T;
+ // expected-error at -1 {{'T' does not refer to a value}}
+ // expected-note@#dr458-g-T {{declared here}}
}
template<typename T>
int B<T>::f() {
return T;
}
- template<typename T> template<typename U>
+ template<typename T>
+ template<typename U>
int B<T>::g() {
return T;
}
- template<typename U> template<typename T> // expected-note {{declared here}}
+ template<typename U>
+ template<typename T> // #dr458-h-T
int B<U>::h() {
- return T; // expected-error {{'T' does not refer to a value}}
+ return T;
+ // expected-error at -1 {{'T' does not refer to a value}}
+ // expected-note@#dr458-h-T {{declared here}}
}
}
namespace dr460 { // dr460: yes
namespace X { namespace Q { int n; } }
namespace Y {
- using X; // expected-error {{requires a qualified name}}
- using dr460::X; // expected-error {{cannot refer to a namespace}}
- using X::Q; // expected-error {{cannot refer to a namespace}}
+ using X;
+ // expected-error at -1 {{using declaration requires a qualified name}}
+ using dr460::X;
+ // expected-error at -1 {{using declaration cannot refer to a namespace}}
+ using X::Q;
+ // expected-error at -1 {{using declaration cannot refer to a namespace}}
}
}
@@ -867,8 +991,10 @@ namespace dr467 { // dr467: yes
return k;
}
int g() {
- goto later; // expected-error {{cannot jump}}
- int k = stuff(); // expected-note {{bypasses variable initialization}}
+ goto later;
+ // expected-error at -1 {{cannot jump from this goto statement to its label}}
+ // expected-note@#dr467-k {{jump bypasses variable initialization}}
+ int k = stuff(); // #dr467-k
later:
return k;
}
@@ -882,15 +1008,16 @@ namespace dr468 { // dr468: yes c++11
};
};
int k = dr468::template A<int>::template B<char>::C;
-#if __cplusplus < 201103L
- // expected-error at -2 2{{'template' keyword outside of a template}}
-#endif
+ // cxx98-error at -1 {{'template' keyword outside of a template}}
+ // cxx98-error at -2 {{'template' keyword outside of a template}}
}
namespace dr469 { // dr469: no
- template<typename T> struct X; // expected-note {{here}}
+ template<typename T> struct X; // #dr469-X
template<typename T> struct X<const T> {};
- X<int&> x; // expected-error {{undefined}}
+ X<int&> x;
+ // expected-error at -1 {{implicit instantiation of undefined template 'dr469::X<int &>'}}
+ // expected-note@#dr469-X {{template is declared here}}
}
namespace dr470 { // dr470: yes
@@ -900,8 +1027,10 @@ namespace dr470 { // dr470: yes
template<typename T> struct C {
};
- template struct A<int>; // expected-note {{previous}}
- template struct A<int>::B; // expected-error {{duplicate explicit instantiation}}
+ template struct A<int>; // #dr470-A-int
+ template struct A<int>::B;
+ // expected-error at -1 {{duplicate explicit instantiation of 'B'}}
+ // expected-note@#dr470-A-int {{previous explicit instantiation is here}}
// ok, instantiating C<char> doesn't instantiate base class members.
template struct A<char>;
@@ -919,9 +1048,11 @@ namespace dr471 { // dr471: 2.8
struct F : E, B { int f() { return n; } };
struct G : virtual A {
private:
- using A::n; // expected-note {{here}}
+ using A::n; // #dr471-G-using
};
- struct H : B, G { int f() { return n; } }; // expected-error {{private}}
+ struct H : B, G { int f() { return n; } };
+ // expected-error at -1 {{'n' is a private member of 'dr471::G'}}
+ // expected-note@#dr471-G-using {{declared private here}}
}
namespace dr474 { // dr474: 3.4
@@ -931,11 +1062,13 @@ namespace dr474 { // dr474: 3.4
};
}
void N::S::f() {
- void g(); // expected-note {{previous}}
+ void g(); // #dr474-g
}
int g();
namespace N {
- int g(); // expected-error {{cannot be overloaded}}
+ int g();
+ // expected-error at -1 {{functions that differ only in their return type cannot be overloaded}}
+ // expected-note@#dr474-g {{previous declaration is here}}
}
}
@@ -947,50 +1080,62 @@ namespace dr477 { // dr477: 3.5
virtual void f();
};
struct B {
- friend explicit A::A(); // expected-error {{'explicit' is invalid in friend declarations}}
- friend virtual void A::f(); // expected-error {{'virtual' is invalid in friend declarations}}
+ friend explicit A::A();
+ // expected-error at -1 {{'explicit' is invalid in friend declarations}}
+ friend virtual void A::f();
+ // expected-error at -1 {{'virtual' is invalid in friend declarations}}
};
- explicit A::A() {} // expected-error {{can only be specified inside the class definition}}
- virtual void A::f() {} // expected-error {{can only be specified inside the class definition}}
+ explicit A::A() {}
+ // expected-error at -1 {{can only be specified inside the class definition}}
+ virtual void A::f() {}
+ // expected-error at -1 {{can only be specified inside the class definition}}
}
namespace dr478 { // dr478: yes
- struct A { virtual void f() = 0; }; // expected-note {{unimplemented}}
+ struct A { virtual void f() = 0; }; // #dr478-f
void f(A *a);
- void f(A a[10]); // expected-error {{array of abstract class type}}
+ void f(A a[10]);
+ // expected-error at -1 {{array of abstract class type 'A'}}
+ // expected-note@#dr478-f {{unimplemented pure virtual method 'f' in 'A'}}
}
namespace dr479 { // dr479: 2.8
struct S {
S();
private:
- S(const S&); // expected-note +{{here}}
- ~S(); // expected-note +{{here}}
+ S(const S&); // #dr479-S-copy-ctor
+ ~S(); // #dr479-S-dtor
};
void f() {
throw S();
// expected-error at -1 {{temporary of type 'S' has private destructor}}
- // expected-error at -2 {{exception object of type 'S' has private destructor}}
-#if __cplusplus < 201103L
- // expected-error at -4 {{C++98 requires an accessible copy constructor}}
-#endif
-#if __cplusplus <= 201402L
- // expected-error at -7 {{calling a private constructor}} (copy ctor)
-#endif
+ // expected-note@#dr479-S-dtor {{declared private here}}
+ // expected-error at -3 {{exception object of type 'S' has private destructor}}
+ // expected-note@#dr479-S-dtor {{declared private here}}
+ // cxx98-error at -5 {{C++98 requires an accessible copy constructor for class 'dr479::S' when binding a reference to a temporary; was private}}
+ // cxx98-note@#dr479-S-copy-ctor {{declared private here}}
+ // cxx98-14-error at -7 {{calling a private constructor of class 'dr479::S'}}
+ // cxx98-14-note@#dr479-S-copy-ctor {{declared private here}}
}
void g() {
- S s; // expected-error {{private destructor}}}
+ S s;
+ // expected-error at -1 {{variable of type 'S' has private destructor}}
+ // expected-note@#dr479-S-dtor {{declared private here}}
throw s;
- // expected-error at -1 {{calling a private constructor}}
- // expected-error at -2 {{exception object of type 'S' has private destructor}}
+ // expected-error at -1 {{exception object of type 'S' has private destructor}}
+ // expected-note@#dr479-S-dtor {{declared private here}}
+ // expected-error at -3 {{calling a private constructor of class 'dr479::S'}}
+ // expected-note@#dr479-S-copy-ctor {{declared private here}}
}
void h() {
try {
f();
g();
} catch (S s) {
- // expected-error at -1 {{calling a private constructor}}
- // expected-error at -2 {{variable of type 'S' has private destructor}}
+ // expected-error at -1 {{calling a private constructor of class 'dr479::S'}}
+ // expected-note@#dr479-S-copy-ctor {{declared private here}}
+ // expected-error at -3 {{variable of type 'S' has private destructor}}
+ // expected-note@#dr479-S-dtor {{declared private here}}
}
}
}
@@ -1002,23 +1147,28 @@ namespace dr480 { // dr480: yes
struct D : C {};
int A::*a = &A::n;
- int D::*b = a; // expected-error {{virtual base}}
+ int D::*b = a;
+ // expected-error at -1 {{conversion from pointer to member of class 'dr480::A' to pointer to member of class 'dr480::D' via virtual base 'dr480::B' is not allowed}}
extern int D::*c;
- int A::*d = static_cast<int A::*>(c); // expected-error {{virtual base}}
+ int A::*d = static_cast<int A::*>(c);
+ // expected-error at -1 {{conversion from pointer to member of class 'dr480::D' to pointer to member of class 'dr480::A' via virtual base 'dr480::B' is not allowed}}
D *e;
A *f = e;
- D *g = static_cast<D*>(f); // expected-error {{virtual base}}
+ D *g = static_cast<D*>(f);
+ // expected-error at -1 {{cannot cast 'dr480::A *' to 'D *' via virtual base 'dr480::B'}}
extern D &i;
A &j = i;
- D &k = static_cast<D&>(j); // expected-error {{virtual base}}
+ D &k = static_cast<D&>(j);
+ // expected-error at -1 {{cannot cast 'A' to 'D &' via virtual base 'dr480::B'}}
}
namespace dr481 { // dr481: 2.8
template<class T, T U> class A { T *x; };
- T *x; // expected-error {{unknown type}}
+ T *x;
+ // expected-error at -1 {{unknown type name 'T'}}
template<class T *U> class B { T *x; };
T *y; // ok
@@ -1030,18 +1180,23 @@ namespace dr481 { // dr481: 2.8
template<typename A = C, typename C = A> struct E {
void f() {
- typedef ::dr481::C c; // expected-note {{previous}}
- typedef C c; // expected-error {{different type}}
+ typedef ::dr481::C c; // #dr481-c
+ typedef C c;
+ // expected-error at -1 {{typedef redefinition with different types ('int' vs '::dr481::C')}}
+ // expected-note@#dr481-E-int {{in instantiation of member function 'dr481::E<int>::f' requested here}}
+ // expected-note@#dr481-c {{previous definition is here}}
}
};
template struct E<>; // ok
- template struct E<int>; // expected-note {{instantiation of}}
+ template struct E<int>; // #dr481-E-int
template<template<typename U_no_typo_correction> class A,
A<int> *B,
- U_no_typo_correction *C> // expected-error {{unknown type}}
+ U_no_typo_correction *C>
+ // expected-error at -1 {{unknown type name 'U_no_typo_correction'}}
struct F {
- U_no_typo_correction *x; // expected-error {{unknown type}}
+ U_no_typo_correction *x;
+ // expected-error at -1 {{unknown type name 'U_no_typo_correction'}}
};
template<template<class H *> class> struct G {
@@ -1059,32 +1214,43 @@ namespace dr482 { // dr482: 3.5
extern int a;
void f();
- int dr482::a = 0; // expected-warning {{extra qualification}}
- void dr482::f() {} // expected-warning {{extra qualification}}
+ int dr482::a = 0;
+ // expected-warning at -1 {{extra qualification on member 'a'}}
+ void dr482::f() {}
+ // expected-warning at -1 {{extra qualification on member 'f'}}
- inline namespace X { // expected-error 0-1{{C++11 feature}}
+ inline namespace X {
+ // cxx98-error at -1 {{inline namespaces are a C++11 feature}}
extern int b;
void g();
struct S;
}
- int dr482::b = 0; // expected-warning {{extra qualification}}
- void dr482::g() {} // expected-warning {{extra qualification}}
- struct dr482::S {}; // expected-warning {{extra qualification}}
-
- void dr482::f(); // expected-warning {{extra qualification}}
- void dr482::g(); // expected-warning {{extra qualification}}
+ int dr482::b = 0;
+ // expected-warning at -1 {{extra qualification on member 'b'}}
+ void dr482::g() {}
+ // expected-warning at -1 {{extra qualification on member 'g'}}
+ struct dr482::S {};
+ // expected-warning at -1 {{extra qualification on member 'S'}}
+
+ void dr482::f();
+ // expected-warning at -1 {{extra qualification on member 'f'}}
+ void dr482::g();
+ // expected-warning at -1 {{extra qualification on member 'g'}}
// FIXME: The following are valid in DR482's wording, but these are bugs in
// the wording which we deliberately don't implement.
namespace N { typedef int type; }
- typedef int N::type; // expected-error {{typedef declarator cannot be qualified}}
+ typedef int N::type;
+ // expected-error at -1 {{typedef declarator cannot be qualified}}
struct A {
struct B;
- struct A::B {}; // expected-error {{extra qualification}}
+ struct A::B {};
+ // expected-error at -1 {{extra qualification on member 'B'}}
#if __cplusplus >= 201103L
enum class C;
- enum class A::C {}; // expected-error {{extra qualification}}
+ enum class A::C {};
+ // expected-error at -1 {{extra qualification on member 'C'}}
#endif
};
}
@@ -1096,9 +1262,8 @@ namespace dr483 { // dr483: yes
int check3[__INT_MAX__ >= 32767 ? 1 : -1];
int check4[__LONG_MAX__ >= 2147483647 ? 1 : -1];
int check5[__LONG_LONG_MAX__ >= 9223372036854775807 ? 1 : -1];
-#if __cplusplus < 201103L
- // expected-error at -2 1+{{extension}}
-#endif
+ // cxx98-error at -1 {{'long long' is a C++11 extension}}
+ // endill-note at -2 {{expanded from macro '__LONG_LONG_MAX__'}}
}
namespace cstdint {
int check1[__PTRDIFF_WIDTH__ >= 16 ? 1 : -1];
@@ -1128,17 +1293,22 @@ namespace dr484 { // dr484: yes
};
struct C;
- typedef C CT; // expected-note {{here}}
- struct CT {}; // expected-error {{conflicts with typedef}}
+ typedef C CT; // #dr484-typedef-CT
+ struct CT {};
+ // expected-error at -1 {{definition of type 'CT' conflicts with typedef of the same name}}
+ // expected-note@#dr484-typedef-CT {{'CT' declared here}}
namespace N {
struct D;
- typedef D DT; // expected-note {{here}}
+ typedef D DT; // #dr484-typedef-DT
}
- struct N::DT {}; // expected-error {{conflicts with typedef}}
+ struct N::DT {};
+ // expected-error at -1 {{definition of type 'DT' conflicts with typedef of the same name}}
+ // expected-note@#dr484-typedef-DT {{'DT' declared here}}
typedef struct {
- S(); // expected-error {{a type specifier is required}}
+ S();
+ // expected-error at -1 {{a type specifier is required for all declarations}}
} S;
}
@@ -1156,7 +1326,7 @@ namespace dr485 { // dr485: yes
}
namespace dr486 { // dr486: yes
- template<typename T> T f(T *); // expected-note 2{{substitution failure}}
+ template<typename T> T f(T *); // #dr486-f
int &f(...);
void g();
@@ -1165,15 +1335,23 @@ namespace dr486 { // dr486: yes
void h() {
int &a = f(&g);
int &b = f(&n);
- f<void()>(&g); // expected-error {{no match}}
- f<int[10]>(&n); // expected-error {{no match}}
+ f<void()>(&g);
+ // expected-error at -1 {{no matching function for call to 'f'}}
+ // expected-note@#dr486-f {{candidate template ignored: substitution failure [with T = void ()]: function cannot return function type 'void ()'}}
+ f<int[10]>(&n);
+ // expected-error at -1 {{no matching function for call to 'f'}}
+ // expected-note@#dr486-f {{candidate template ignored: substitution failure [with T = int[10]]: function cannot return array type 'int[10]'}}
}
}
namespace dr487 { // dr487: yes
enum E { e };
- int operator+(int, E); // expected-note 0-1{{here}}
- int i[4 + e]; // expected-error 2{{variable length array}} expected-note 0-1{{non-constexpr}}
+ int operator+(int, E); // #dr487-operator-plus
+ int i[4 + e]; // #dr487-i
+ // expected-error at -1 {{variable length arrays in C++ are a Clang extension}}
+ // since-cxx11-note at -2 {{non-constexpr function 'operator+' cannot be used in a constant expression}}
+ // since-cxx11-note@#dr487-operator-plus {{declared here}}
+ // expected-error@#dr487-i {{variable length array declaration not allowed at file scope}}
}
namespace dr488 { // dr488: yes c++11
@@ -1186,9 +1364,7 @@ namespace dr488 { // dr488: yes c++11
// failure.
enum E { e };
f(e);
-#if __cplusplus < 201103L
- // expected-error at -2 {{local type}}
-#endif
+ // cxx98-error at -1 {{template argument uses local type 'E'}}
}
}
@@ -1199,12 +1375,12 @@ namespace dr490 { // dr490: 2.8
struct A {
typedef int T;
- struct K {}; // expected-note {{declared}}
+ struct K {}; // #dr490-k
int f(T);
int g(T);
int h(X<T>);
- int X<T>::*i(); // expected-note {{previous}}
+ int X<T>::*i(); // #dr490-i
int K::*j();
template<typename T> T k();
@@ -1221,8 +1397,12 @@ namespace dr490 { // dr490: 2.8
// FIXME: Per this DR, these two are valid! That is another defect
// (no number yet...) which will eventually supersede this one.
- friend int X<T>::*A::i(); // expected-error {{return type}}
- friend int K::*A::j(); // expected-error {{undeclared identifier 'K'; did you mean 'A::K'?}}
+ friend int X<T>::*A::i();
+ // expected-error at -1 {{return type of out-of-line definition of 'dr490::A::i' differs from that in the declaration}}
+ // expected-note@#dr490-i {{previous declaration is here}}
+ friend int K::*A::j();
+ // expected-error at -1 {{use of undeclared identifier 'K'; did you mean 'A::K'?}}
+ // expected-note@#dr490-k {{'A::K' declared here}}
// ok, lookup finds B::T, not A::T, so return type matches
friend char A::k<T>();
@@ -1236,7 +1416,8 @@ namespace dr490 { // dr490: 2.8
namespace dr491 { // dr491: dup 413
struct A {} a, b[3] = { a, {} };
- A c[2] = { a, {}, b[1] }; // expected-error {{excess elements}}
+ A c[2] = { a, {}, b[1] };
+ // expected-error at -1 {{excess elements in array initializer}}
}
// dr492 FIXME write a codegen test
@@ -1301,7 +1482,8 @@ namespace dr497 { // dr497: sup 253
};
const S cs;
int S::*pm = &S::i;
- cs.*pm = 88; // expected-error {{not assignable}}
+ cs.*pm = 88;
+ // expected-error at -1 {{read-only variable is not assignable}}
}
void after() {
@@ -1311,7 +1493,8 @@ namespace dr497 { // dr497: sup 253
};
const S cs;
int S::*pm = &S::i;
- cs.*pm = 88; // expected-error {{not assignable}}
+ cs.*pm = 88;
+ // expected-error at -1 {{read-only variable is not assignable}}
}
}
>From 385d250a368c91e8ba55010e58ca5a007bbaa110 Mon Sep 17 00:00:00 2001
From: Vlad Serebrennikov <serebrennikov.vladislav at gmail.com>
Date: Mon, 4 Dec 2023 16:01:49 +0300
Subject: [PATCH 2/3] Remove commented-out expected directive
---
clang/test/CXX/drs/dr4xx.cpp | 1 -
1 file changed, 1 deletion(-)
diff --git a/clang/test/CXX/drs/dr4xx.cpp b/clang/test/CXX/drs/dr4xx.cpp
index b7df9decdd4b8..f176313ba2735 100644
--- a/clang/test/CXX/drs/dr4xx.cpp
+++ b/clang/test/CXX/drs/dr4xx.cpp
@@ -1263,7 +1263,6 @@ namespace dr483 { // dr483: yes
int check4[__LONG_MAX__ >= 2147483647 ? 1 : -1];
int check5[__LONG_LONG_MAX__ >= 9223372036854775807 ? 1 : -1];
// cxx98-error at -1 {{'long long' is a C++11 extension}}
- // endill-note at -2 {{expanded from macro '__LONG_LONG_MAX__'}}
}
namespace cstdint {
int check1[__PTRDIFF_WIDTH__ >= 16 ? 1 : -1];
>From 74a50dc5138823b89dbf860084f8c91613d2c7ce Mon Sep 17 00:00:00 2001
From: Vlad Serebrennikov <serebrennikov.vladislav at gmail.com>
Date: Mon, 4 Dec 2023 16:20:10 +0300
Subject: [PATCH 3/3] Add expected directive to accomodate for windows triple
---
clang/test/CXX/drs/dr4xx.cpp | 2 ++
1 file changed, 2 insertions(+)
diff --git a/clang/test/CXX/drs/dr4xx.cpp b/clang/test/CXX/drs/dr4xx.cpp
index f176313ba2735..3f9597278d6fd 100644
--- a/clang/test/CXX/drs/dr4xx.cpp
+++ b/clang/test/CXX/drs/dr4xx.cpp
@@ -1261,8 +1261,10 @@ namespace dr483 { // dr483: yes
int check2[__SHRT_MAX__ >= 32767 ? 1 : -1];
int check3[__INT_MAX__ >= 32767 ? 1 : -1];
int check4[__LONG_MAX__ >= 2147483647 ? 1 : -1];
+ // FIXME: we issue 2 diagnostics for windows triple, but only 1 for linux triple.
int check5[__LONG_LONG_MAX__ >= 9223372036854775807 ? 1 : -1];
// cxx98-error at -1 {{'long long' is a C++11 extension}}
+ // cxx98-error at -2 0-1{{'long long' is a C++11 extension}}
}
namespace cstdint {
int check1[__PTRDIFF_WIDTH__ >= 16 ? 1 : -1];
More information about the cfe-commits
mailing list