Hermes
Loading...
Searching...
No Matches
index.h
Go to the documentation of this file.
1
30
31#ifndef HERMES_COMMON_INDEX_H
32#define HERMES_COMMON_INDEX_H
33
34#include <hermes/common/size.h>
35#include <hermes/common/debug.h>
36
37namespace hermes {
38
39// *********************************************************************************************************************
40// Index2
41// *********************************************************************************************************************
50template<typename T> struct Index2 {
51 static_assert(std::is_same<T, i8>::value || std::is_same<T, i16>::value ||
52 std::is_same<T, i32>::value || std::is_same<T, i64>::value,
53 "Index2 must hold an integer type!");
54 // *******************************************************************************************************************
55 // FRIEND FUNCTIONS
56 // *******************************************************************************************************************
57 // arithmetic
59#define ARITHMETIC_OP(OP) \
60 template<typename U> \
61 HERMES_DEVICE_CALLABLE friend Index2<T> operator OP (const Size2<U> &b, const Index2<T> &a) { \
62 return Index2<T>(b.width OP a.i, b.height OP a.j); } \
63 HERMES_DEVICE_CALLABLE friend Index2<T> operator OP (const Index2<T> &b, const T &a) { \
64 return Index2<T>(b.i OP a, b.j OP a); }
69#undef ARITHMETIC_OP
70 HERMES_DEVICE_CALLABLE Index2<T> operator-() const { return {-i, -j}; }
80#if defined(__CUDA_ARCH__) && __CUDA_ARCH__ > 0
82#else
83 return std::abs(a.i - b.i) + std::abs(a.j - b.j);
84#endif
85 }
87#define MATH_OP(NAME, OP) \
88 HERMES_DEVICE_CALLABLE friend Index2<T> NAME(const Index2<T>& a, const Index2<T>& b) { \
89 return Index2<T>(OP(a.i, b.i), OP(a.j, b.j)); }
90#if defined(__CUDA_ARCH__) && __CUDA_ARCH__ > 0
91 MATH_OP(max, max)
92 MATH_OP(min, min)
93#else
94 MATH_OP(max, std::max)
95 MATH_OP(min, std::min)
96#endif
97#undef MATH_OP
98 // *******************************************************************************************************************
99 // CONSTRUCTORS
100 // *******************************************************************************************************************
105 HERMES_DEVICE_CALLABLE explicit Index2(T v) : i(v), j(v) {}
114 template<typename S>
115 HERMES_DEVICE_CALLABLE explicit Index2(const Size2<S> &size) : i(size.width), j(size.height) {}
116 // *******************************************************************************************************************
117 // OPERATORS
118 // *******************************************************************************************************************
119 // access
124 HERMES_DEVICE_CALLABLE T operator[](int d) const { return (&i)[d]; }
129 HERMES_DEVICE_CALLABLE T &operator[](int d) { return (&i)[d]; }
130 // arithmetic
131#define ARITHMETIC_OP(OP) \
132 HERMES_DEVICE_CALLABLE Index2<T>& operator OP##= (const Index2<T> &b) { \
133 i OP##= b.i; j OP##= b.j; return *this; } \
134 HERMES_DEVICE_CALLABLE Index2<T> operator OP (const Index2<T>& b) const { \
135 return {i OP b.i, j OP b.j}; } \
136 template<typename U> \
137 HERMES_DEVICE_CALLABLE Index2<T>& operator OP##= (const Size2<U> &b) { \
138 i OP##= static_cast<T>(b.width); j OP##= static_cast<T>(b.height); return *this; } \
139 template<typename U> \
140 HERMES_DEVICE_CALLABLE Index2<T> operator OP (const Size2<U>& b) const { \
141 return {i OP static_cast<T>(b.width), j OP static_cast<T>(b.height)}; }
146#undef ARITHMETIC_OP
147 // relational
148#define RELATIONAL_OP(OP, CO) \
149 HERMES_DEVICE_CALLABLE bool operator OP (const Index2<T> &b) const { \
150 return i OP b.i CO j OP b.j; } \
151 template<typename U> \
152 HERMES_DEVICE_CALLABLE bool operator OP (const Size2<U>& b) const { \
153 return i OP static_cast<T>(b.width) CO j OP static_cast<T>(b.height); }
154 RELATIONAL_OP(==, &&)
155 RELATIONAL_OP(!=, ||)
156 RELATIONAL_OP(>=, &&)
157 RELATIONAL_OP(<=, &&)
158 RELATIONAL_OP(<, &&)
159 RELATIONAL_OP(>, &&)
160#undef RELATIONAL_OP
161 // *******************************************************************************************************************
162 // METHODS
163 // *******************************************************************************************************************
172 HERMES_DEVICE_CALLABLE Index2<T> left(T d = T(1)) const { return Index2<T>(i - d, j); }
176 HERMES_DEVICE_CALLABLE Index2<T> right(T d = T(1)) const { return Index2<T>(i + d, j); }
180 HERMES_DEVICE_CALLABLE Index2<T> down(T d = T(1)) const { return Index2<T>(i, j - d); }
184 HERMES_DEVICE_CALLABLE Index2<T> up(T d = T(1)) const { return Index2<T>(i, j + d); }
188#if defined(__CUDA_ARCH__) && __CUDA_ARCH__ > 0
190#else
191 i = std::max(0, std::min(i, static_cast<T>(s.width)));
192 j = std::max(0, std::min(j, static_cast<T>(s.height)));
193#endif
194 }
195 // *******************************************************************************************************************
196 // PUBLIC FIELDS
197 // *******************************************************************************************************************
199 T i = T(0);
201 T j = T(0);
202};
203
204// *********************************************************************************************************************
205// Index2Iterator
206// *********************************************************************************************************************
208template<typename T> class Index2Iterator {
209public:
210 // *******************************************************************************************************************
211 // CONSTRUCTORS
212 // *******************************************************************************************************************
219 : index_(lower), lower_(lower), upper_(upper) {}
225 : index_(start), lower_(lower), upper_(upper) {}
226 // *******************************************************************************************************************
227 // OPERATORS
228 // *******************************************************************************************************************
229 // arithmetic
232 index_.i++;
233 if (index_.i >= upper_.i) {
234 index_.i = lower_.i;
235 index_.j++;
236 if (index_.j >= upper_.j)
237 index_ = upper_;
238 }
239 return *this;
240 }
241 // access
243 HERMES_DEVICE_CALLABLE const Index2<T> &operator*() const { return index_; }
249 auto size = upper_ - lower_;
250 return index_.j * size.i + index_.i;
251 }
252 // boolean
257 return index_ == other.index_;
258 }
263 return index_ != other.index_;
264 }
265
266private:
267 Index2<T> index_, lower_, upper_;
268};
269
270// *********************************************************************************************************************
271// Index2Range
272// *********************************************************************************************************************
283template<typename T> class Index2Range {
284public:
285 // *******************************************************************************************************************
286 // FRIEND FUNCTIONS
287 // *******************************************************************************************************************
292#if defined(__CUDA_ARCH__) && __CUDA_ARCH__ > 0
293 return Index2Range<T>(Index2<T>(max(a.lower_.i, b.lower_.i), max(a.lower_.j, b.lower_.j)),
294 Index2<T>(min(a.upper_.i, b.upper_.i), min(a.upper_.j, b.upper_.j)));
295#else
296 return {{std::max(a.lower_.i, b.lower_.i), std::max(a.lower_.j, b.lower_.j)},
297 {std::min(a.upper_.i, b.upper_.i), std::min(a.upper_.j, b.upper_.j)}};
298#endif
299 }
300 // *******************************************************************************************************************
301 // CONSTRUCTORS
302 // *******************************************************************************************************************
317 lower_(Index2<T>()), upper_(Index2<T>(upper.width, upper.height)) {}
321 return ij >= lower_ && ij < upper_;
322 }
323// *******************************************************************************************************************
324// OPERATORS
325// *******************************************************************************************************************
326// relational
330 return lower_ == r.lower_ && upper_ == r.upper_;
331 }
332// *******************************************************************************************************************
333// METHODS
334// *******************************************************************************************************************
337 return Index2Iterator<T>(lower_, upper_, lower_);
338 }
341 return Index2Iterator<T>(lower_, upper_, upper_);
342 }
344 [[nodiscard]] HERMES_DEVICE_CALLABLE const Index2<T> &lower() const { return lower_; }
346 [[nodiscard]] HERMES_DEVICE_CALLABLE const Index2<T> &upper() const { return upper_; }
349 auto d = upper_ - lower_;
350 return d.i * d.j;
351 }
352
353private:
354 Index2<T> lower_, upper_;
355};
356
357// *********************************************************************************************************************
358// Index2
359// *********************************************************************************************************************
362template<typename T> struct Index3 {
363 static_assert(std::is_same<T, i8>::value || std::is_same<T, i16>::value ||
364 std::is_same<T, i32>::value || std::is_same<T, i64>::value,
365 "Index3 must hold an integer type!");
366 // *******************************************************************************************************************
367 // FRIEND FUNCTIONS
368 // *******************************************************************************************************************
369 // arithmetic
372#define ARITHMETIC_OP(OP) \
373 template<typename U> \
374 HERMES_DEVICE_CALLABLE friend Index3<T> operator OP (const Size3<U> &b, const Index3<T> &a) { \
375 return Index3<T>(b.width OP a.i, b.height OP a.j, b.depth OP a.k); }
380#undef ARITHMETIC_OP
381 // *******************************************************************************************************************
382 // CONSTRUCTORS
383 // *******************************************************************************************************************
385 HERMES_DEVICE_CALLABLE Index3() : i(0), j(0), k(0) {}
387 HERMES_DEVICE_CALLABLE explicit Index3(T v) : i(v), j(v), k(v) {}
393 // *******************************************************************************************************************
394 // OPERATORS
395 // *******************************************************************************************************************
396 // access
401 HERMES_DEVICE_CALLABLE T operator[](int _i) const { return (&i)[_i]; }
406 HERMES_DEVICE_CALLABLE T &operator[](int _i) { return (&i)[_i]; }
407 // arithmetic
408#define ARITHMETIC_OP(OP) \
409 HERMES_DEVICE_CALLABLE Index3<T>& operator OP##= (const Index3<T> &b) { \
410 i OP##= b.i; j OP##= b.j; k OP##= b.k; return *this; } \
411 HERMES_DEVICE_CALLABLE Index3<T> operator OP (const Index3<T>& b) const { \
412 return {i OP b.i, j OP b.j, k OP b.k}; } \
413 template<typename U> \
414 HERMES_DEVICE_CALLABLE Index3<T>& operator OP##= (const Size3<U> &b) { \
415 i OP##= b.width; j OP##= b.height; k OP##= b.depth; return *this; } \
416 template<typename U> \
417 HERMES_DEVICE_CALLABLE Index3<T> operator OP (const Size3<U>& b) const { \
418 return {i OP static_cast<T>(b.width), j OP static_cast<T>(b.height), \
419 k OP static_cast<T>(b.depth)}; }
424#undef ARITHMETIC_OP
425 // relational
426#define RELATIONAL_OP(OP, CO) \
427 HERMES_DEVICE_CALLABLE bool operator OP (const Index3<T> &b) const { \
428 return i OP b.i CO j OP b.j CO k OP b.k; } \
429 template<typename U> \
430 HERMES_DEVICE_CALLABLE bool operator OP (const Size3<U>& b) const { \
431 return i OP static_cast<T>(b.width) CO \
432 j OP static_cast<T>(b.height) CO k OP static_cast<T>(b.depth); }
433 RELATIONAL_OP(==, &&)
434 RELATIONAL_OP(!=, ||)
435 RELATIONAL_OP(>=, &&)
436 RELATIONAL_OP(<=, &&)
437 RELATIONAL_OP(<, &&)
438 RELATIONAL_OP(>, &&)
439#undef RELATIONAL_OP
440 // *******************************************************************************************************************
441 // PUBLIC FIELDS
442 // *******************************************************************************************************************
444 T i{0};
446 T j{0};
448 T k{0};
449};
450
451// *********************************************************************************************************************
452// Index3Iterator
453// *********************************************************************************************************************
455template<typename T> class Index3Iterator {
456public:
457 // *******************************************************************************************************************
458 // CONSTRUCTORS
459 // *******************************************************************************************************************
466 : index_(start), lower_(lower), upper_(upper) {}
468 HERMES_DEVICE_CALLABLE explicit Index3Iterator(Index3<T> upper) : upper_(upper) {}
469 // *******************************************************************************************************************
470 // OPERATORS
471 // *******************************************************************************************************************
472 // access
474 HERMES_DEVICE_CALLABLE const Index3<T> &operator*() const { return index_; }
482 auto size = upper_ - lower_;
483 return index_.k * (size.i * size.j) + index_.j * size.i + index_.i;
484 }
485 // arithmetic
488 index_.i++;
489 if (index_.i >= upper_.i) {
490 index_.i = lower_.i;
491 index_.j++;
492 if (index_.j >= upper_.j) {
493 index_.j = lower_.j;
494 index_.k++;
495 if (index_.k >= upper_.k)
496 index_ = upper_;
497 }
498 }
499 return *this;
500 }
501 // boolean
506 return index_ == other.index_;
507 }
512 return index_ != other.index_;
513 }
514
515private:
516 Index3<T> index_, lower_, upper_;
517};
518
519// *********************************************************************************************************************
520// Index3Range
521// *********************************************************************************************************************
532template<typename T> class Index3Range {
533public:
534 // *******************************************************************************************************************
535 // FRIEND FUNCTIONS
536 // *******************************************************************************************************************
542#if defined(__CUDA_ARCH__) && __CUDA_ARCH__ > 0
543 return {Index3<T>(max(a.lower_.i, b.lower_.i), max(a.lower_.j, b.lower_.j),
544 max(a.lower_.k, b.lower_.k)),
545 Index3<T>(min(a.upper_.i, b.upper_.i), min(a.upper_.j, b.upper_.j),
546 min(a.upper_.k, b.upper_.k))};
547#else
548 return {{std::max(a.lower_.i, b.lower_.i), std::max(a.lower_.i, b.lower_.j),
549 std::max(a.lower_.k, b.lower_.k)},
550 {std::min(a.upper_.i, b.upper_.i), std::min(a.upper_.i, b.upper_.j),
551 std::min(a.upper_.k, b.upper_.k)}};
552#endif
553 }
554 // *******************************************************************************************************************
555 // CONSTRUCTORS
556 // *******************************************************************************************************************
566 : upper_(upper) {}
569 : lower_(Index3<T>()),
570 upper_(Index3<T>(upper.width, upper.height, upper.depth)) {}
574 : lower_(lower), upper_(upper) {}
575 // *******************************************************************************************************************
576 // METHODS
577 // *******************************************************************************************************************
586 auto size = upper_ - lower_;
587 return ijk.k * (size.i * size.j) + ijk.j * size.i + ijk.i;
588 }
591 return Index3Iterator<T>(lower_, upper_, lower_);
592 }
595 return Index3Iterator<T>(lower_, upper_, upper_);
596 }
600#if defined(__CUDA_ARCH__) && __CUDA_ARCH__ > 0
601 return size3(std::abs(upper_[0] - lower_[0]),
602 std::abs(upper_[1] - lower_[1]),
603 std::abs(upper_[2] - lower_[2]));
604#else
605 return size3(std::abs(upper_[0] - lower_[0]),
606 std::abs(upper_[1] - lower_[1]),
607 std::abs(upper_[2] - lower_[2]));
608#endif
609 }
610
611private:
612 Index3<T> lower_, upper_;
613};
614
615// *********************************************************************************************************************
616// IO
617// *********************************************************************************************************************
622template<typename T>
623std::ostream &operator<<(std::ostream &o, const Index2<T> &ij) {
624 o << "Index[" << ij.i << ", " << ij.j << "]";
625 return o;
626}
631template<typename T>
632std::ostream &operator<<(std::ostream &o, const Index3<T> &ijk) {
633 o << "Index[" << ijk.i << ", " << ijk.j << ", " << ijk.k << "]";
634 return o;
635}
636
637// *********************************************************************************************************************
638// TYPEDEFS
639// *********************************************************************************************************************
653
654} // namespace hermes
655
656#endif
657
Definition index.h:208
HERMES_DEVICE_CALLABLE Index2Iterator(Index2< T > lower, Index2< T > upper, Index2< T > start)
Construct a new Index2Iterator object.
Definition index.h:224
HERMES_DEVICE_CALLABLE bool operator==(const Index2Iterator< T > &other) const
are equal? operator
Definition index.h:256
HERMES_DEVICE_CALLABLE size_t flatIndex() const
Computes a flat index based on size.
Definition index.h:248
HERMES_DEVICE_CALLABLE const Index2< T > & operator*() const
Definition index.h:243
HERMES_DEVICE_CALLABLE bool operator!=(const Index2Iterator< T > &other) const
are different? operator
Definition index.h:262
HERMES_DEVICE_CALLABLE Index2Iterator & operator++()
Definition index.h:231
HERMES_DEVICE_CALLABLE Index2Iterator()
Default constructor.
Definition index.h:214
HERMES_DEVICE_CALLABLE Index2Iterator(Index2< T > lower, Index2< T > upper)
Constructor.
Definition index.h:218
Represents a closed-open range of indices [lower, upper)
Definition index.h:283
HERMES_DEVICE_CALLABLE T area() const
Definition index.h:348
HERMES_DEVICE_CALLABLE bool operator==(const Index2Range< T > &r) const
Definition index.h:329
HERMES_DEVICE_CALLABLE const Index2< T > & upper() const
Definition index.h:346
HERMES_DEVICE_CALLABLE bool contains(const Index2< T > &ij) const
Definition index.h:320
HERMES_DEVICE_CALLABLE Index2Range(T upper_i, T upper_j)
Constructs an index range [0, {upper_i,upper_j})
Definition index.h:307
HERMES_DEVICE_CALLABLE Index2Range(size2 upper)
Constructs an index range [0, upper)
Definition index.h:316
HERMES_DEVICE_CALLABLE friend Index2Range< T > intersect(const Index2Range< T > &a, const Index2Range< T > &b)
Definition index.h:291
HERMES_DEVICE_CALLABLE Index2Range(Index2< T > lower, Index2< T > upper)
Constructs an index range [lower, upper)
Definition index.h:312
HERMES_DEVICE_CALLABLE Index2Iterator< T > begin() const
Definition index.h:336
HERMES_DEVICE_CALLABLE const Index2< T > & lower() const
Definition index.h:344
HERMES_DEVICE_CALLABLE Index2Iterator< T > end() const
Definition index.h:340
Definition index.h:455
HERMES_DEVICE_CALLABLE bool operator==(const Index3Iterator< T > &other) const
are equal? operator
Definition index.h:505
HERMES_DEVICE_CALLABLE bool operator!=(const Index3Iterator< T > &other) const
are different? operator
Definition index.h:511
HERMES_DEVICE_CALLABLE size_t flatIndex() const
Computes a flat index based on size.
Definition index.h:481
HERMES_DEVICE_CALLABLE Index3Iterator(Index3< T > lower, Index3< T > upper, Index3< T > start)
Construct a new Index3Iterator object.
Definition index.h:465
HERMES_DEVICE_CALLABLE Index3Iterator & operator++()
Definition index.h:487
HERMES_DEVICE_CALLABLE const Index3< T > & operator*() const
Definition index.h:474
HERMES_DEVICE_CALLABLE Index3Iterator(Index3< T > upper)
Definition index.h:468
Represents a closed-open range of indices [lower, upper),.
Definition index.h:532
HERMES_DEVICE_CALLABLE Index3Range(size3 upper)
Definition index.h:568
HERMES_DEVICE_CALLABLE Index3Iterator< T > end() const
Definition index.h:594
HERMES_DEVICE_CALLABLE size3 size() const
Definition index.h:599
HERMES_DEVICE_CALLABLE friend Index3Range< T > intersect(const Index3Range< T > &a, const Index3Range< T > &b)
Definition index.h:541
HERMES_DEVICE_CALLABLE size_t flatIndex(const Index3< T > &ijk) const
Computes a flat index based on size.
Definition index.h:585
HERMES_DEVICE_CALLABLE Index3Range(Index3< T > lower, Index3< T > upper)
Definition index.h:573
HERMES_DEVICE_CALLABLE Index3Range(Index3< T > upper)
Construct a new Index3Range object.
Definition index.h:565
HERMES_DEVICE_CALLABLE Index3Range(T upper_i, T upper_j, T upper_k)
Construct a new Index3Range object.
Definition index.h:561
HERMES_DEVICE_CALLABLE Index3Iterator< T > begin() const
Definition index.h:590
Holds 2-dimensional size.
Definition size.h:47
Holds 2-dimensional size.
Definition size.h:142
Debug, logging and assertion macros.
#define HERMES_UNUSED_VARIABLE(x)
Specifies that variable is not used in this scope.
Definition debug.h:62
#define HERMES_DEVICE_CALLABLE
Specifies that the function can be called from both host and device sides.
Definition defs.h:45
#define HERMES_NOT_IMPLEMENTED
Logs "calling code not implemented" warning.
Definition debug.h:67
#define ARITHMETIC_OP(OP)
asd
Definition index.h:59
Set of multi-dimensional size representations.
Holds 2-dimensional integer index coordinates.
Definition index.h:50
HERMES_DEVICE_CALLABLE Index2()
Default constructor.
Definition index.h:102
HERMES_DEVICE_CALLABLE void clampTo(const size2 &s)
Clamps to the inclusive range [0, size]
Definition index.h:187
T i
0-th coordinate value
Definition index.h:199
HERMES_DEVICE_CALLABLE T & operator[](int d)
Definition index.h:129
T j
1-th coordinate value
Definition index.h:201
HERMES_DEVICE_CALLABLE Index2(T v)
Constructor.
Definition index.h:105
HERMES_DEVICE_CALLABLE Index2< T > right(T d=T(1)) const
Generates a copy with i incremented by d
Definition index.h:176
HERMES_DEVICE_CALLABLE friend T distance(const Index2< T > &a, const Index2< T > &b)
Computes the manhattan distance between two indices.
Definition index.h:79
HERMES_DEVICE_CALLABLE Index2(const Size2< S > &size)
Constructor from a Size2 object.
Definition index.h:115
HERMES_DEVICE_CALLABLE Index2< T > left(T d=T(1)) const
Generates a copy with i decremented by d
Definition index.h:172
HERMES_DEVICE_CALLABLE Index2< T > plus(T _i, T _j) const
Generates an index with incremented values.
Definition index.h:168
HERMES_DEVICE_CALLABLE Index2< T > up(T d=T(1)) const
Generates a copy with j incremented by d
Definition index.h:184
HERMES_DEVICE_CALLABLE Index2(T i, T j)
Constructor.
Definition index.h:109
HERMES_DEVICE_CALLABLE Index2< T > down(T d=T(1)) const
Generates a copy with j decremented by d
Definition index.h:180
HERMES_DEVICE_CALLABLE T operator[](int d) const
Definition index.h:124
Holds 3-dimensional index coordinates.
Definition index.h:362
HERMES_DEVICE_CALLABLE T & operator[](int _i)
Definition index.h:406
HERMES_DEVICE_CALLABLE T operator[](int _i) const
Definition index.h:401
T k
2-th coordinate value
Definition index.h:448
HERMES_DEVICE_CALLABLE Index3(T i, T j, T k)
Construct a new Index2 object.
Definition index.h:392
T j
1-th coordinate value
Definition index.h:446
HERMES_DEVICE_CALLABLE Index3(T v)
Definition index.h:387
T i
0-th coordinate value
Definition index.h:444