-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathEnumFlag.h
114 lines (101 loc) · 4.14 KB
/
EnumFlag.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
// (C) Copyright 2015 Just Software Solutions Ltd
//
// Distributed under the Boost Software License, Version 1.0.
//
// Boost Software License - Version 1.0 - August 17th, 2003
//
// Permission is hereby granted, free of charge, to any person or
// organization obtaining a copy of the software and accompanying
// documentation covered by this license (the "Software") to use,
// reproduce, display, distribute, execute, and transmit the
// Software, and to prepare derivative works of the Software, and
// to permit third-parties to whom the Software is furnished to
// do so, all subject to the following:
//
// The copyright notices in the Software and this entire
// statement, including the above license grant, this restriction
// and the following disclaimer, must be included in all copies
// of the Software, in whole or in part, and all derivative works
// of the Software, unless such copies or derivative works are
// solely in the form of machine-executable object code generated
// by a source language processor.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY
// KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
// WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR
// PURPOSE, TITLE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE
// COPYRIGHT HOLDERS OR ANYONE DISTRIBUTING THE SOFTWARE BE
// LIABLE FOR ANY DAMAGES OR OTHER LIABILITY, WHETHER IN
// CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
// CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE.
// http://stackoverflow.com/q/42481154/2880699
// https://www.justsoftwaresolutions.co.uk/cplusplus/using-enum-classes-as-bitfields.html
#pragma once
#include <type_traits>
template<typename E> // as suggested by Jay Miller
constexpr bool enable_bitmask_operators(E) {
return false;
}
#define ENABLE_BIT_OPERATORS(E) constexpr bool enable_bitmask_operators(E) { return true; }
template<typename E>
typename std::enable_if<enable_bitmask_operators(E()), E>::type
operator|(E lhs, E rhs) {
using underlying = typename std::underlying_type<E>::type;
return static_cast<E>(
static_cast<underlying>(lhs) | static_cast<underlying>(rhs));
}
template<typename E>
typename std::enable_if<enable_bitmask_operators(E()), E>::type
operator&(E lhs, E rhs) {
using underlying = typename std::underlying_type<E>::type;
return static_cast<E>(
static_cast<underlying>(lhs) & static_cast<underlying>(rhs));
}
template<typename E>
typename std::enable_if<enable_bitmask_operators(E()), E>::type
operator^(E lhs, E rhs) {
using underlying = typename std::underlying_type<E>::type;
return static_cast<E>(
static_cast<underlying>(lhs) ^ static_cast<underlying>(rhs));
}
template<typename E>
typename std::enable_if<enable_bitmask_operators(E()), E>::type
operator~(E lhs) {
using underlying = typename std::underlying_type<E>::type;
return static_cast<E>(
~static_cast<underlying>(lhs));
}
template<typename E>
typename std::enable_if<enable_bitmask_operators(E()), E&>::type
operator|=(E& lhs, E rhs) {
using underlying = typename std::underlying_type<E>::type;
lhs = static_cast<E>(
static_cast<underlying>(lhs) | static_cast<underlying>(rhs));
return lhs;
}
template<typename E>
typename std::enable_if<enable_bitmask_operators(E()), E&>::type
operator&=(E& lhs, E rhs) {
using underlying = typename std::underlying_type<E>::type;
lhs = static_cast<E>(
static_cast<underlying>(lhs) & static_cast<underlying>(rhs));
return lhs;
}
template<typename E>
typename std::enable_if<enable_bitmask_operators(E()), E&>::type
operator^=(E& lhs, E rhs) {
using underlying = typename std::underlying_type<E>::type;
lhs = static_cast<E>(
static_cast<underlying>(lhs) ^ static_cast<underlying>(rhs));
return lhs;
}
/// <summary>
/// Performs bitwise AND between two bitmasks and returns true if the result is not zero.
/// Typically used to check a bitmask for having a particular bit set.
/// </summary>
template<typename E>
bool operator%(E lhs, E rhs) {
typedef typename std::underlying_type<E>::type underlying;
return (static_cast<underlying>(lhs) & static_cast<underlying>(rhs)) != 0;
}