-
Notifications
You must be signed in to change notification settings - Fork 2.1k
/
strong_integers.h
483 lines (422 loc) · 20.4 KB
/
strong_integers.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
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
// Copyright 2010-2024 Google LLC
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
// Generates strongly typed integer types.
//
// StrongIndex is a simple template class mechanism for defining "logical"
// index-like class types that support some of the same functionalities
// as int, but which prevent assignment, construction, and
// other operations from other similar integer-like types. Essentially, the
// template class StrongIndex<StrongIndexName> has the additional
// property that it cannot be assigned to or constructed from another
// StrongIndex with a different StrongIndexName.
//
// Usage
// DEFINE_STRONG_INDEX_TYPE(name);
// where name is the desired (unique) name for the "logical" index type.
//
// StrongInt64 is a more general strong integer class based on int64_t.
// It has the same general type safeness, and supports more integer operators.
//
// Usage
// DEFINE_STRONG_INT64_TYPE(name);
// where name is the desired (unique) name for the "logical" int64_t type.
//
// SUPPORTED OPERATIONS --------------------------------------------------------
//
// The StrongIndex type is limited and only supports following operators are
// supported: unary: ++ (both prefix and postfix), comparison: ==, !=, <, <=, >,
// >=; assignment: =, +=, -=,; stream: <<. Each operator allows the same
// StrongIndexName and the int to be used on both left- and right-hand sides.
//
// The StrongInt64 type supports all integer operators across StrongInt64
// with the same StrongIntegerName and int64_t.
//
// Both support an accessor value() returning the stored value.
//
// The classes also define hash functors that allows the strong types to be used
// as key to hashable containers.
#ifndef OR_TOOLS_UTIL_STRONG_INTEGERS_H_
#define OR_TOOLS_UTIL_STRONG_INTEGERS_H_
#include <stddef.h>
#include <cstdint>
#include <iosfwd>
#include <limits>
#include <ostream> // NOLINT
#include "absl/base/attributes.h"
#include "absl/base/port.h"
#include "absl/strings/str_format.h"
#include "absl/strings/string_view.h"
namespace operations_research {
// Defines the StrongIndex and typedefs it to index_type_name.
//
// Note: The struct index_type_name ## _index_tag_ trickery is needed to ensure
// that a new type is created per index_type_name.
#define DEFINE_STRONG_INDEX_TYPE(index_type_name) \
struct index_type_name##_index_tag_ { \
static constexpr absl::string_view TypeName() { return #index_type_name; } \
}; \
typedef ::operations_research::StrongIndex<index_type_name##_index_tag_> \
index_type_name;
// Defines the StrongInt64 and typedefs it to integer_type_name.
//
// Note: The struct integer_type_name ## _integer_tag_ trickery is needed to
// ensure that a new type is created per integer_type_name.
#define DEFINE_STRONG_INT64_TYPE(integer_type_name) \
struct integer_type_name##_integer_tag_ { \
static constexpr absl::string_view TypeName() { \
return #integer_type_name; \
} \
}; \
typedef ::operations_research::StrongInt64<integer_type_name##_integer_tag_> \
integer_type_name;
// ----------- Implementation ------------
// Note: we need two classes as it is the only way to have different set of
// operators on each class.
// Note: We use to class to easily define a different set of operators for the
// index and int64_t type.
#define STRONG_ASSIGNMENT_OP(StrongClass, IntType, op) \
ThisType& operator op(const ThisType & arg_value) { \
value_ op arg_value.value(); \
return *this; \
} \
ThisType& operator op(IntType arg_value) { \
value_ op arg_value; \
return *this; \
}
#define INCREMENT_AND_DECREMENT_OPERATORS \
ThisType& operator++() { \
++value_; \
return *this; \
} \
const ThisType operator++(int) { \
ThisType temp(*this); \
++value_; \
return temp; \
} \
ThisType& operator--() { \
--value_; \
return *this; \
} \
const ThisType operator--(int) { \
ThisType temp(*this); \
--value_; \
return temp; \
}
// Holds an int value and behaves as an int by exposing assignment,
// unary, comparison, and arithmetic operators.
//
// The template parameter StrongIndexName defines the name for the int type and
// must be unique within a binary (the convenient DEFINE_STRONG_INDEX_TYPE macro
// at the start of the file generates a unique StrongIndexName).
//
// This class is NOT thread-safe.
template <typename StrongIndexName>
class StrongIndex {
public:
typedef int ValueType; // Needed for StrongVector.
typedef StrongIndex<StrongIndexName> ThisType; // Syntactic sugar.
static constexpr absl::string_view TypeName() {
return StrongIndexName::TypeName();
}
struct ABSL_DEPRECATED("Use absl::Hash instead") Hasher {
size_t operator()(const StrongIndex& x) const {
return static_cast<size_t>(x.value());
}
};
constexpr StrongIndex() : value_(0) {}
explicit constexpr StrongIndex(int value) : value_(value) {}
StrongIndex& operator=(int arg_value) {
value_ = arg_value;
return *this;
}
// The class provides a value() accessor returning the stored int value_
// as well as a templatized accessor that is just a syntactic sugar for
// static_cast<T>(var.value());
constexpr int value() const { return value_; }
template <typename ValType> // Needed for StrongVector.
constexpr ValType value() const {
return static_cast<ValType>(value_);
}
constexpr ThisType operator+() const { return ThisType(value_); }
constexpr ThisType operator-() const { return ThisType(-value_); }
INCREMENT_AND_DECREMENT_OPERATORS;
STRONG_ASSIGNMENT_OP(StrongIndex, int, +=);
STRONG_ASSIGNMENT_OP(StrongIndex, int, -=);
private:
int value_;
};
// Holds an int64_t value and behaves as an int64_t by exposing assignment,
// unary, comparison, and arithmetic operators.
//
// The template parameter StrongIntegerName defines the name for the int type
// and must be unique within a binary (the convenient DEFINE_STRONG_INTEGER_TYPE
// macro at the start of the file generates a unique StrongIntegerName).
//
// This class is NOT thread-safe.
template <typename StrongIntegerName>
class StrongInt64 {
public:
typedef int64_t ValueType; // Needed for StrongVector.
typedef StrongInt64<StrongIntegerName> ThisType; // Syntactic sugar.
static constexpr absl::string_view TypeName() {
return StrongIntegerName::TypeName();
}
struct ABSL_DEPRECATED("Use absl::Hash instead") Hasher {
size_t operator()(const StrongInt64& x) const {
return static_cast<size_t>(x.value());
}
};
constexpr StrongInt64() : value_(0) {}
// NOLINTBEGIN(google-explicit-constructor)
constexpr StrongInt64(int64_t value) : value_(value) {}
// NOLINTEND(google-explicit-constructor)
StrongInt64& operator=(int64_t arg_value) {
value_ = arg_value;
return *this;
}
constexpr int64_t value() const { return value_; }
template <typename ValType> // Needed for StrongVector.
constexpr ValType value() const {
return static_cast<ValType>(value_);
}
INCREMENT_AND_DECREMENT_OPERATORS;
constexpr ThisType operator+() const { return ThisType(value_); }
constexpr ThisType operator-() const { return ThisType(-value_); }
constexpr ThisType operator~() const { return ThisType(~value_); }
STRONG_ASSIGNMENT_OP(StrongInt64, int64_t, +=);
STRONG_ASSIGNMENT_OP(StrongInt64, int64_t, -=);
STRONG_ASSIGNMENT_OP(StrongInt64, int64_t, *=);
STRONG_ASSIGNMENT_OP(StrongInt64, int64_t, /=);
STRONG_ASSIGNMENT_OP(StrongInt64, int64_t, <<=);
STRONG_ASSIGNMENT_OP(StrongInt64, int64_t, >>=);
STRONG_ASSIGNMENT_OP(StrongInt64, int64_t, %=);
private:
int64_t value_;
};
#undef STRONG_ASSIGNMENT_OP
#undef INCREMENT_AND_DECREMENT_OPERATORS
// -- NON-MEMBER STREAM OPERATORS ----------------------------------------------
// We provide the << operator, primarily for logging purposes. Currently, there
// seems to be no need for an >> operator.
template <typename StrongIndexName>
std::ostream& operator<<(std::ostream& os, // NOLINT
StrongIndex<StrongIndexName> arg) {
return os << arg.value();
}
template <typename Sink, typename... T>
void AbslStringify(Sink& sink, StrongIndex<T...> arg) {
absl::Format(&sink, "%v", arg.value());
}
template <typename StrongIntegerName>
std::ostream& operator<<(std::ostream& os, // NOLINT
StrongInt64<StrongIntegerName> arg) {
return os << arg.value();
}
template <typename Sink, typename... T>
void AbslStringify(Sink& sink, StrongInt64<T...> arg) {
absl::Format(&sink, "%v", arg.value());
}
// -- NON-MEMBER ARITHMETIC OPERATORS ------------------------------------------
#define STRONG_TYPE_ARITHMETIC_OP(StrongType, IntType, op) \
template <typename StrongName> \
constexpr StrongType<StrongName> operator op(StrongType<StrongName> id_1, \
StrongType<StrongName> id_2) { \
return StrongType<StrongName>(id_1.value() op id_2.value()); \
} \
template <typename StrongName> \
constexpr StrongType<StrongName> operator op(StrongType<StrongName> id, \
IntType arg_val) { \
return StrongType<StrongName>(id.value() op arg_val); \
} \
template <typename StrongName> \
constexpr StrongType<StrongName> operator op(IntType arg_val, \
StrongType<StrongName> id) { \
return StrongType<StrongName>(arg_val op id.value()); \
}
STRONG_TYPE_ARITHMETIC_OP(StrongIndex, int, +);
STRONG_TYPE_ARITHMETIC_OP(StrongIndex, int, -);
STRONG_TYPE_ARITHMETIC_OP(StrongIndex, int, *);
STRONG_TYPE_ARITHMETIC_OP(StrongIndex, int, %);
STRONG_TYPE_ARITHMETIC_OP(StrongInt64, int64_t, +);
STRONG_TYPE_ARITHMETIC_OP(StrongInt64, int64_t, -);
STRONG_TYPE_ARITHMETIC_OP(StrongInt64, int64_t, *);
STRONG_TYPE_ARITHMETIC_OP(StrongInt64, int64_t, /);
STRONG_TYPE_ARITHMETIC_OP(StrongInt64, int64_t, <<);
STRONG_TYPE_ARITHMETIC_OP(StrongInt64, int64_t, >>);
STRONG_TYPE_ARITHMETIC_OP(StrongInt64, int64_t, %);
#undef STRONG_TYPE_ARITHMETIC_OP
// -- NON-MEMBER COMPARISON OPERATORS ------------------------------------------
#define STRONG_TYPE_COMPARISON_OP(StrongType, IntType, op) \
template <typename StrongName> \
static inline constexpr bool operator op(StrongType<StrongName> id_1, \
StrongType<StrongName> id_2) { \
return id_1.value() op id_2.value(); \
} \
template <typename StrongName> \
static inline constexpr bool operator op(StrongType<StrongName> id, \
IntType val) { \
return id.value() op val; \
} \
template <typename StrongName> \
static inline constexpr bool operator op(IntType val, \
StrongType<StrongName> id) { \
return val op id.value(); \
}
STRONG_TYPE_COMPARISON_OP(StrongIndex, int, ==); // NOLINT
STRONG_TYPE_COMPARISON_OP(StrongIndex, int, !=); // NOLINT
STRONG_TYPE_COMPARISON_OP(StrongIndex, int, <); // NOLINT
STRONG_TYPE_COMPARISON_OP(StrongIndex, int, <=); // NOLINT
STRONG_TYPE_COMPARISON_OP(StrongIndex, int, >); // NOLINT
STRONG_TYPE_COMPARISON_OP(StrongIndex, int, >=); // NOLINT
STRONG_TYPE_COMPARISON_OP(StrongInt64, int64_t, ==); // NOLINT
STRONG_TYPE_COMPARISON_OP(StrongInt64, int64_t, !=); // NOLINT
STRONG_TYPE_COMPARISON_OP(StrongInt64, int64_t, <); // NOLINT
STRONG_TYPE_COMPARISON_OP(StrongInt64, int64_t, <=); // NOLINT
STRONG_TYPE_COMPARISON_OP(StrongInt64, int64_t, >); // NOLINT
STRONG_TYPE_COMPARISON_OP(StrongInt64, int64_t, >=); // NOLINT
#undef STRONG_TYPE_COMPARISON_OP
// -- ABSL HASHING SUPPORT -----------------------------------------------------
template <typename StrongIndexName, typename H>
H AbslHashValue(H h, const StrongIndex<StrongIndexName>& i) {
return H::combine(std::move(h), i.value());
}
template <typename StrongIntegerName, typename H>
H AbslHashValue(H h, const StrongInt64<StrongIntegerName>& i) {
return H::combine(std::move(h), i.value());
}
} // namespace operations_research
// -- STD HASHING SUPPORT -----------------------------------------------------
namespace std {
template <typename Tag>
struct hash<operations_research::StrongIndex<Tag>>
: ::operations_research::StrongIndex<Tag>::Hasher {};
template <typename Tag>
struct hash<operations_research::StrongInt64<Tag>>
: ::operations_research::StrongInt64<Tag>::Hasher {};
} // namespace std
// -- STD NUMERIC_LIMITS SUPPORT ----------------------------------------------
namespace std {
template <typename Tag>
struct numeric_limits<operations_research::StrongIndex<Tag>> {
private:
using StrongIntT = operations_research::StrongIndex<Tag>;
using NativeTypeT = typename operations_research::StrongIndex<Tag>::ValueType;
public:
// NOLINTBEGIN(google3-readability-class-member-naming)
static constexpr bool is_specialized = true;
static constexpr bool is_signed = numeric_limits<NativeTypeT>::is_signed;
static constexpr bool is_integer = numeric_limits<NativeTypeT>::is_integer;
static constexpr bool is_exact = numeric_limits<NativeTypeT>::is_exact;
static constexpr bool has_infinity =
numeric_limits<NativeTypeT>::has_infinity;
static constexpr bool has_quiet_NaN =
numeric_limits<NativeTypeT>::has_quiet_NaN;
static constexpr bool has_signaling_NaN =
numeric_limits<NativeTypeT>::has_signaling_NaN;
static constexpr float_denorm_style has_denorm =
numeric_limits<NativeTypeT>::has_denorm;
static constexpr bool has_denorm_loss =
numeric_limits<NativeTypeT>::has_denorm_loss;
static constexpr float_round_style round_style =
numeric_limits<NativeTypeT>::round_style;
static constexpr bool is_iec559 = numeric_limits<NativeTypeT>::is_iec559;
static constexpr bool is_bounded = numeric_limits<NativeTypeT>::is_bounded;
static constexpr bool is_modulo = numeric_limits<NativeTypeT>::is_modulo;
static constexpr int digits = numeric_limits<NativeTypeT>::digits;
static constexpr int digits10 = numeric_limits<NativeTypeT>::digits10;
static constexpr int max_digits10 = numeric_limits<NativeTypeT>::max_digits10;
static constexpr int radix = numeric_limits<NativeTypeT>::radix;
static constexpr int min_exponent = numeric_limits<NativeTypeT>::min_exponent;
static constexpr int min_exponent10 =
numeric_limits<NativeTypeT>::min_exponent10;
static constexpr int max_exponent = numeric_limits<NativeTypeT>::max_exponent;
static constexpr int max_exponent10 =
numeric_limits<NativeTypeT>::max_exponent10;
static constexpr bool traps = numeric_limits<NativeTypeT>::traps;
static constexpr bool tinyness_before =
numeric_limits<NativeTypeT>::tinyness_before;
// NOLINTEND(google3-readability-class-member-naming)
static constexpr StrongIntT(min)() {
return StrongIntT(numeric_limits<NativeTypeT>::min());
}
static constexpr StrongIntT lowest() {
return StrongIntT(numeric_limits<NativeTypeT>::lowest());
}
static constexpr StrongIntT(max)() {
return StrongIntT(numeric_limits<NativeTypeT>::max());
}
static constexpr StrongIntT epsilon() { return StrongIntT(); }
static constexpr StrongIntT round_error() { return StrongIntT(); }
static constexpr StrongIntT infinity() { return StrongIntT(); }
static constexpr StrongIntT quiet_NaN() { return StrongIntT(); }
static constexpr StrongIntT signaling_NaN() { return StrongIntT(); }
static constexpr StrongIntT denorm_min() { return StrongIntT(); }
};
template <typename Tag>
struct numeric_limits<operations_research::StrongInt64<Tag>> {
private:
using StrongIntT = operations_research::StrongInt64<Tag>;
using NativeTypeT = typename operations_research::StrongInt64<Tag>::ValueType;
public:
// NOLINTBEGIN(google3-readability-class-member-naming)
static constexpr bool is_specialized = true;
static constexpr bool is_signed = numeric_limits<NativeTypeT>::is_signed;
static constexpr bool is_integer = numeric_limits<NativeTypeT>::is_integer;
static constexpr bool is_exact = numeric_limits<NativeTypeT>::is_exact;
static constexpr bool has_infinity =
numeric_limits<NativeTypeT>::has_infinity;
static constexpr bool has_quiet_NaN =
numeric_limits<NativeTypeT>::has_quiet_NaN;
static constexpr bool has_signaling_NaN =
numeric_limits<NativeTypeT>::has_signaling_NaN;
static constexpr float_denorm_style has_denorm =
numeric_limits<NativeTypeT>::has_denorm;
static constexpr bool has_denorm_loss =
numeric_limits<NativeTypeT>::has_denorm_loss;
static constexpr float_round_style round_style =
numeric_limits<NativeTypeT>::round_style;
static constexpr bool is_iec559 = numeric_limits<NativeTypeT>::is_iec559;
static constexpr bool is_bounded = numeric_limits<NativeTypeT>::is_bounded;
static constexpr bool is_modulo = numeric_limits<NativeTypeT>::is_modulo;
static constexpr int digits = numeric_limits<NativeTypeT>::digits;
static constexpr int digits10 = numeric_limits<NativeTypeT>::digits10;
static constexpr int max_digits10 = numeric_limits<NativeTypeT>::max_digits10;
static constexpr int radix = numeric_limits<NativeTypeT>::radix;
static constexpr int min_exponent = numeric_limits<NativeTypeT>::min_exponent;
static constexpr int min_exponent10 =
numeric_limits<NativeTypeT>::min_exponent10;
static constexpr int max_exponent = numeric_limits<NativeTypeT>::max_exponent;
static constexpr int max_exponent10 =
numeric_limits<NativeTypeT>::max_exponent10;
static constexpr bool traps = numeric_limits<NativeTypeT>::traps;
static constexpr bool tinyness_before =
numeric_limits<NativeTypeT>::tinyness_before;
// NOLINTEND(google3-readability-class-member-naming)
static constexpr StrongIntT(min)() {
return StrongIntT(numeric_limits<NativeTypeT>::min());
}
static constexpr StrongIntT lowest() {
return StrongIntT(numeric_limits<NativeTypeT>::lowest());
}
static constexpr StrongIntT(max)() {
return StrongIntT(numeric_limits<NativeTypeT>::max());
}
static constexpr StrongIntT epsilon() { return StrongIntT(); }
static constexpr StrongIntT round_error() { return StrongIntT(); }
static constexpr StrongIntT infinity() { return StrongIntT(); }
static constexpr StrongIntT quiet_NaN() { return StrongIntT(); }
static constexpr StrongIntT signaling_NaN() { return StrongIntT(); }
static constexpr StrongIntT denorm_min() { return StrongIntT(); }
};
} // namespace std
#endif // OR_TOOLS_UTIL_STRONG_INTEGERS_H_