#pragma once namespace nall { template struct iterator { iterator(T* self, uint64_t offset) : _self(self), _offset(offset) {} auto operator*() -> T& { return _self[_offset]; } auto operator!=(const iterator& source) const -> bool { return _offset != source._offset; } auto operator++() -> iterator& { return _offset++, *this; } auto offset() const -> uint64_t { return _offset; } private: T* _self; uint64_t _offset; }; template struct iterator_const { iterator_const(const T* self, uint64_t offset) : _self(self), _offset(offset) {} auto operator*() -> const T& { return _self[_offset]; } auto operator!=(const iterator_const& source) const -> bool { return _offset != source._offset; } auto operator++() -> iterator_const& { return _offset++, *this; } auto offset() const -> uint64_t { return _offset; } private: const T* _self; uint64_t _offset; }; template struct reverse_iterator { reverse_iterator(T* self, uint64_t offset) : _self(self), _offset(offset) {} auto operator*() -> T& { return _self[_offset]; } auto operator!=(const reverse_iterator& source) const -> bool { return _offset != source._offset; } auto operator++() -> reverse_iterator& { return _offset--, *this; } auto offset() const -> uint64_t { return _offset; } private: T* _self; uint64_t _offset; }; template struct reverse_iterator_const { reverse_iterator_const(const T* self, uint64_t offset) : _self(self), _offset(offset) {} auto operator*() -> const T& { return _self[_offset]; } auto operator!=(const reverse_iterator_const& source) const -> bool { return _offset != source._offset; } auto operator++() -> reverse_iterator_const& { return _offset--, *this; } auto offset() const -> uint64_t { return _offset; } private: const T* _self; uint64_t _offset; }; //std::rbegin(), std::rend() is missing from GCC 4.9; which I still target template auto rbegin(T (&array)[Size]) { return reverse_iterator{array, Size - 1}; } template auto rend(T (&array)[Size]) { return reverse_iterator{array, (uint64_t)-1}; } template auto rbegin(T& self) { return self.rbegin(); } template auto rend(T& self) { return self.rend(); } template struct reverse_wrapper { auto begin() { return rbegin(_self); } auto end() { return rend(_self); } auto begin() const { return rbegin(_self); } auto end() const { return rend(_self); } T _self; }; template auto reverse(T& object) -> reverse_wrapper { return {object}; } template auto reverse(T&& object) -> reverse_wrapper { return {object}; } }