| 1 | //===----------------------------------------------------------------------===// |
| 2 | // |
| 3 | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
| 4 | // See https://llvm.org/LICENSE.txt for license information. |
| 5 | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
| 6 | // |
| 7 | //===----------------------------------------------------------------------===// |
| 8 | |
| 9 | // UNSUPPORTED: c++03, c++11, c++14 |
| 10 | // type_traits |
| 11 | |
| 12 | // template<class... B> struct disjunction; // C++17 |
| 13 | // template<class... B> |
| 14 | // constexpr bool disjunction_v = disjunction<B...>::value; // C++17 |
| 15 | |
| 16 | #include <type_traits> |
| 17 | #include <cassert> |
| 18 | |
| 19 | #include "test_macros.h" |
| 20 | |
| 21 | struct True { static constexpr bool value = true; }; |
| 22 | struct False { static constexpr bool value = false; }; |
| 23 | |
| 24 | struct MySpecialTrueType { static constexpr auto value = -1; static constexpr auto MySpecial = 37; }; |
| 25 | struct MySpecialFalseType { static constexpr auto value = false; static constexpr auto MySpecial = 23; }; |
| 26 | struct MyOtherSpecialFalseType { static constexpr auto value = false; static constexpr auto MySpecial = 46; }; |
| 27 | struct HasNoValue {}; |
| 28 | struct ExplicitlyConvertibleToBool { explicit constexpr operator bool() const { return false; } }; |
| 29 | struct ValueExplicitlyConvertible { static constexpr ExplicitlyConvertibleToBool value {}; }; |
| 30 | |
| 31 | static_assert(!std::disjunction<>::value); |
| 32 | static_assert( std::disjunction<std::true_type >::value); |
| 33 | static_assert(!std::disjunction<std::false_type>::value); |
| 34 | |
| 35 | static_assert(!std::disjunction_v<>); |
| 36 | static_assert( std::disjunction_v<std::true_type >); |
| 37 | static_assert(!std::disjunction_v<std::false_type>); |
| 38 | |
| 39 | static_assert( std::disjunction<std::true_type, std::true_type >::value); |
| 40 | static_assert( std::disjunction<std::true_type, std::false_type>::value); |
| 41 | static_assert( std::disjunction<std::false_type, std::true_type >::value); |
| 42 | static_assert(!std::disjunction<std::false_type, std::false_type>::value); |
| 43 | |
| 44 | static_assert( std::disjunction_v<std::true_type, std::true_type >); |
| 45 | static_assert( std::disjunction_v<std::true_type, std::false_type>); |
| 46 | static_assert( std::disjunction_v<std::false_type, std::true_type >); |
| 47 | static_assert(!std::disjunction_v<std::false_type, std::false_type>); |
| 48 | |
| 49 | static_assert( std::disjunction<std::true_type, std::true_type, std::true_type >::value); |
| 50 | static_assert( std::disjunction<std::true_type, std::false_type, std::true_type >::value); |
| 51 | static_assert( std::disjunction<std::false_type, std::true_type, std::true_type >::value); |
| 52 | static_assert( std::disjunction<std::false_type, std::false_type, std::true_type >::value); |
| 53 | static_assert( std::disjunction<std::true_type, std::true_type, std::false_type>::value); |
| 54 | static_assert( std::disjunction<std::true_type, std::false_type, std::false_type>::value); |
| 55 | static_assert( std::disjunction<std::false_type, std::true_type, std::false_type>::value); |
| 56 | static_assert(!std::disjunction<std::false_type, std::false_type, std::false_type>::value); |
| 57 | |
| 58 | static_assert( std::disjunction_v<std::true_type, std::true_type, std::true_type >); |
| 59 | static_assert( std::disjunction_v<std::true_type, std::false_type, std::true_type >); |
| 60 | static_assert( std::disjunction_v<std::false_type, std::true_type, std::true_type >); |
| 61 | static_assert( std::disjunction_v<std::false_type, std::false_type, std::true_type >); |
| 62 | static_assert( std::disjunction_v<std::true_type, std::true_type, std::false_type>); |
| 63 | static_assert( std::disjunction_v<std::true_type, std::false_type, std::false_type>); |
| 64 | static_assert( std::disjunction_v<std::false_type, std::true_type, std::false_type>); |
| 65 | static_assert(!std::disjunction_v<std::false_type, std::false_type, std::false_type>); |
| 66 | |
| 67 | static_assert ( std::disjunction<True >::value, "" ); |
| 68 | static_assert (!std::disjunction<False>::value, "" ); |
| 69 | |
| 70 | static_assert ( std::disjunction_v<True >, "" ); |
| 71 | static_assert (!std::disjunction_v<False>, "" ); |
| 72 | |
| 73 | static_assert(std::is_base_of_v<MySpecialFalseType, std::disjunction<MyOtherSpecialFalseType, MySpecialFalseType>>); |
| 74 | static_assert(std::is_base_of_v<MyOtherSpecialFalseType, std::disjunction<MySpecialFalseType, MyOtherSpecialFalseType>>); |
| 75 | static_assert(std::is_base_of_v<MySpecialTrueType, std::disjunction<MySpecialTrueType, MyOtherSpecialFalseType>>); |
| 76 | static_assert(std::is_base_of_v<MySpecialTrueType, std::disjunction<MyOtherSpecialFalseType, MySpecialTrueType>>); |
| 77 | |
| 78 | static_assert(std::is_base_of_v<std::true_type, std::disjunction<std::true_type, HasNoValue>>); |
| 79 | |
| 80 | static_assert(std::disjunction<std::true_type, HasNoValue>::value); |
| 81 | static_assert(std::disjunction_v<std::true_type, HasNoValue>); |
| 82 | |
| 83 | // Also check the case where HasNoValue is not the last in the list (https://llvm.org/PR584900). |
| 84 | static_assert(std::disjunction<std::true_type, HasNoValue, std::true_type>::value); |
| 85 | static_assert(std::disjunction_v<std::true_type, HasNoValue, std::true_type>); |
| 86 | |
| 87 | static_assert(std::disjunction<std::true_type, HasNoValue, std::false_type>::value); |
| 88 | static_assert(std::disjunction_v<std::true_type, HasNoValue, std::false_type>); |
| 89 | |
| 90 | static_assert(std::disjunction<MySpecialTrueType>::value == -1); |
| 91 | static_assert(std::disjunction_v<MySpecialTrueType>); |
| 92 | |
| 93 | static_assert(std::is_base_of_v<ValueExplicitlyConvertible, std::disjunction<ValueExplicitlyConvertible>>); |
| 94 | static_assert(std::disjunction_v<ValueExplicitlyConvertible, std::true_type>); |
| 95 | |