// RUN: %clang_cc1 -std=c++98 %s -verify=expected -fexceptions -fcxx-exceptions -pedantic-errors
// RUN: %clang_cc1 -std=c++11 %s -verify=expected -fexceptions -fcxx-exceptions -pedantic-errors
// RUN: %clang_cc1 -std=c++14 %s -verify=expected,since-cxx14 -fexceptions -fcxx-exceptions -pedantic-errors
// RUN: %clang_cc1 -std=c++17 %s -verify=expected,since-cxx14 -fexceptions -fcxx-exceptions -pedantic-errors
// RUN: %clang_cc1 -std=c++20 %s -verify=expected,since-cxx14 -fexceptions -fcxx-exceptions -pedantic-errors
// RUN: %clang_cc1 -std=c++23 %s -verify=expected,since-cxx14 -fexceptions -fcxx-exceptions -pedantic-errors
namespace std {
__extension__ typedef __SIZE_TYPE__ size_t;
template<typename T> struct initializer_list {
const T *p; size_t n;
initializer_list(const T *p, size_t n);
};
}
namespace cwg1004 { // cwg1004: 5
template<typename> struct A {};
template<typename> struct B1 {};
template<template<typename> class> struct B2 {};
template<typename X> void f(); // #cwg1004-f-1
template<template<typename> class X> void f(); // #cwg1004-f-2
template<template<typename> class X> void g(); // #cwg1004-g-1
template<typename X> void g(); // #cwg1004-g-2
struct C : A<int> {
B1<A> b1a;
B2<A> b2a;
void h() {
f<A>();
// expected-error@-1 {{call to 'f' is ambiguous}}
// expected-note@#cwg1004-f-1 {{candidate function [with X = cwg1004::A<int>]}}
// expected-note@#cwg1004-f-2 {{candidate function [with X = cwg1004::A]}}
g<A>();
// expected-error@-1 {{call to 'g' is ambiguous}}
// expected-note@#cwg1004-g-1 {{candidate function [with X = cwg1004::A]}}
// expected-note@#cwg1004-g-2 {{candidate function [with X = cwg1004::A<int>]}}
}
};
// This example (from the standard) is actually ill-formed, because
// name lookup of "T::template A" names the constructor.
template<class T, template<class> class U = T::template A> struct Third { };
// expected-error@-1 {{is a constructor name}}
// expected-note@#cwg1004-t {{in instantiation of default argument}}
Third<A<int> > t; // #cwg1004-t
}
namespace cwg1042 { // cwg1042: 3.5
#if __cplusplus >= 201402L
// C++14 added an attribute that we can test the semantics of.
using foo [[deprecated]] = int; // #cwg1042-using
foo f = 12;
// since-cxx14-warning@-1 {{'foo' is deprecated}}
// since-cxx14-note@#cwg1042-using {{'foo' has been explicitly marked deprecated here}}
#elif __cplusplus >= 201103L
// C++11 did not have any attributes that could be applied to an alias
// declaration, so the best we can test is that we accept an empty attribute
// list in this mode.
using foo [[]] = int;
#endif
}
namespace cwg1048 { // cwg1048: 3.6
struct A {};
const A f();
A g();
typedef const A CA;
#if __cplusplus >= 201103L
// ok: we deduce non-const A in each case.
A &&a = [] (int n) {
while (1) switch (n) {
case 0: return f();
case 1: return g();
case 2: return A();
case 3: return CA();
}
} (0);
#endif
}
namespace cwg1054 { // cwg1054: no
// FIXME: Test is incomplete.
struct A {} volatile a;
void f() {
// FIXME: This is wrong: an lvalue-to-rvalue conversion is applied here,
// which copy-initializes a temporary from 'a'. Therefore this is
// ill-formed because A does not have a volatile copy constructor.
// (We might want to track this aspect under cwg1383 instead?)
a;
// expected-warning@-1 {{expression result unused; assign into a variable to force a volatile load}}
}
}
namespace cwg1070 { // cwg1070: 3.5
#if __cplusplus >= 201103L
struct A {
A(std::initializer_list<int>);
};
struct B {
int i;
A a;
};
B b = {1};
struct C {
std::initializer_list<int> a;
B b;
std::initializer_list<double> c;
};
C c = {};
#endif
}