blob: a2a22ebebea13a29af7613aa5285005795862a9f [file] [log] [blame]
Michael Wright44753b12020-07-08 13:48:11 +01001/*
Dominik Laskowski75788452021-02-09 18:51:25 -08002 * Copyright 2020 The Android Open Source Project
Michael Wright44753b12020-07-08 13:48:11 +01003 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
Dominik Laskowski75788452021-02-09 18:51:25 -080017#pragma once
Michael Wright44753b12020-07-08 13:48:11 +010018
Dominik Laskowski75788452021-02-09 18:51:25 -080019#include <ftl/enum.h>
20#include <ftl/string.h>
21
Dominik Laskowskia7fe7eb2022-02-16 10:44:28 -080022#include <bitset>
Michael Wright44753b12020-07-08 13:48:11 +010023#include <cstdint>
Dominik Laskowski75788452021-02-09 18:51:25 -080024#include <iterator>
Lloyd Pique83c6f9e2024-11-20 18:06:07 -080025#include <initializer_list>
Michael Wright44753b12020-07-08 13:48:11 +010026#include <string>
27#include <type_traits>
28
Dominik Laskowski2f01d772022-03-23 16:01:29 -070029// TODO(b/185536303): Align with FTL style.
Michael Wright44753b12020-07-08 13:48:11 +010030
Dominik Laskowski2f01d772022-03-23 16:01:29 -070031namespace android::ftl {
Michael Wright44753b12020-07-08 13:48:11 +010032
Michael Wright44753b12020-07-08 13:48:11 +010033/* A class for handling flags defined by an enum or enum class in a type-safe way. */
Michael Wright8759d672020-07-21 00:46:45 +010034template <typename F>
Michael Wright44753b12020-07-08 13:48:11 +010035class Flags {
36 // F must be an enum or its underlying type is undefined. Theoretically we could specialize this
37 // further to avoid this restriction but in general we want to encourage the use of enums
38 // anyways.
Michael Wright8759d672020-07-21 00:46:45 +010039 static_assert(std::is_enum_v<F>, "Flags type must be an enum");
Dominik Laskowski75788452021-02-09 18:51:25 -080040 using U = std::underlying_type_t<F>;
Michael Wright44753b12020-07-08 13:48:11 +010041
42public:
Michael Wright75d9e662020-07-11 23:54:40 +010043 constexpr Flags(F f) : mFlags(static_cast<U>(f)) {}
Lloyd Pique83c6f9e2024-11-20 18:06:07 -080044 constexpr Flags(std::initializer_list<F> fs) : mFlags(combine(fs)) {}
Michael Wright75d9e662020-07-11 23:54:40 +010045 constexpr Flags() : mFlags(0) {}
46 constexpr Flags(const Flags<F>& f) : mFlags(f.mFlags) {}
Michael Wright44753b12020-07-08 13:48:11 +010047
48 // Provide a non-explicit construct for non-enum classes since they easily convert to their
49 // underlying types (e.g. when used with bitwise operators). For enum classes, however, we
50 // should force them to be explicitly constructed from their underlying types to make full use
51 // of the type checker.
52 template <typename T = U>
Dominik Laskowski2f01d772022-03-23 16:01:29 -070053 constexpr Flags(T t, std::enable_if_t<!is_scoped_enum_v<F>, T>* = nullptr) : mFlags(t) {}
Dominik Laskowski75788452021-02-09 18:51:25 -080054
Michael Wright44753b12020-07-08 13:48:11 +010055 template <typename T = U>
Dominik Laskowski2f01d772022-03-23 16:01:29 -070056 explicit constexpr Flags(T t, std::enable_if_t<is_scoped_enum_v<F>, T>* = nullptr)
Michael Wright75d9e662020-07-11 23:54:40 +010057 : mFlags(t) {}
58
59 class Iterator {
Dominik Laskowskia7fe7eb2022-02-16 10:44:28 -080060 using Bits = std::uint64_t;
61 static_assert(sizeof(U) <= sizeof(Bits));
Michael Wright75d9e662020-07-11 23:54:40 +010062
63 public:
Dominik Laskowskia7fe7eb2022-02-16 10:44:28 -080064 constexpr Iterator() = default;
Michael Wright75d9e662020-07-11 23:54:40 +010065 Iterator(Flags<F> flags) : mRemainingFlags(flags.mFlags) { (*this)++; }
Michael Wright75d9e662020-07-11 23:54:40 +010066
67 // Pre-fix ++
68 Iterator& operator++() {
Dominik Laskowskia7fe7eb2022-02-16 10:44:28 -080069 if (mRemainingFlags.none()) {
70 mCurrFlag = 0;
Michael Wright75d9e662020-07-11 23:54:40 +010071 } else {
Dominik Laskowskia7fe7eb2022-02-16 10:44:28 -080072 // TODO: Replace with std::countr_zero in C++20.
73 const Bits bit = static_cast<Bits>(__builtin_ctzll(mRemainingFlags.to_ullong()));
74 mRemainingFlags.reset(static_cast<std::size_t>(bit));
75 mCurrFlag = static_cast<U>(static_cast<Bits>(1) << bit);
Michael Wright75d9e662020-07-11 23:54:40 +010076 }
77 return *this;
78 }
79
80 // Post-fix ++
81 Iterator operator++(int) {
82 Iterator iter = *this;
83 ++*this;
84 return iter;
85 }
86
87 bool operator==(Iterator other) const {
88 return mCurrFlag == other.mCurrFlag && mRemainingFlags == other.mRemainingFlags;
89 }
90
91 bool operator!=(Iterator other) const { return !(*this == other); }
92
Dominik Laskowskia7fe7eb2022-02-16 10:44:28 -080093 F operator*() const { return F{mCurrFlag}; }
Michael Wright75d9e662020-07-11 23:54:40 +010094
95 // iterator traits
96
97 // In the future we could make this a bidirectional const iterator instead of a forward
98 // iterator but it doesn't seem worth the added complexity at this point. This could not,
99 // however, be made a non-const iterator as assigning one flag to another is a non-sensical
100 // operation.
101 using iterator_category = std::input_iterator_tag;
102 using value_type = F;
103 // Per the C++ spec, because input iterators are not assignable the iterator's reference
104 // type does not actually need to be a reference. In fact, making it a reference would imply
105 // that modifying it would change the underlying Flags object, which is obviously wrong for
106 // the same reason this can't be a non-const iterator.
107 using reference = F;
108 using difference_type = void;
109 using pointer = void;
110
111 private:
Dominik Laskowskia7fe7eb2022-02-16 10:44:28 -0800112 std::bitset<sizeof(Bits) * 8> mRemainingFlags;
113 U mCurrFlag = 0;
Michael Wright75d9e662020-07-11 23:54:40 +0100114 };
115
Michael Wright44753b12020-07-08 13:48:11 +0100116 /*
117 * Tests whether the given flag is set.
118 */
119 bool test(F flag) const {
120 U f = static_cast<U>(flag);
Michael Wright75d9e662020-07-11 23:54:40 +0100121 return (f & mFlags) == f;
Michael Wright44753b12020-07-08 13:48:11 +0100122 }
123
124 /* Tests whether any of the given flags are set */
Prabir Pradhan12f89132023-04-18 20:47:03 +0000125 bool any(Flags<F> f = ~Flags<F>()) const { return (mFlags & f.mFlags) != 0; }
Michael Wright44753b12020-07-08 13:48:11 +0100126
127 /* Tests whether all of the given flags are set */
Prabir Pradhanf0a3d812022-02-07 02:54:39 -0800128 bool all(Flags<F> f) const { return (mFlags & f.mFlags) == f.mFlags; }
Michael Wright44753b12020-07-08 13:48:11 +0100129
Siarhei Vishniakou253f4642022-11-09 13:42:06 -0800130 constexpr Flags<F> operator|(Flags<F> rhs) const { return static_cast<F>(mFlags | rhs.mFlags); }
Michael Wright44753b12020-07-08 13:48:11 +0100131 Flags<F>& operator|=(Flags<F> rhs) {
Michael Wright75d9e662020-07-11 23:54:40 +0100132 mFlags = mFlags | rhs.mFlags;
Michael Wright44753b12020-07-08 13:48:11 +0100133 return *this;
134 }
135
Michael Wright75d9e662020-07-11 23:54:40 +0100136 Flags<F> operator&(Flags<F> rhs) const { return static_cast<F>(mFlags & rhs.mFlags); }
Michael Wright44753b12020-07-08 13:48:11 +0100137 Flags<F>& operator&=(Flags<F> rhs) {
Michael Wright75d9e662020-07-11 23:54:40 +0100138 mFlags = mFlags & rhs.mFlags;
Michael Wright44753b12020-07-08 13:48:11 +0100139 return *this;
140 }
141
Michael Wright75d9e662020-07-11 23:54:40 +0100142 Flags<F> operator^(Flags<F> rhs) const { return static_cast<F>(mFlags ^ rhs.mFlags); }
Michael Wright44753b12020-07-08 13:48:11 +0100143 Flags<F>& operator^=(Flags<F> rhs) {
Michael Wright75d9e662020-07-11 23:54:40 +0100144 mFlags = mFlags ^ rhs.mFlags;
Michael Wright44753b12020-07-08 13:48:11 +0100145 return *this;
146 }
147
Michael Wright75d9e662020-07-11 23:54:40 +0100148 Flags<F> operator~() { return static_cast<F>(~mFlags); }
Michael Wright44753b12020-07-08 13:48:11 +0100149
Michael Wright75d9e662020-07-11 23:54:40 +0100150 bool operator==(Flags<F> rhs) const { return mFlags == rhs.mFlags; }
Michael Wright44753b12020-07-08 13:48:11 +0100151 bool operator!=(Flags<F> rhs) const { return !operator==(rhs); }
152
153 Flags<F>& operator=(const Flags<F>& rhs) {
Michael Wright75d9e662020-07-11 23:54:40 +0100154 mFlags = rhs.mFlags;
Michael Wright44753b12020-07-08 13:48:11 +0100155 return *this;
156 }
157
Prabir Pradhanf0a3d812022-02-07 02:54:39 -0800158 inline Flags<F>& clear(Flags<F> f = static_cast<F>(~static_cast<U>(0))) {
159 return *this &= ~f;
160 }
161
Michael Wright75d9e662020-07-11 23:54:40 +0100162 Iterator begin() const { return Iterator(*this); }
163
164 Iterator end() const { return Iterator(); }
165
Michael Wright44753b12020-07-08 13:48:11 +0100166 /*
167 * Returns the stored set of flags.
168 *
169 * Note that this returns the underlying type rather than the base enum class. This is because
170 * the value is no longer necessarily a strict member of the enum since the returned value could
171 * be multiple enum variants OR'd together.
172 */
Michael Wright75d9e662020-07-11 23:54:40 +0100173 U get() const { return mFlags; }
Michael Wright44753b12020-07-08 13:48:11 +0100174
Michael Wright8759d672020-07-21 00:46:45 +0100175 std::string string() const {
Michael Wright44753b12020-07-08 13:48:11 +0100176 std::string result;
177 bool first = true;
178 U unstringified = 0;
Michael Wright75d9e662020-07-11 23:54:40 +0100179 for (const F f : *this) {
Dominik Laskowski2f01d772022-03-23 16:01:29 -0700180 if (const auto flagName = flag_name(f)) {
Dominik Laskowski75788452021-02-09 18:51:25 -0800181 appendFlag(result, flagName.value(), first);
Michael Wright44753b12020-07-08 13:48:11 +0100182 } else {
Michael Wright75d9e662020-07-11 23:54:40 +0100183 unstringified |= static_cast<U>(f);
Michael Wright44753b12020-07-08 13:48:11 +0100184 }
185 }
186
187 if (unstringified != 0) {
Dominik Laskowski2f01d772022-03-23 16:01:29 -0700188 constexpr auto radix = sizeof(U) == 1 ? Radix::kBin : Radix::kHex;
189 appendFlag(result, to_string(unstringified, radix), first);
Michael Wright44753b12020-07-08 13:48:11 +0100190 }
191
192 if (first) {
193 result += "0x0";
194 }
195
196 return result;
197 }
198
199private:
Michael Wright75d9e662020-07-11 23:54:40 +0100200 U mFlags;
Michael Wright44753b12020-07-08 13:48:11 +0100201
Lloyd Pique83c6f9e2024-11-20 18:06:07 -0800202 static constexpr U combine(std::initializer_list<F> fs) {
203 U result = 0;
204 for (const F f : fs) {
205 result |= static_cast<U>(f);
206 }
207 return result;
208 }
209
Michael Wright8759d672020-07-21 00:46:45 +0100210 static void appendFlag(std::string& str, const std::string_view& flag, bool& first) {
Michael Wright44753b12020-07-08 13:48:11 +0100211 if (first) {
212 first = false;
213 } else {
214 str += " | ";
215 }
216 str += flag;
217 }
218};
219
220// This namespace provides operator overloads for enum classes to make it easier to work with them
221// as flags. In order to use these, add them via a `using namespace` declaration.
222namespace flag_operators {
223
Dominik Laskowski2f01d772022-03-23 16:01:29 -0700224template <typename F, typename = std::enable_if_t<is_scoped_enum_v<F>>>
Michael Wright44753b12020-07-08 13:48:11 +0100225inline Flags<F> operator~(F f) {
Dominik Laskowski2f01d772022-03-23 16:01:29 -0700226 return static_cast<F>(~to_underlying(f));
Michael Wright44753b12020-07-08 13:48:11 +0100227}
Dominik Laskowski75788452021-02-09 18:51:25 -0800228
Dominik Laskowski2f01d772022-03-23 16:01:29 -0700229template <typename F, typename = std::enable_if_t<is_scoped_enum_v<F>>>
Siarhei Vishniakou253f4642022-11-09 13:42:06 -0800230constexpr Flags<F> operator|(F lhs, F rhs) {
Dominik Laskowski2f01d772022-03-23 16:01:29 -0700231 return static_cast<F>(to_underlying(lhs) | to_underlying(rhs));
Michael Wright44753b12020-07-08 13:48:11 +0100232}
233
234} // namespace flag_operators
Dominik Laskowski2f01d772022-03-23 16:01:29 -0700235} // namespace android::ftl