2015-10-22 19:31:44 +08:00
// RUN: %check_clang_tidy %s google-explicit-constructor %t
2014-11-27 19:11:47 +08:00
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_ ; }
} ;
}
struct A {
A ( ) { }
A ( int x , int y ) { }
explicit A ( void * x ) { }
explicit A ( void * x , void * y ) { }
2016-12-23 23:03:12 +08:00
explicit operator bool ( ) const { return true ; }
2014-11-27 19:11:47 +08:00
2017-03-17 16:40:07 +08:00
operator double ( ) const = delete ;
2014-11-27 19:11:47 +08:00
explicit A ( const A & a ) { }
// CHECK-MESSAGES: :[[@LINE-1]]:12: warning: copy constructor should not be declared explicit [google-explicit-constructor]
// CHECK-FIXES: {{^ }}A(const A& a) {}
2017-04-19 01:26:00 +08:00
A ( int x1 ) ;
2015-11-28 10:25:02 +08:00
// CHECK-MESSAGES: :[[@LINE-1]]:3: warning: single-argument constructors must be marked explicit to avoid unintentional implicit conversions [google-explicit-constructor]
2017-04-19 01:26:00 +08:00
// CHECK-FIXES: {{^ }}explicit A(int x1);
2014-11-27 19:11:47 +08:00
A ( double x2 , double y = 3.14 ) { }
2015-11-28 10:25:02 +08:00
// CHECK-MESSAGES: :[[@LINE-1]]:3: warning: constructors that are callable with a single argument must be marked explicit to avoid unintentional implicit conversions [google-explicit-constructor]
2014-11-27 19:11:47 +08:00
// CHECK-FIXES: {{^ }}explicit A(double x2, double y = 3.14) {}
2015-04-01 00:24:44 +08:00
template < typename . . . T >
A ( T & & . . . args ) ;
// CHECK-MESSAGES: :[[@LINE-1]]:3: warning: constructors that are callable with a single argument
// CHECK-FIXES: {{^ }}explicit A(T&&... args);
2014-11-27 19:11:47 +08:00
} ;
2014-11-29 01:31:51 +08:00
2017-04-19 01:26:00 +08:00
inline A : : A ( int x1 ) { }
2014-11-29 01:31:51 +08:00
struct B {
B ( std : : initializer_list < int > list1 ) { }
B ( const std : : initializer_list < unsigned > & list2 ) { }
B ( std : : initializer_list < unsigned > & & list3 ) { }
2016-12-23 23:03:12 +08:00
operator bool ( ) const { return true ; }
// CHECK-MESSAGES: :[[@LINE-1]]:3: warning: 'operator bool' must be marked explicit to avoid unintentional implicit conversions [google-explicit-constructor]
// CHECK-FIXES: {{^ }}explicit operator bool() const { return true; }
2017-04-19 01:26:00 +08:00
operator double ( ) const ;
// CHECK-MESSAGES: :[[@LINE-1]]:3: warning: 'operator double' must be marked explicit to avoid unintentional implicit conversions [google-explicit-constructor]
// CHECK-FIXES: {{^ }}explicit operator double() const;
2014-11-29 01:31:51 +08:00
explicit B ( : : std : : initializer_list < double > list4 ) { }
// CHECK-MESSAGES: :[[@LINE-1]]:12: warning: initializer-list constructor should not be declared explicit [google-explicit-constructor]
// CHECK-FIXES: {{^ }}B(::std::initializer_list<double> list4) {}
explicit B ( const : : std : : initializer_list < char > & list5 ) { }
2015-02-05 20:49:07 +08:00
// CHECK-MESSAGES: :[[@LINE-1]]:12: warning: initializer-list constructor
2014-11-29 01:31:51 +08:00
// CHECK-FIXES: {{^ }}B(const ::std::initializer_list<char> &list5) {}
explicit B ( : : std : : initializer_list < char > & & list6 ) { }
2015-02-05 20:49:07 +08:00
// CHECK-MESSAGES: :[[@LINE-1]]:12: warning: initializer-list constructor
2014-11-29 01:31:51 +08:00
// CHECK-FIXES: {{^ }}B(::std::initializer_list<char> &&list6) {}
} ;
2017-04-19 01:26:00 +08:00
inline B : : operator double ( ) const { return 0.0 ; }
2016-12-28 21:48:03 +08:00
struct StructWithFnPointer {
void ( * f ) ( ) ;
} struct_with_fn_pointer = { [ ] { } } ;
2014-11-29 01:31:51 +08:00
using namespace std ;
struct C {
C ( initializer_list < int > list1 ) { }
C ( const initializer_list < unsigned > & list2 ) { }
C ( initializer_list < unsigned > & & list3 ) { }
} ;
2014-12-01 03:41:41 +08:00
2015-02-05 20:49:07 +08:00
template < typename T >
struct C2 {
C2 ( initializer_list < int > list1 ) { }
C2 ( const initializer_list < unsigned > & list2 ) { }
C2 ( initializer_list < unsigned > & & list3 ) { }
explicit C2 ( initializer_list < double > list4 ) { }
// CHECK-MESSAGES: :[[@LINE-1]]:12: warning: initializer-list constructor
// CHECK-FIXES: {{^ }}C2(initializer_list<double> list4) {}
} ;
template < typename T >
struct C3 {
C3 ( initializer_list < T > list1 ) { }
C3 ( const std : : initializer_list < T * > & list2 ) { }
C3 ( : : std : : initializer_list < T * * > & & list3 ) { }
template < typename U >
C3 ( initializer_list < U > list3 ) { }
} ;
2014-12-01 03:41:41 +08:00
struct D {
template < typename T >
explicit D ( T t ) { }
} ;
template < typename T >
struct E {
2015-02-05 20:49:07 +08:00
E ( T * pt ) { }
// CHECK-MESSAGES: :[[@LINE-1]]:3: warning: single-argument constructors
// CHECK-FIXES: {{^ }}explicit E(T *pt) {}
template < typename U >
E ( U * pu ) { }
// CHECK-MESSAGES: :[[@LINE-1]]:3: warning: single-argument constructors
// CHECK-FIXES: {{^ }}explicit E(U *pu) {}
2014-12-01 03:41:41 +08:00
explicit E ( T t ) { }
template < typename U >
explicit E ( U u ) { }
} ;
void f ( std : : initializer_list < int > list ) {
D d ( list ) ;
E < decltype ( list ) > e ( list ) ;
E < int > e2 ( list ) ;
}
2016-12-30 21:25:03 +08:00
template < typename T >
struct F { } ;
template < typename T >
struct G {
operator bool ( ) const ;
// CHECK-MESSAGES: :[[@LINE-1]]:3: warning: 'operator bool' must be marked
// CHECK-FIXES: {{^}} explicit operator bool() const;
operator F < T > ( ) const ;
// CHECK-MESSAGES: :[[@LINE-1]]:3: warning: 'operator F<type-parameter-0-0>' must be marked
// CHECK-FIXES: {{^}} explicit operator F<T>() const;
template < typename U >
operator F < U > * ( ) const ;
// CHECK-MESSAGES: :[[@LINE-1]]:3: warning: 'operator F<type-parameter-1-0> *' must be marked
// CHECK-FIXES: {{^}} explicit operator F<U>*() const;
} ;
void f2 ( ) {
G < int > a ;
( void ) ( F < int > ) a ;
if ( a ) { }
( void ) ( F < int > * ) a ;
( void ) ( F < int * > * ) a ;
G < double > b ;
( void ) ( F < double > ) b ;
if ( b ) { }
( void ) ( F < double > * ) b ;
( void ) ( F < double * > * ) b ;
}
2016-12-30 23:15:14 +08:00
# define DEFINE_STRUCT_WITH_OPERATOR_BOOL(name) \
struct name { \
operator bool ( ) const ; \
}
DEFINE_STRUCT_WITH_OPERATOR_BOOL ( H ) ;