368 lines
9.2 KiB
C++
368 lines
9.2 KiB
C++
|
// RUN: %clang_cc1 -std=c++0x -fsyntax-only -verify %s
|
||
|
|
||
|
struct one { char c[1]; };
|
||
|
struct two { char c[2]; };
|
||
|
|
||
|
namespace std {
|
||
|
typedef decltype(sizeof(int)) size_t;
|
||
|
|
||
|
// libc++'s implementation
|
||
|
template <class _E>
|
||
|
class initializer_list
|
||
|
{
|
||
|
const _E* __begin_;
|
||
|
size_t __size_;
|
||
|
|
||
|
initializer_list(const _E* __b, size_t __s)
|
||
|
: __begin_(__b),
|
||
|
__size_(__s)
|
||
|
{}
|
||
|
|
||
|
public:
|
||
|
typedef _E value_type;
|
||
|
typedef const _E& reference;
|
||
|
typedef const _E& const_reference;
|
||
|
typedef size_t size_type;
|
||
|
|
||
|
typedef const _E* iterator;
|
||
|
typedef const _E* const_iterator;
|
||
|
|
||
|
initializer_list() : __begin_(nullptr), __size_(0) {}
|
||
|
|
||
|
size_t size() const {return __size_;}
|
||
|
const _E* begin() const {return __begin_;}
|
||
|
const _E* end() const {return __begin_ + __size_;}
|
||
|
};
|
||
|
}
|
||
|
|
||
|
namespace objects {
|
||
|
|
||
|
struct X1 { X1(int); };
|
||
|
struct X2 { explicit X2(int); }; // expected-note {{constructor declared here}}
|
||
|
|
||
|
template <int N>
|
||
|
struct A {
|
||
|
A() { static_assert(N == 0, ""); }
|
||
|
A(int, double) { static_assert(N == 1, ""); }
|
||
|
};
|
||
|
|
||
|
template <int N>
|
||
|
struct F {
|
||
|
F() { static_assert(N == 0, ""); }
|
||
|
F(int, double) { static_assert(N == 1, ""); }
|
||
|
F(std::initializer_list<int>) { static_assert(N == 3, ""); }
|
||
|
};
|
||
|
|
||
|
template <int N>
|
||
|
struct D {
|
||
|
D(std::initializer_list<int>) { static_assert(N == 0, ""); } // expected-note 1 {{candidate}}
|
||
|
D(std::initializer_list<double>) { static_assert(N == 1, ""); } // expected-note 1 {{candidate}}
|
||
|
};
|
||
|
|
||
|
template <int N>
|
||
|
struct E {
|
||
|
E(int, int) { static_assert(N == 0, ""); }
|
||
|
E(X1, int) { static_assert(N == 1, ""); }
|
||
|
};
|
||
|
|
||
|
void overload_resolution() {
|
||
|
{ A<0> a{}; }
|
||
|
{ A<0> a = {}; }
|
||
|
{ A<1> a{1, 1.0}; }
|
||
|
{ A<1> a = {1, 1.0}; }
|
||
|
|
||
|
{ F<0> f{}; }
|
||
|
{ F<0> f = {}; }
|
||
|
// Narrowing conversions don't affect viability. The next two choose
|
||
|
// the initializer_list constructor.
|
||
|
{ F<3> f{1, 1.0}; } // expected-error {{type 'double' cannot be narrowed to 'int' in initializer list}} expected-note {{override}}
|
||
|
{ F<3> f = {1, 1.0}; } // expected-error {{type 'double' cannot be narrowed to 'int' in initializer list}} expected-note {{override}}
|
||
|
{ F<3> f{1, 2, 3, 4, 5, 6, 7, 8}; }
|
||
|
{ F<3> f = {1, 2, 3, 4, 5, 6, 7, 8}; }
|
||
|
{ F<3> f{1, 2, 3, 4, 5, 6, 7, 8}; }
|
||
|
{ F<3> f{1, 2}; }
|
||
|
|
||
|
{ D<0> d{1, 2, 3}; }
|
||
|
{ D<1> d{1.0, 2.0, 3.0}; }
|
||
|
{ D<-1> d{1, 2.0}; } // expected-error {{ambiguous}}
|
||
|
|
||
|
{ E<0> e{1, 2}; }
|
||
|
}
|
||
|
|
||
|
void explicit_implicit() {
|
||
|
{ X1 x{0}; }
|
||
|
{ X1 x = {0}; }
|
||
|
{ X2 x{0}; }
|
||
|
{ X2 x = {0}; } // expected-error {{constructor is explicit}}
|
||
|
}
|
||
|
|
||
|
struct C {
|
||
|
C();
|
||
|
C(int, double);
|
||
|
C(int, int);
|
||
|
|
||
|
int operator[](C);
|
||
|
};
|
||
|
|
||
|
C function_call() {
|
||
|
void takes_C(C);
|
||
|
takes_C({1, 1.0});
|
||
|
|
||
|
C c;
|
||
|
c[{1, 1.0}];
|
||
|
|
||
|
return {1, 1.0};
|
||
|
}
|
||
|
|
||
|
void inline_init() {
|
||
|
(void) C{1, 1.0};
|
||
|
(void) new C{1, 1.0};
|
||
|
(void) A<1>{1, 1.0};
|
||
|
(void) new A<1>{1, 1.0};
|
||
|
}
|
||
|
|
||
|
struct B { // expected-note 2 {{candidate constructor}}
|
||
|
B(C, int, C); // expected-note {{candidate constructor not viable: cannot convert initializer list argument to 'objects::C'}}
|
||
|
};
|
||
|
|
||
|
void nested_init() {
|
||
|
B b1{{1, 1.0}, 2, {3, 4}};
|
||
|
B b2{{1, 1.0, 4}, 2, {3, 4}}; // expected-error {{no matching constructor for initialization of 'objects::B'}}
|
||
|
}
|
||
|
|
||
|
void overloaded_call() {
|
||
|
one ov1(B); // expected-note {{not viable: cannot convert initializer list}}
|
||
|
two ov1(C); // expected-note {{not viable: cannot convert initializer list}}
|
||
|
|
||
|
static_assert(sizeof(ov1({})) == sizeof(two), "bad overload");
|
||
|
static_assert(sizeof(ov1({1, 2})) == sizeof(two), "bad overload");
|
||
|
static_assert(sizeof(ov1({{1, 1.0}, 2, {3, 4}})) == sizeof(one), "bad overload");
|
||
|
|
||
|
ov1({1}); // expected-error {{no matching function}}
|
||
|
|
||
|
one ov2(int);
|
||
|
two ov2(F<3>);
|
||
|
static_assert(sizeof(ov2({1})) == sizeof(one), "bad overload"); // list -> int ranks as identity
|
||
|
static_assert(sizeof(ov2({1, 2, 3})) == sizeof(two), "bad overload"); // list -> F only viable
|
||
|
}
|
||
|
|
||
|
struct G { // expected-note 6 {{not viable}}
|
||
|
// This is not an initializer-list constructor.
|
||
|
template<typename ...T>
|
||
|
G(std::initializer_list<int>, T ...); // expected-note 3 {{not viable}}
|
||
|
};
|
||
|
|
||
|
struct H { // expected-note 6 {{not viable}}
|
||
|
explicit H(int, int); // expected-note 3 {{not viable}} expected-note {{declared here}}
|
||
|
H(int, void*); // expected-note 3 {{not viable}}
|
||
|
};
|
||
|
|
||
|
void edge_cases() {
|
||
|
// invalid (the first phase only considers init-list ctors)
|
||
|
// (for the second phase, no constructor is viable)
|
||
|
G g1{1, 2, 3}; // expected-error {{no matching constructor}}
|
||
|
(void) new G{1, 2, 3}; // expected-error {{no matching constructor}}
|
||
|
(void) G{1, 2, 3} // expected-error {{no matching constructor}}
|
||
|
|
||
|
// valid (T deduced to <>).
|
||
|
G g2({1, 2, 3});
|
||
|
(void) new G({1, 2, 3});
|
||
|
(void) G({1, 2, 3});
|
||
|
|
||
|
// invalid
|
||
|
H h1({1, 2}); // expected-error {{no matching constructor}}
|
||
|
(void) new H({1, 2}); // expected-error {{no matching constructor}}
|
||
|
// FIXME: Bad diagnostic, mentions void type instead of init list.
|
||
|
(void) H({1, 2}); // expected-error {{no matching conversion}}
|
||
|
|
||
|
// valid (by copy constructor).
|
||
|
H h2({1, nullptr});
|
||
|
(void) new H({1, nullptr});
|
||
|
(void) H({1, nullptr});
|
||
|
|
||
|
// valid
|
||
|
H h3{1, 2};
|
||
|
(void) new H{1, 2};
|
||
|
(void) H{1, 2};
|
||
|
}
|
||
|
|
||
|
struct memberinit {
|
||
|
H h1{1, nullptr};
|
||
|
H h2 = {1, nullptr};
|
||
|
H h3{1, 1};
|
||
|
H h4 = {1, 1}; // expected-error {{constructor is explicit}}
|
||
|
};
|
||
|
}
|
||
|
|
||
|
namespace PR12092 {
|
||
|
|
||
|
struct S {
|
||
|
S(const char*);
|
||
|
};
|
||
|
struct V {
|
||
|
template<typename T> V(T, T);
|
||
|
void f(std::initializer_list<S>);
|
||
|
void f(const V &);
|
||
|
};
|
||
|
|
||
|
void g() {
|
||
|
extern V s;
|
||
|
s.f({"foo", "bar"});
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
namespace PR12117 {
|
||
|
struct A { A(int); };
|
||
|
struct B { B(A); } b{{0}};
|
||
|
struct C { C(int); } c{0};
|
||
|
}
|
||
|
|
||
|
namespace PR12167 {
|
||
|
template<int N> struct string {};
|
||
|
|
||
|
struct X {
|
||
|
X(const char v);
|
||
|
template<typename T> bool operator()(T) const;
|
||
|
};
|
||
|
|
||
|
template<int N, class Comparator> bool g(const string<N>& s, Comparator cmp) {
|
||
|
return cmp(s);
|
||
|
}
|
||
|
template<int N> bool f(const string<N> &s) {
|
||
|
return g(s, X{'x'});
|
||
|
}
|
||
|
|
||
|
bool s = f(string<1>());
|
||
|
}
|
||
|
|
||
|
namespace PR12257_PR12241 {
|
||
|
struct command_pair
|
||
|
{
|
||
|
command_pair(int, int);
|
||
|
};
|
||
|
|
||
|
struct command_map
|
||
|
{
|
||
|
command_map(std::initializer_list<command_pair>);
|
||
|
};
|
||
|
|
||
|
struct generator_pair
|
||
|
{
|
||
|
generator_pair(const command_map);
|
||
|
};
|
||
|
|
||
|
// 5 levels: init list, gen_pair, command_map, init list, command_pair
|
||
|
const std::initializer_list<generator_pair> x = {{{{{3, 4}}}}};
|
||
|
|
||
|
// 4 levels: init list, gen_pair, command_map via init list, command_pair
|
||
|
const std::initializer_list<generator_pair> y = {{{{1, 2}}}};
|
||
|
}
|
||
|
|
||
|
namespace PR12120 {
|
||
|
struct A { explicit A(int); A(float); }; // expected-note {{declared here}}
|
||
|
A a = { 0 }; // expected-error {{constructor is explicit}}
|
||
|
|
||
|
struct B { explicit B(short); B(long); }; // expected-note 2 {{candidate}}
|
||
|
B b = { 0 }; // expected-error {{ambiguous}}
|
||
|
}
|
||
|
|
||
|
namespace PR12498 {
|
||
|
class ArrayRef; // expected-note{{forward declaration}}
|
||
|
|
||
|
struct C {
|
||
|
void foo(const ArrayRef&); // expected-note{{passing argument to parameter here}}
|
||
|
};
|
||
|
|
||
|
static void bar(C* c)
|
||
|
{
|
||
|
c->foo({ nullptr, 1 }); // expected-error{{initialization of incomplete type 'const PR12498::ArrayRef'}}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
namespace explicit_default {
|
||
|
struct A {
|
||
|
explicit A(); // expected-note{{here}}
|
||
|
};
|
||
|
A a {}; // ok
|
||
|
// This is copy-list-initialization, and we choose an explicit constructor
|
||
|
// (even though we do so via value-initialization), so the initialization is
|
||
|
// ill-formed.
|
||
|
A b = {}; // expected-error{{chosen constructor is explicit}}
|
||
|
}
|
||
|
|
||
|
namespace init_list_default {
|
||
|
struct A {
|
||
|
A(std::initializer_list<int>);
|
||
|
};
|
||
|
A a {}; // calls initializer list constructor
|
||
|
|
||
|
struct B {
|
||
|
B();
|
||
|
B(std::initializer_list<int>) = delete;
|
||
|
};
|
||
|
B b {}; // calls default constructor
|
||
|
}
|
||
|
|
||
|
|
||
|
// PR13470, <rdar://problem/11974632>
|
||
|
namespace PR13470 {
|
||
|
struct W {
|
||
|
explicit W(int); // expected-note {{here}}
|
||
|
};
|
||
|
|
||
|
struct X {
|
||
|
X(const X&) = delete; // expected-note 3 {{here}}
|
||
|
X(int);
|
||
|
};
|
||
|
|
||
|
template<typename T, typename Fn> void call(Fn f) {
|
||
|
f({1}); // expected-error {{constructor is explicit}}
|
||
|
f(T{1}); // expected-error {{call to deleted constructor}}
|
||
|
}
|
||
|
|
||
|
void ref_w(const W &); // expected-note 2 {{not viable}}
|
||
|
void call_ref_w() {
|
||
|
ref_w({1}); // expected-error {{no matching function}}
|
||
|
ref_w(W{1});
|
||
|
call<W>(ref_w); // expected-note {{instantiation of}}
|
||
|
}
|
||
|
|
||
|
void ref_x(const X &);
|
||
|
void call_ref_x() {
|
||
|
ref_x({1});
|
||
|
ref_x(X{1});
|
||
|
call<X>(ref_x); // ok
|
||
|
}
|
||
|
|
||
|
void val_x(X); // expected-note 2 {{parameter}}
|
||
|
void call_val_x() {
|
||
|
val_x({1});
|
||
|
val_x(X{1}); // expected-error {{call to deleted constructor}}
|
||
|
call<X>(val_x); // expected-note {{instantiation of}}
|
||
|
}
|
||
|
|
||
|
template<typename T>
|
||
|
struct Y {
|
||
|
X x{1};
|
||
|
void f() { X x{1}; }
|
||
|
void h() {
|
||
|
ref_w({1}); // expected-error {{no matching function}}
|
||
|
ref_w(W{1});
|
||
|
ref_x({1});
|
||
|
ref_x(X{1});
|
||
|
val_x({1});
|
||
|
val_x(X{1}); // expected-error {{call to deleted constructor}}
|
||
|
}
|
||
|
Y() {}
|
||
|
Y(int) : x{1} {}
|
||
|
};
|
||
|
|
||
|
Y<int> yi;
|
||
|
Y<int> yi2(0);
|
||
|
void g() {
|
||
|
yi.f();
|
||
|
yi.h(); // ok, all diagnostics produced in template definition
|
||
|
}
|
||
|
}
|