2014-11-08 13:07:16 +08:00
// RUN: %clang_cc1 -std=c++1z -verify %s
2017-12-14 23:16:18 +08:00
// RUN: %clang_cc1 -std=c++2a -verify %s
2014-11-08 13:07:16 +08:00
template < typename . . . T > constexpr auto sum ( T . . . t ) { return ( . . . + t ) ; }
template < typename . . . T > constexpr auto product ( T . . . t ) { return ( t * . . . ) ; }
template < typename . . . T > constexpr auto all ( T . . . t ) { return ( true & & . . . & & t ) ; }
template < typename . . . T > constexpr auto dumb ( T . . . t ) { return ( false & & . . . & & t ) ; }
static_assert ( sum ( 1 , 2 , 3 , 4 , 5 ) = = 15 ) ;
static_assert ( product ( 1 , 2 , 3 , 4 , 5 ) = = 120 ) ;
static_assert ( ! all ( true , true , false , true , false ) ) ;
static_assert ( all ( true , true , true , true , true ) ) ;
static_assert ( ! dumb ( true , true , true , true , true ) ) ;
struct S {
int a , b , c , d , e ;
} ;
template < typename . . . T > constexpr auto increment_all ( T & . . . t ) {
( + + t , . . . ) ;
}
constexpr bool check ( ) {
S s = { 1 , 2 , 3 , 4 , 5 } ;
increment_all ( s . a , s . b , s . c , s . d , s . e ) ;
return s . a = = 2 & & s . b = = 3 & & s . c = = 4 & & s . d = = 5 & & s . e = = 6 ;
}
static_assert ( check ( ) ) ;
template < int . . . N > void empty ( ) {
static_assert ( ( N | | . . . ) = = false ) ;
static_assert ( ( N & & . . . ) = = true ) ;
( N , . . . ) ;
}
template void empty < > ( ) ;
// An empty fold-expression isn't a null pointer just because it's an integer
2016-03-05 05:27:21 +08:00
// with value 0. (This is no longer an issue since empty pack expansions don't
// produce integers any more.)
2014-11-08 13:07:16 +08:00
template < int . . . N > void null_ptr ( ) {
2016-03-05 05:27:21 +08:00
void * p = ( N | | . . . ) ; // expected-error {{rvalue of type 'bool'}}
void * q = ( N , . . . ) ; // expected-error {{rvalue of type 'void'}}
2014-11-08 13:07:16 +08:00
}
template void null_ptr < > ( ) ; // expected-note {{in instantiation of}}
template < int . . . N > void bad_empty ( ) {
2016-03-05 05:27:21 +08:00
( N + . . . ) ; // expected-error {{empty expansion for operator '+' with no fallback}}
( N * . . . ) ; // expected-error {{empty expansion for operator '*' with no fallback}}
( N | . . . ) ; // expected-error {{empty expansion for operator '|' with no fallback}}
( N & . . . ) ; // expected-error {{empty expansion for operator '&' with no fallback}}
2014-11-08 13:07:16 +08:00
( N - . . . ) ; // expected-error {{empty expansion for operator '-' with no fallback}}
( N / . . . ) ; // expected-error {{empty expansion for operator '/' with no fallback}}
( N % . . . ) ; // expected-error {{empty expansion for operator '%' with no fallback}}
( N = . . . ) ; // expected-error {{empty expansion for operator '=' with no fallback}}
}
template void bad_empty < > ( ) ; // expected-note {{in instantiation of}}
template < int . . . N > void empty_with_base ( ) {
extern int k ;
( k = . . . = N ) ; // expected-warning{{unused}}
void ( k = . . . = N ) ; // expected-error {{expected ')'}} expected-note {{to match}}
2014-11-11 11:28:50 +08:00
void ( ( k = . . . = N ) ) ;
( void ) ( k = . . . = N ) ;
2014-11-08 13:07:16 +08:00
}
template void empty_with_base < > ( ) ; // expected-note {{in instantiation of}}
template void empty_with_base < 1 > ( ) ;
struct A {
struct B {
struct C {
struct D {
int e ;
} d ;
} c ;
} b ;
} a ;
template < typename T , typename . . . Ts > constexpr decltype ( auto ) apply ( T & t , Ts . . . ts ) {
return ( t . * . . . . * ts ) ;
}
static_assert ( & apply ( a , & A : : b , & A : : B : : c , & A : : B : : C : : d , & A : : B : : C : : D : : e ) = = & a . b . c . d . e ) ;
2017-12-14 23:16:18 +08:00
# if __cplusplus > 201703L
// The <=> operator is unique among binary operators in not being a
// fold-operator.
// FIXME: This diagnostic is not great.
template < typename . . . T > constexpr auto spaceship1 ( T . . . t ) { return ( t < = > . . . ) ; } // expected-error {{expected expression}}
template < typename . . . T > constexpr auto spaceship2 ( T . . . t ) { return ( . . . < = > t ) ; } // expected-error {{expected expression}}
template < typename . . . T > constexpr auto spaceship3 ( T . . . t ) { return ( t < = > . . . < = > 0 ) ; } // expected-error {{expected expression}}
# endif
// The GNU binary conditional operator ?: is not recognized as a fold-operator.
// FIXME: Why not? This seems like it would be useful.
template < typename . . . T > constexpr auto binary_conditional1 ( T . . . t ) { return ( t ? : . . . ) ; } // expected-error {{expected expression}}
template < typename . . . T > constexpr auto binary_conditional2 ( T . . . t ) { return ( . . . ? : t ) ; } // expected-error {{expected expression}}
template < typename . . . T > constexpr auto binary_conditional3 ( T . . . t ) { return ( t ? : . . . ? : 0 ) ; } // expected-error {{expected expression}}
2019-05-13 16:31:14 +08:00
namespace PR41845 {
template < int I > struct Constant { } ;
template < int . . . Is > struct Sum {
template < int . . . Js > using type = Constant < ( ( Is + Js ) + . . . + 0 ) > ; // expected-error {{pack expansion contains parameter pack 'Js' that has a different length (1 vs. 2) from outer parameter packs}}
} ;
Sum < 1 > : : type < 1 , 2 > x ; // expected-note {{instantiation of}}
}
2020-08-07 06:57:35 +08:00
namespace PR30738 {
namespace N {
struct S { } ;
}
namespace T {
void operator + ( N : : S , N : : S ) { }
template < typename . . . Ts > void f ( ) { ( Ts { } + . . . ) ; }
}
void g ( ) { T : : f < N : : S , N : : S > ( ) ; }
template < typename T , typename . . . U > auto h ( U . . . v ) {
T operator + ( T , T ) ; // expected-note {{candidate}}
return ( v + . . . ) ; // expected-error {{invalid operands}}
}
int test_h1 = h < N : : S > ( 1 , 2 , 3 ) ;
N : : S test_h2 = h < N : : S > ( N : : S ( ) , N : : S ( ) , N : : S ( ) ) ;
int test_h3 = h < struct X > ( 1 , 2 , 3 ) ;
N : : S test_h4 = h < struct X > ( N : : S ( ) , N : : S ( ) , N : : S ( ) ) ; // expected-note {{instantiation of}}
}