// Copyright 2004-present Facebook. All Rights Reserved.
#pragma once
#include <c10/util/Exception.h>
#include <iterator>
#include <type_traits>
namespace c10 {
namespace detail {
template <typename I, std::enable_if_t<std::is_integral<I>{}, int> = 0>
struct integer_iterator : std::iterator<std::input_iterator_tag, I> {
explicit integer_iterator(I value) : value(value) {}
I operator*() const { return value; }
I const* operator->() const { return &value; }
integer_iterator& operator++() {
++value;
return *this;
}
integer_iterator operator++(int) {
const auto copy = *this;
++*this;
return copy;
}
bool operator==(const integer_iterator& other) const {
return value == other.value;
}
bool operator!=(const integer_iterator& other) const {
return value != other.value;
}
protected:
I value;
};
} // namespace detail
template <typename I, std::enable_if_t<std::is_integral<I>{}, int> = 0>
struct integer_range {
public:
integer_range(I begin, I end) : begin_(begin), end_(end) {}
detail::integer_iterator<I> begin() const { return begin_; }
detail::integer_iterator<I> end() const { return end_; }
private:
detail::integer_iterator<I> begin_;
detail::integer_iterator<I> end_;
};
/// Creates an integer range for the half-open interval [begin, end)
/// If end<=begin, then the range is empty
template <typename Integer, std::enable_if_t<std::is_integral<Integer>::value, bool> = true>
integer_range<Integer> irange(Integer begin, Integer end) {
//If end<=begin then the range is empty; we can achieve this effect by
//choosing the larger of {begin, end} as the loop terminator
return {begin, std::max(begin, end)};
}
/// Creates an integer range for the half-open interval [0, end)
/// If end<=begin, then the range is empty
template <typename Integer, std::enable_if_t<std::is_integral<Integer>::value, bool> = true>
integer_range<Integer> irange(Integer end) {
//If end<=begin then the range is empty; we can achieve this effect by
//choosing the larger of {0, end} as the loop terminator
return {Integer(), std::max(Integer(), end)};
}
} // namespace torch