2009-12-16 04:14:24 +08:00
|
|
|
// RUN: %clang_cc1 -fsyntax-only -verify %s
|
2015-11-12 03:34:47 +08:00
|
|
|
// RUN: %clang_cc1 -fsyntax-only -verify -std=c++98 %s
|
|
|
|
// RUN: %clang_cc1 -fsyntax-only -verify -std=c++11 %s
|
|
|
|
|
2009-01-16 11:02:29 +08:00
|
|
|
struct ConvToBool {
|
|
|
|
operator bool() const;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct ConvToInt {
|
|
|
|
operator int();
|
|
|
|
};
|
|
|
|
|
|
|
|
struct ExplicitConvToBool {
|
2015-11-12 03:34:47 +08:00
|
|
|
explicit operator bool();
|
|
|
|
#if __cplusplus <= 199711L // C++03 or earlier modes
|
|
|
|
// expected-warning@-2{{explicit conversion functions are a C++11 extension}}
|
|
|
|
#endif
|
2009-01-16 11:02:29 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
void test_conv_to_bool(ConvToBool ctb, ConvToInt cti, ExplicitConvToBool ecb) {
|
|
|
|
if (ctb) { }
|
|
|
|
if (cti) { }
|
|
|
|
if (ecb) { }
|
|
|
|
for (; ctb; ) { }
|
|
|
|
for (; cti; ) { }
|
|
|
|
for (; ecb; ) { }
|
|
|
|
while (ctb) { };
|
|
|
|
while (cti) { }
|
|
|
|
while (ecb) { }
|
|
|
|
do { } while (ctb);
|
|
|
|
do { } while (cti);
|
|
|
|
do { } while (ecb);
|
|
|
|
|
|
|
|
if (!ctb) { }
|
|
|
|
if (!cti) { }
|
|
|
|
if (!ecb) { }
|
|
|
|
|
|
|
|
bool b1 = !ecb;
|
|
|
|
if (ctb && ecb) { }
|
|
|
|
bool b2 = ctb && ecb;
|
|
|
|
if (ctb || ecb) { }
|
|
|
|
bool b3 = ctb || ecb;
|
|
|
|
}
|
|
|
|
|
2009-02-04 08:32:51 +08:00
|
|
|
void accepts_bool(bool) { } // expected-note{{candidate function}}
|
2009-01-16 11:02:29 +08:00
|
|
|
|
|
|
|
struct ExplicitConvToRef {
|
2015-11-12 03:34:47 +08:00
|
|
|
explicit operator int&();
|
|
|
|
#if (__cplusplus <= 199711L) // C++03 or earlier modes
|
|
|
|
// expected-warning@-2{{explicit conversion functions are a C++11 extension}}
|
|
|
|
#endif
|
2009-01-16 11:02:29 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
void test_explicit_bool(ExplicitConvToBool ecb) {
|
|
|
|
bool b1(ecb); // okay
|
2010-03-10 19:27:22 +08:00
|
|
|
bool b2 = ecb; // expected-error{{no viable conversion from 'ExplicitConvToBool' to 'bool'}}
|
2009-02-04 08:32:51 +08:00
|
|
|
accepts_bool(ecb); // expected-error{{no matching function for call to}}
|
2009-01-16 11:02:29 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void test_explicit_conv_to_ref(ExplicitConvToRef ecr) {
|
2010-03-10 19:27:22 +08:00
|
|
|
int& i1 = ecr; // expected-error{{non-const lvalue reference to type 'int' cannot bind to a value of unrelated type 'ExplicitConvToRef'}}
|
2009-01-16 11:02:29 +08:00
|
|
|
int& i2(ecr); // okay
|
|
|
|
}
|
|
|
|
|
|
|
|
struct A { };
|
|
|
|
struct B { };
|
|
|
|
struct C {
|
2015-11-12 03:34:47 +08:00
|
|
|
explicit operator A&();
|
|
|
|
#if __cplusplus <= 199711L // C++03 or earlier modes
|
|
|
|
// expected-warning@-2{{explicit conversion functions are a C++11 extension}}
|
|
|
|
#endif
|
2009-12-10 07:02:17 +08:00
|
|
|
operator B&(); // expected-note{{candidate}}
|
2009-01-16 11:02:29 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
void test_copy_init_conversions(C c) {
|
2010-03-10 19:27:22 +08:00
|
|
|
A &a = c; // expected-error{{no viable conversion from 'C' to 'A'}}
|
2012-08-17 18:12:33 +08:00
|
|
|
B &b = c; // okay
|
2009-01-16 11:02:29 +08:00
|
|
|
}
|