Skip to content

Commit b0ceab2

Browse files
author
Kim Barrett
committedAug 4, 2020
8250652: Add logical operations on types
Add stand-ins for C++17 logical operations on types. Reviewed-by: eosterlund, tschatzl
1 parent 39616b4 commit b0ceab2

File tree

2 files changed

+155
-0
lines changed

2 files changed

+155
-0
lines changed
 
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,64 @@
1+
/*
2+
* Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*
23+
*/
24+
25+
#ifndef SHARE_METAPROGRAMMING_LOGICAL_HPP
26+
#define SHARE_METAPROGRAMMING_LOGICAL_HPP
27+
28+
// Stand-ins for C++17 logical operations on types.
29+
30+
#include <type_traits>
31+
32+
// Stand-in for C++17 std::bool_constant<value>.
33+
template<bool Value>
34+
using BoolConstant = std::integral_constant<bool, Value>;
35+
36+
// Stand-in for C++17 std::conjunction<T...>
37+
template<typename... T>
38+
struct Conjunction : public std::true_type {};
39+
40+
template<typename T1>
41+
struct Conjunction<T1> : public T1 {};
42+
43+
template<typename T1, typename... T>
44+
struct Conjunction<T1, T...> :
45+
public std::conditional_t<bool(T1::value), Conjunction<T...>, T1>
46+
{};
47+
48+
// Stand-in for C++17 std::disjunction<T...>.
49+
template<typename... T>
50+
struct Disjunction : public std::false_type {};
51+
52+
template<typename T1>
53+
struct Disjunction<T1> : public T1 {};
54+
55+
template<typename T1, typename... T>
56+
struct Disjunction<T1, T...> :
57+
public std::conditional_t<bool(T1::value), T1, Disjunction<T...>>
58+
{};
59+
60+
// Stand-in for C++17 std::negation<T>.
61+
template<typename T>
62+
using Negation = BoolConstant<!bool(T::value)>;
63+
64+
#endif // SHARE_METAPROGRAMMING_LOGICAL_HPP
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,91 @@
1+
/*
2+
* Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*
23+
*/
24+
25+
#include "precompiled.hpp"
26+
#include "metaprogramming/logical.hpp"
27+
#include <type_traits>
28+
29+
class TestBoolConstant {
30+
static_assert(BoolConstant<true>::value, "true");
31+
static_assert(!BoolConstant<false>::value, "false");
32+
};
33+
34+
class TestConjunction {
35+
class A : public std::true_type {};
36+
class B : public std::true_type {};
37+
class C : public std::false_type {};
38+
class D : public std::false_type {};
39+
40+
static_assert(Conjunction<>::value, "nullary value");
41+
42+
static_assert(Conjunction<A>::value, "true value");
43+
static_assert(std::is_base_of<A, Conjunction<A>>::value, "true type");
44+
45+
static_assert(!Conjunction<C>::value, "false value");
46+
static_assert(std::is_base_of<C, Conjunction<C>>::value, "false type");
47+
48+
static_assert(Conjunction<A, B>::value, "true/true value");
49+
static_assert(std::is_base_of<B, Conjunction<A, B>>::value, "true/true type");
50+
51+
static_assert(!Conjunction<A, C>::value, "true/false value");
52+
static_assert(std::is_base_of<C, Conjunction<A, C>>::value, "true/false type");
53+
54+
static_assert(!Conjunction<C, A>::value, "false/true value");
55+
static_assert(std::is_base_of<C, Conjunction<C, A>>::value, "false/true type");
56+
57+
static_assert(!Conjunction<C, D>::value, "false/false value");
58+
static_assert(std::is_base_of<C, Conjunction<C, D>>::value, "false/false type");
59+
};
60+
61+
class TestDisjunction {
62+
class A : public std::true_type {};
63+
class B : public std::true_type {};
64+
class C : public std::false_type {};
65+
class D : public std::false_type {};
66+
67+
static_assert(!Disjunction<>::value, "nullary value");
68+
69+
static_assert(Disjunction<A>::value, "true value");
70+
static_assert(std::is_base_of<A, Disjunction<A>>::value, "true type");
71+
72+
static_assert(!Disjunction<C>::value, "false value");
73+
static_assert(std::is_base_of<C, Disjunction<C>>::value, "false type");
74+
75+
static_assert(Disjunction<A, B>::value, "true/true value");
76+
static_assert(std::is_base_of<A, Disjunction<A, B>>::value, "true/true type");
77+
78+
static_assert(Disjunction<A, C>::value, "true/false value");
79+
static_assert(std::is_base_of<A, Disjunction<A, C>>::value, "true/false type");
80+
81+
static_assert(Disjunction<C, A>::value, "false/true value");
82+
static_assert(std::is_base_of<A, Disjunction<C, A>>::value, "false/true type");
83+
84+
static_assert(!Disjunction<C, D>::value, "false/false value");
85+
static_assert(std::is_base_of<D, Disjunction<C, D>>::value, "false/false type");
86+
};
87+
88+
class TestNegation {
89+
static_assert(Negation<std::false_type>::value, "false -> true");
90+
static_assert(!Negation<std::true_type>::value, "true -> false");
91+
};

0 commit comments

Comments
 (0)
Please sign in to comment.