You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
127 lines
4.9 KiB
127 lines
4.9 KiB
4 months ago
|
// RUN: %clang_cc1 -std=c++1z -verify %s
|
||
|
// RUN: %clang_cc1 -std=c++2a -verify %s
|
||
|
|
||
|
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
|
||
|
// with value 0. (This is no longer an issue since empty pack expansions don't
|
||
|
// produce integers any more.)
|
||
|
template<int ...N> void null_ptr() {
|
||
|
void *p = (N || ...); // expected-error {{rvalue of type 'bool'}}
|
||
|
void *q = (N , ...); // expected-error {{rvalue of type 'void'}}
|
||
|
}
|
||
|
template void null_ptr<>(); // expected-note {{in instantiation of}}
|
||
|
|
||
|
template<int ...N> void bad_empty() {
|
||
|
(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}}
|
||
|
(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}}
|
||
|
void ((k = ... = N));
|
||
|
(void) (k = ... = N);
|
||
|
}
|
||
|
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);
|
||
|
|
||
|
#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}}
|
||
|
|
||
|
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}}
|
||
|
}
|
||
|
|
||
|
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}}
|
||
|
}
|