/* * Copyright (C) 2015 The Android Open Source Project * * 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. */ #ifndef ART_LIBARTBASE_BASE_STRIDE_ITERATOR_H_ #define ART_LIBARTBASE_BASE_STRIDE_ITERATOR_H_ #include #include namespace art { template class StrideIterator : public std::iterator { public: using difference_type = typename std::iterator::difference_type; StrideIterator(const StrideIterator&) = default; StrideIterator(StrideIterator&&) = default; StrideIterator& operator=(const StrideIterator&) = default; StrideIterator& operator=(StrideIterator&&) = default; StrideIterator(T* ptr, size_t stride) : ptr_(reinterpret_cast(ptr)), stride_(stride) {} bool operator==(const StrideIterator& other) const { DCHECK_EQ(stride_, other.stride_); return ptr_ == other.ptr_; } bool operator!=(const StrideIterator& other) const { return !(*this == other); } StrideIterator& operator++() { // Value after modification. ptr_ += stride_; return *this; } StrideIterator operator++(int) { StrideIterator temp = *this; ++*this; return temp; } StrideIterator& operator--() { // Value after modification. ptr_ -= stride_; return *this; } StrideIterator operator--(int) { StrideIterator temp = *this; --*this; return temp; } StrideIterator& operator+=(difference_type delta) { ptr_ += static_cast(stride_) * delta; return *this; } StrideIterator operator+(difference_type delta) const { StrideIterator temp = *this; temp += delta; return temp; } StrideIterator& operator-=(difference_type delta) { ptr_ -= static_cast(stride_) * delta; return *this; } StrideIterator operator-(difference_type delta) const { StrideIterator temp = *this; temp -= delta; return temp; } difference_type operator-(const StrideIterator& rhs) { DCHECK_EQ(stride_, rhs.stride_); DCHECK_EQ((ptr_ - rhs.ptr_) % stride_, 0u); return (ptr_ - rhs.ptr_) / stride_; } T& operator*() const { return *reinterpret_cast(ptr_); } T* operator->() const { return &**this; } T& operator[](difference_type n) { return *(*this + n); } private: uintptr_t ptr_; // Not const for operator=. size_t stride_; template friend bool operator<(const StrideIterator& lhs, const StrideIterator& rhs); }; template StrideIterator operator+(typename StrideIterator::difference_type dist, const StrideIterator& it) { return it + dist; } template bool operator<(const StrideIterator& lhs, const StrideIterator& rhs) { DCHECK_EQ(lhs.stride_, rhs.stride_); return lhs.ptr_ < rhs.ptr_; } template bool operator>(const StrideIterator& lhs, const StrideIterator& rhs) { return rhs < lhs; } template bool operator<=(const StrideIterator& lhs, const StrideIterator& rhs) { return !(rhs < lhs); } template bool operator>=(const StrideIterator& lhs, const StrideIterator& rhs) { return !(lhs < rhs); } } // namespace art #endif // ART_LIBARTBASE_BASE_STRIDE_ITERATOR_H_