|  | // RUN: %clang_cc1 -std=c++98 -triple x86_64-unknown-unknown %s -verify -fexceptions -fcxx-exceptions -pedantic-errors | 
|  | // RUN: %clang_cc1 -std=c++11 -triple x86_64-unknown-unknown %s -verify -fexceptions -fcxx-exceptions -pedantic-errors | 
|  | // RUN: %clang_cc1 -std=c++14 -triple x86_64-unknown-unknown %s -verify -fexceptions -fcxx-exceptions -pedantic-errors | 
|  | // RUN: %clang_cc1 -std=c++17 -triple x86_64-unknown-unknown %s -verify -fexceptions -fcxx-exceptions -pedantic-errors | 
|  | // RUN: %clang_cc1 -std=c++2a -triple x86_64-unknown-unknown %s -verify -fexceptions -fcxx-exceptions -pedantic-errors | 
|  |  | 
|  | #if __cplusplus < 201103L | 
|  | // expected-error@+1 {{variadic macro}} | 
|  | #define static_assert(...) __extension__ _Static_assert(__VA_ARGS__) | 
|  | #endif | 
|  |  | 
|  | #if __cplusplus >= 201103L | 
|  | namespace std { | 
|  | typedef decltype(sizeof(int)) size_t; | 
|  |  | 
|  | template<typename E> class initializer_list { | 
|  | const E *begin; | 
|  | size_t size; | 
|  |  | 
|  | public: | 
|  | initializer_list(); | 
|  | }; | 
|  | } // std | 
|  | #endif | 
|  |  | 
|  | namespace dr1611 { // dr1611: dup 1658 | 
|  | struct A { A(int); }; | 
|  | struct B : virtual A { virtual void f() = 0; }; | 
|  | struct C : B { C() : A(0) {} void f(); }; | 
|  | C c; | 
|  | } | 
|  |  | 
|  | namespace dr1684 { // dr1684: 3.6 | 
|  | #if __cplusplus >= 201103L | 
|  | struct NonLiteral { // expected-note {{because}} | 
|  | NonLiteral(); | 
|  | constexpr int f() { return 0; } // expected-warning 0-1{{will not be implicitly 'const'}} | 
|  | }; | 
|  | constexpr int f(NonLiteral &) { return 0; } | 
|  | constexpr int f(NonLiteral) { return 0; } // expected-error {{not a literal type}} | 
|  | #endif | 
|  | } | 
|  |  | 
|  | namespace dr1631 {  // dr1631: 3.7 | 
|  | #if __cplusplus >= 201103L | 
|  | // Incorrect overload resolution for single-element initializer-list | 
|  |  | 
|  | struct A { int a[1]; }; | 
|  | struct B { B(int); }; | 
|  | void f(B, int); | 
|  | void f(B, int, int = 0); | 
|  | void f(int, A); | 
|  |  | 
|  | void test() { | 
|  | f({0}, {{1}}); // expected-warning {{braces around scalar init}} | 
|  | } | 
|  |  | 
|  | namespace with_error { | 
|  | void f(B, int);           // TODO: expected- note {{candidate function}} | 
|  | void f(int, A);           // expected-note {{candidate function}} | 
|  | void f(int, A, int = 0);  // expected-note {{candidate function}} | 
|  |  | 
|  | void test() { | 
|  | f({0}, {{1}});        // expected-error{{call to 'f' is ambiguous}} | 
|  | } | 
|  | } | 
|  | #endif | 
|  | } | 
|  |  | 
|  | namespace dr1638 { // dr1638: yes | 
|  | #if __cplusplus >= 201103L | 
|  | template<typename T> struct A { | 
|  | enum class E; // expected-note {{previous}} | 
|  | enum class F : T; // expected-note 2{{previous}} | 
|  | }; | 
|  |  | 
|  | template<> enum class A<int>::E; | 
|  | template<> enum class A<int>::E {}; | 
|  | template<> enum class A<int>::F : int; | 
|  | template<> enum class A<int>::F : int {}; | 
|  |  | 
|  | template<> enum class A<short>::E : int; | 
|  | template<> enum class A<short>::E : int {}; | 
|  |  | 
|  | template<> enum class A<short>::F; // expected-error {{different underlying type}} | 
|  | template<> enum class A<char>::E : char; // expected-error {{different underlying type}} | 
|  | template<> enum class A<char>::F : int; // expected-error {{different underlying type}} | 
|  |  | 
|  | enum class A<unsigned>::E; // expected-error {{template specialization requires 'template<>'}} expected-error {{nested name specifier}} | 
|  | template enum class A<unsigned>::E; // expected-error {{enumerations cannot be explicitly instantiated}} | 
|  | enum class A<unsigned>::E *e; // expected-error {{must use 'enum' not 'enum class'}} | 
|  |  | 
|  | struct B { | 
|  | friend enum class A<unsigned>::E; // expected-error {{must use 'enum' not 'enum class'}} | 
|  | }; | 
|  | #endif | 
|  | } | 
|  |  | 
|  | namespace dr1645 { // dr1645: 3.9 | 
|  | #if __cplusplus >= 201103L | 
|  | struct A { | 
|  | constexpr A(int, float = 0); // expected-note 2{{candidate}} | 
|  | explicit A(int, int = 0); // expected-note 2{{candidate}} | 
|  | A(int, int, int = 0) = delete; // expected-note {{candidate}} | 
|  | }; | 
|  |  | 
|  | struct B : A { // expected-note 2{{candidate}} | 
|  | using A::A; // expected-note 5{{inherited here}} | 
|  | }; | 
|  |  | 
|  | constexpr B a(0); // expected-error {{ambiguous}} | 
|  | constexpr B b(0, 0); // expected-error {{ambiguous}} | 
|  | #endif | 
|  | } | 
|  |  | 
|  | namespace dr1653 { // dr1653: 4 c++17 | 
|  | void f(bool b) { | 
|  | ++b; | 
|  | b++; | 
|  | #if __cplusplus <= 201402L | 
|  | // expected-warning@-3 {{deprecated}} expected-warning@-2 {{deprecated}} | 
|  | #else | 
|  | // expected-error@-5 {{incrementing expression of type bool}} expected-error@-4 {{incrementing expression of type bool}} | 
|  | #endif | 
|  | --b; // expected-error {{cannot decrement expression of type bool}} | 
|  | b--; // expected-error {{cannot decrement expression of type bool}} | 
|  | b += 1; // ok | 
|  | b -= 1; // ok | 
|  | } | 
|  | } | 
|  |  | 
|  | namespace dr1658 { // dr1658: 5 | 
|  | namespace DefCtor { | 
|  | class A { A(); }; // expected-note 0-2{{here}} | 
|  | class B { ~B(); }; // expected-note 0-2{{here}} | 
|  |  | 
|  | // The stars align! An abstract class does not construct its virtual bases. | 
|  | struct C : virtual A { C(); virtual void foo() = 0; }; | 
|  | C::C() = default; // ok, not deleted, expected-error 0-1{{extension}} | 
|  | struct D : virtual B { D(); virtual void foo() = 0; }; | 
|  | D::D() = default; // ok, not deleted, expected-error 0-1{{extension}} | 
|  |  | 
|  | // In all other cases, we are not so lucky. | 
|  | struct E : A { E(); virtual void foo() = 0; }; | 
|  | #if __cplusplus < 201103L | 
|  | E::E() = default; // expected-error {{private default constructor}} expected-error {{extension}} expected-note {{here}} | 
|  | #else | 
|  | E::E() = default; // expected-error {{would delete}} expected-note@-4{{inaccessible default constructor}} | 
|  | #endif | 
|  | struct F : virtual A { F(); }; | 
|  | #if __cplusplus < 201103L | 
|  | F::F() = default; // expected-error {{private default constructor}} expected-error {{extension}} expected-note {{here}} | 
|  | #else | 
|  | F::F() = default; // expected-error {{would delete}} expected-note@-4{{inaccessible default constructor}} | 
|  | #endif | 
|  |  | 
|  | struct G : B { G(); virtual void foo() = 0; }; | 
|  | #if __cplusplus < 201103L | 
|  | G::G() = default; // expected-error@-2 {{private destructor}} expected-error {{extension}} expected-note {{here}} | 
|  | #else | 
|  | G::G() = default; // expected-error {{would delete}} expected-note@-4{{inaccessible destructor}} | 
|  | #endif | 
|  | struct H : virtual B { H(); }; | 
|  | #if __cplusplus < 201103L | 
|  | H::H() = default; // expected-error@-2 {{private destructor}} expected-error {{extension}} expected-note {{here}} | 
|  | #else | 
|  | H::H() = default; // expected-error {{would delete}} expected-note@-4{{inaccessible destructor}} | 
|  | #endif | 
|  | } | 
|  |  | 
|  | namespace Dtor { | 
|  | class B { ~B(); }; // expected-note 0-2{{here}} | 
|  |  | 
|  | struct D : virtual B { ~D(); virtual void foo() = 0; }; | 
|  | D::~D() = default; // ok, not deleted, expected-error 0-1{{extension}} | 
|  |  | 
|  | struct G : B { ~G(); virtual void foo() = 0; }; | 
|  | #if __cplusplus < 201103L | 
|  | G::~G() = default; // expected-error@-2 {{private destructor}} expected-error {{extension}} expected-note {{here}} | 
|  | #else | 
|  | G::~G() = default; // expected-error {{would delete}} expected-note@-4{{inaccessible destructor}} | 
|  | #endif | 
|  | struct H : virtual B { ~H(); }; | 
|  | #if __cplusplus < 201103L | 
|  | H::~H() = default; // expected-error@-2 {{private destructor}} expected-error {{extension}} expected-note {{here}} | 
|  | #else | 
|  | H::~H() = default; // expected-error {{would delete}} expected-note@-4{{inaccessible destructor}} | 
|  | #endif | 
|  | } | 
|  |  | 
|  | namespace MemInit { | 
|  | struct A { A(int); }; // expected-note {{here}} | 
|  | struct B : virtual A { | 
|  | B() {} | 
|  | virtual void f() = 0; | 
|  | }; | 
|  | struct C : virtual A { | 
|  | C() {} // expected-error {{must explicitly initialize}} | 
|  | }; | 
|  | } | 
|  |  | 
|  | namespace CopyCtorParamType { | 
|  | struct A { A(A&); }; | 
|  | struct B : virtual A { virtual void f() = 0; }; | 
|  | struct C : virtual A { virtual void f(); }; | 
|  | struct D : A { virtual void f() = 0; }; | 
|  |  | 
|  | struct X { | 
|  | friend B::B(const B&) throw(); | 
|  | friend C::C(C&); | 
|  | friend D::D(D&); | 
|  | }; | 
|  | } | 
|  |  | 
|  | namespace CopyCtor { | 
|  | class A { A(const A&); A(A&&); }; // expected-note 0-4{{here}} expected-error 0-1{{extension}} | 
|  |  | 
|  | struct C : virtual A { C(const C&); C(C&&); virtual void foo() = 0; }; // expected-error 0-1{{extension}} | 
|  | C::C(const C&) = default; // expected-error 0-1{{extension}} | 
|  | C::C(C&&) = default; // expected-error 0-2{{extension}} | 
|  |  | 
|  | struct E : A { E(const E&); E(E&&); virtual void foo() = 0; }; // expected-error 0-1{{extension}} | 
|  | #if __cplusplus < 201103L | 
|  | E::E(const E&) = default; // expected-error {{private copy constructor}} expected-error {{extension}} expected-note {{here}} | 
|  | E::E(E&&) = default; // expected-error {{private move constructor}} expected-error 2{{extension}} expected-note {{here}} | 
|  | #else | 
|  | E::E(const E&) = default; // expected-error {{would delete}} expected-note@-5{{inaccessible copy constructor}} | 
|  | E::E(E&&) = default; // expected-error {{would delete}} expected-note@-6{{inaccessible move constructor}} | 
|  | #endif | 
|  | struct F : virtual A { F(const F&); F(F&&); }; // expected-error 0-1{{extension}} | 
|  | #if __cplusplus < 201103L | 
|  | F::F(const F&) = default; // expected-error {{private copy constructor}} expected-error {{extension}} expected-note {{here}} | 
|  | F::F(F&&) = default; // expected-error {{private move constructor}} expected-error 2{{extension}} expected-note {{here}} | 
|  | #else | 
|  | F::F(const F&) = default; // expected-error {{would delete}} expected-note@-5{{inaccessible copy constructor}} | 
|  | F::F(F&&) = default; // expected-error {{would delete}} expected-note@-6{{inaccessible move constructor}} | 
|  | #endif | 
|  | } | 
|  |  | 
|  | // assignment case is superseded by dr2180 | 
|  | } | 
|  |  | 
|  | namespace dr1672 { // dr1672: 7 | 
|  | struct Empty {}; | 
|  | struct A : Empty {}; | 
|  | struct B { Empty e; }; | 
|  | struct C : A { B b; int n; }; | 
|  | struct D : A { int n; B b; }; | 
|  |  | 
|  | static_assert(!__is_standard_layout(C), ""); | 
|  | static_assert(__is_standard_layout(D), ""); | 
|  |  | 
|  | struct E { B b; int n; }; | 
|  | struct F { int n; B b; }; | 
|  | union G { B b; int n; }; | 
|  | union H { int n; B b; }; | 
|  |  | 
|  | struct X {}; | 
|  | template<typename T> struct Y : X, A { T t; }; | 
|  |  | 
|  | static_assert(!__is_standard_layout(Y<E>), ""); | 
|  | static_assert(__is_standard_layout(Y<F>), ""); | 
|  | static_assert(!__is_standard_layout(Y<G>), ""); | 
|  | static_assert(!__is_standard_layout(Y<H>), ""); | 
|  | static_assert(!__is_standard_layout(Y<X>), ""); | 
|  | } | 
|  |  | 
|  | namespace dr1687 { // dr1687: 7 | 
|  | template<typename T> struct To { | 
|  | operator T(); // expected-note 2{{first operand was implicitly converted to type 'int *'}} | 
|  | // expected-note@-1 {{second operand was implicitly converted to type 'double'}} | 
|  | #if __cplusplus > 201703L | 
|  | // expected-note@-3 2{{operand was implicitly converted to type 'dr1687::E}} | 
|  | #endif | 
|  | }; | 
|  |  | 
|  | int *a = To<int*>() + 100.0; // expected-error {{invalid operands to binary expression ('To<int *>' and 'double')}} | 
|  | int *b = To<int*>() + To<double>(); // expected-error {{invalid operands to binary expression ('To<int *>' and 'To<double>')}} | 
|  |  | 
|  | #if __cplusplus > 201703L | 
|  | enum E1 {}; | 
|  | enum E2 {}; | 
|  | auto c = To<E1>() <=> To<E2>(); // expected-error {{invalid operands to binary expression ('To<dr1687::E1>' and 'To<dr1687::E2>')}} | 
|  | #endif | 
|  | } | 
|  |  | 
|  | namespace dr1696 { // dr1696: 7 | 
|  | namespace std_examples { | 
|  | #if __cplusplus >= 201402L | 
|  | extern struct A a; | 
|  | struct A { | 
|  | const A &x = { A{a, a} }; | 
|  | const A &y = { A{} }; // expected-error {{default member initializer for 'y' needed within definition of enclosing class 'A' outside of member functions}} expected-note {{here}} | 
|  | }; | 
|  | A a{a, a}; | 
|  | #endif | 
|  | } | 
|  |  | 
|  | struct A { A(); ~A(); }; | 
|  | #if __cplusplus >= 201103L | 
|  | struct B { | 
|  | A &&a; // expected-note {{declared here}} | 
|  | B() : a{} {} // expected-error {{reference member 'a' binds to a temporary object whose lifetime would be shorter than the lifetime of the constructed object}} | 
|  | } b; | 
|  | #endif | 
|  |  | 
|  | struct C { | 
|  | C(); | 
|  | const A &a; // expected-note {{declared here}} | 
|  | }; | 
|  | C::C() : a(A()) {} // expected-error {{reference member 'a' binds to a temporary object whose lifetime would be shorter than the lifetime of the constructed object}} | 
|  |  | 
|  | #if __cplusplus >= 201103L | 
|  | // This is OK in C++14 onwards, per DR1815, though we don't support that yet: | 
|  | //   D1 d1 = {}; | 
|  | // is equivalent to | 
|  | //   D1 d1 = {A()}; | 
|  | // ... which lifetime-extends the A temporary. | 
|  | struct D1 { | 
|  | #if __cplusplus < 201402L | 
|  | // expected-error@-2 {{binds to a temporary}} | 
|  | #endif | 
|  | const A &a = A(); // expected-note {{default member init}} | 
|  | }; | 
|  | D1 d1 = {}; | 
|  | #if __cplusplus < 201402L | 
|  | // expected-note@-2 {{first required here}} | 
|  | #else | 
|  | // expected-warning-re@-4 {{sorry, lifetime extension {{.*}} not supported}} | 
|  | #endif | 
|  |  | 
|  | struct D2 { | 
|  | const A &a = A(); // expected-note {{default member init}} | 
|  | D2() {} // expected-error {{binds to a temporary}} | 
|  | }; | 
|  |  | 
|  | struct D3 { // expected-error {{binds to a temporary}} | 
|  | const A &a = A(); // expected-note {{default member init}} | 
|  | }; | 
|  | D3 d3; // expected-note {{first required here}} | 
|  |  | 
|  | struct haslist1 { | 
|  | std::initializer_list<int> il; // expected-note {{'std::initializer_list' member}} | 
|  | haslist1(int i) : il{i, 2, 3} {} // expected-error {{backing array for 'std::initializer_list' member 'il' is a temporary object}} | 
|  | }; | 
|  |  | 
|  | struct haslist2 { | 
|  | std::initializer_list<int> il; // expected-note {{'std::initializer_list' member}} | 
|  | haslist2(); | 
|  | }; | 
|  | haslist2::haslist2() : il{1, 2} {} // expected-error {{backing array for 'std::initializer_list' member 'il' is a temporary object}} | 
|  |  | 
|  | struct haslist3 { | 
|  | std::initializer_list<int> il = {1, 2, 3}; | 
|  | }; | 
|  |  | 
|  | struct haslist4 { // expected-error {{backing array for 'std::initializer_list' member 'il' is a temporary object}} | 
|  | std::initializer_list<int> il = {1, 2, 3}; // expected-note {{default member initializer}} | 
|  | }; | 
|  | haslist4 hl4; // expected-note {{in implicit default constructor}} | 
|  |  | 
|  | struct haslist5 { | 
|  | std::initializer_list<int> il = {1, 2, 3}; // expected-note {{default member initializer}} | 
|  | haslist5() {} // expected-error {{backing array for 'std::initializer_list' member 'il' is a temporary object}} | 
|  | }; | 
|  | #endif | 
|  | } |