2020-03-27 20:57:29 +00:00
|
|
|
#ifndef KITTY_UTIL_ITERATOR_H
|
|
|
|
#define KITTY_UTIL_ITERATOR_H
|
|
|
|
|
|
|
|
#include <iterator>
|
|
|
|
|
|
|
|
namespace util {
|
|
|
|
template<class V, class T>
|
|
|
|
class it_wrap_t : public std::iterator<std::random_access_iterator_tag, V> {
|
|
|
|
public:
|
|
|
|
typedef T iterator;
|
|
|
|
typedef typename std::iterator<std::random_access_iterator_tag, V>::value_type class_t;
|
|
|
|
|
2021-05-17 19:21:57 +00:00
|
|
|
typedef class_t &reference;
|
|
|
|
typedef class_t *pointer;
|
2020-03-27 20:57:29 +00:00
|
|
|
|
|
|
|
typedef std::ptrdiff_t diff_t;
|
|
|
|
|
2021-05-17 19:21:57 +00:00
|
|
|
iterator operator+=(diff_t step) {
|
2020-03-27 20:57:29 +00:00
|
|
|
while(step-- > 0) {
|
|
|
|
++_this();
|
|
|
|
}
|
|
|
|
|
|
|
|
return _this();
|
|
|
|
}
|
|
|
|
|
2021-05-17 19:21:57 +00:00
|
|
|
iterator operator-=(diff_t step) {
|
2020-03-27 20:57:29 +00:00
|
|
|
while(step-- > 0) {
|
|
|
|
--_this();
|
|
|
|
}
|
|
|
|
|
|
|
|
return _this();
|
|
|
|
}
|
|
|
|
|
2021-05-17 19:21:57 +00:00
|
|
|
iterator operator+(diff_t step) {
|
2020-03-27 20:57:29 +00:00
|
|
|
iterator new_ = _this();
|
|
|
|
|
|
|
|
return new_ += step;
|
|
|
|
}
|
|
|
|
|
2021-05-17 19:21:57 +00:00
|
|
|
iterator operator-(diff_t step) {
|
2020-03-27 20:57:29 +00:00
|
|
|
iterator new_ = _this();
|
|
|
|
|
|
|
|
return new_ -= step;
|
|
|
|
}
|
|
|
|
|
2021-05-17 19:21:57 +00:00
|
|
|
diff_t operator-(iterator first) {
|
2020-03-27 20:57:29 +00:00
|
|
|
diff_t step = 0;
|
|
|
|
while(first != _this()) {
|
|
|
|
++step;
|
|
|
|
++first;
|
|
|
|
}
|
|
|
|
|
|
|
|
return step;
|
|
|
|
}
|
|
|
|
|
2021-05-17 19:21:57 +00:00
|
|
|
iterator operator++() {
|
|
|
|
_this().inc();
|
|
|
|
return _this();
|
|
|
|
}
|
|
|
|
iterator operator--() {
|
|
|
|
_this().dec();
|
|
|
|
return _this();
|
|
|
|
}
|
2020-03-27 20:57:29 +00:00
|
|
|
|
|
|
|
iterator operator++(int) {
|
|
|
|
iterator new_ = _this();
|
|
|
|
|
|
|
|
++_this();
|
|
|
|
|
|
|
|
return new_;
|
|
|
|
}
|
|
|
|
|
|
|
|
iterator operator--(int) {
|
|
|
|
iterator new_ = _this();
|
|
|
|
|
|
|
|
--_this();
|
|
|
|
|
|
|
|
return new_;
|
|
|
|
}
|
|
|
|
|
|
|
|
reference operator*() { return *_this().get(); }
|
|
|
|
const reference operator*() const { return *_this().get(); }
|
|
|
|
|
|
|
|
pointer operator->() { return &*_this(); }
|
|
|
|
const pointer operator->() const { return &*_this(); }
|
|
|
|
|
2021-05-17 19:21:57 +00:00
|
|
|
bool operator!=(const iterator &other) const {
|
2020-03-27 20:57:29 +00:00
|
|
|
return !(_this() == other);
|
|
|
|
}
|
|
|
|
|
2021-05-17 19:21:57 +00:00
|
|
|
bool operator<(const iterator &other) const {
|
2020-03-27 20:57:29 +00:00
|
|
|
return !(_this() >= other);
|
|
|
|
}
|
|
|
|
|
2021-05-17 19:21:57 +00:00
|
|
|
bool operator>=(const iterator &other) const {
|
2020-03-27 20:57:29 +00:00
|
|
|
return _this() == other || _this() > other;
|
|
|
|
}
|
|
|
|
|
2021-05-17 19:21:57 +00:00
|
|
|
bool operator<=(const iterator &other) const {
|
2020-03-27 20:57:29 +00:00
|
|
|
return _this() == other || _this() < other;
|
|
|
|
}
|
|
|
|
|
2021-05-17 19:21:57 +00:00
|
|
|
bool operator==(const iterator &other) const { return _this().eq(other); };
|
|
|
|
bool operator>(const iterator &other) const { return _this().gt(other); }
|
2020-03-27 20:57:29 +00:00
|
|
|
|
2021-05-17 19:21:57 +00:00
|
|
|
private:
|
|
|
|
iterator &_this() { return *static_cast<iterator *>(this); }
|
|
|
|
const iterator &_this() const { return *static_cast<const iterator *>(this); }
|
2020-03-27 20:57:29 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
template<class V, class It>
|
|
|
|
class round_robin_t : public it_wrap_t<V, round_robin_t<V, It>> {
|
|
|
|
public:
|
|
|
|
using iterator = It;
|
2021-05-17 19:21:57 +00:00
|
|
|
using pointer = V *;
|
2020-03-27 20:57:29 +00:00
|
|
|
|
|
|
|
round_robin_t(iterator begin, iterator end) : _begin(begin), _end(end), _pos(begin) {}
|
|
|
|
|
|
|
|
void inc() {
|
|
|
|
++_pos;
|
|
|
|
|
|
|
|
if(_pos == _end) {
|
|
|
|
_pos = _begin;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-10 12:39:50 +00:00
|
|
|
void dec() {
|
2021-05-17 19:21:57 +00:00
|
|
|
if(_pos == _begin) {
|
2020-04-10 12:39:50 +00:00
|
|
|
_pos = _end;
|
|
|
|
}
|
2021-05-17 19:21:57 +00:00
|
|
|
|
2020-04-10 12:39:50 +00:00
|
|
|
--_pos;
|
|
|
|
}
|
|
|
|
|
2020-03-27 20:57:29 +00:00
|
|
|
bool eq(const round_robin_t &other) const {
|
|
|
|
return *_pos == *other._pos;
|
|
|
|
}
|
|
|
|
|
|
|
|
pointer get() const {
|
|
|
|
return &*_pos;
|
|
|
|
}
|
2021-05-17 19:21:57 +00:00
|
|
|
|
2020-03-27 20:57:29 +00:00
|
|
|
private:
|
|
|
|
It _begin;
|
|
|
|
It _end;
|
|
|
|
|
|
|
|
It _pos;
|
|
|
|
};
|
|
|
|
|
|
|
|
template<class V, class It>
|
|
|
|
round_robin_t<V, It> make_round_robin(It begin, It end) {
|
|
|
|
return round_robin_t<V, It>(begin, end);
|
|
|
|
}
|
2021-05-17 19:21:57 +00:00
|
|
|
} // namespace util
|
2020-03-27 20:57:29 +00:00
|
|
|
|
|
|
|
#endif
|