mirror of
https://github.com/microsoft/GSL.git
synced 2024-11-03 17:56:43 -05:00
Support iterator conversions as per Issue #252
This commit is contained in:
commit
85b5c3770c
@ -1066,12 +1066,14 @@ struct dim_t<dynamic_range>
|
|||||||
};
|
};
|
||||||
|
|
||||||
template <std::ptrdiff_t N>
|
template <std::ptrdiff_t N>
|
||||||
constexpr std::enable_if_t<(N >= 0),dim_t<N>> dim() noexcept {
|
constexpr std::enable_if_t<(N >= 0), dim_t<N>> dim() noexcept
|
||||||
|
{
|
||||||
return dim_t<N>();
|
return dim_t<N>();
|
||||||
}
|
}
|
||||||
|
|
||||||
template <std::ptrdiff_t N = dynamic_range>
|
template <std::ptrdiff_t N = dynamic_range>
|
||||||
constexpr std::enable_if_t<N == dynamic_range,dim_t<N>> dim(std::ptrdiff_t n) noexcept {
|
constexpr std::enable_if_t<N == dynamic_range, dim_t<N>> dim(std::ptrdiff_t n) noexcept
|
||||||
|
{
|
||||||
return dim_t<>(n);
|
return dim_t<>(n);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
225
gsl/span
225
gsl/span
@ -166,159 +166,33 @@ namespace details
|
|||||||
{
|
{
|
||||||
};
|
};
|
||||||
|
|
||||||
template <class Span>
|
template <class Span, bool IsConst>
|
||||||
class const_span_iterator
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
using iterator_category = std::random_access_iterator_tag;
|
|
||||||
using value_type = typename Span::element_type;
|
|
||||||
using difference_type = std::ptrdiff_t;
|
|
||||||
|
|
||||||
using const_pointer = std::add_const_t<value_type*>;
|
|
||||||
using pointer = const_pointer;
|
|
||||||
|
|
||||||
using const_reference = std::add_const_t<value_type&>;
|
|
||||||
using reference = const_reference;
|
|
||||||
|
|
||||||
constexpr const_span_iterator() : const_span_iterator(nullptr, 0) {}
|
|
||||||
constexpr const_span_iterator(const Span* span, typename Span::index_type index)
|
|
||||||
: span_(span), index_(index)
|
|
||||||
{
|
|
||||||
Expects(span == nullptr || (index_ >= 0 && index <= span_->length()));
|
|
||||||
}
|
|
||||||
|
|
||||||
constexpr reference operator*() const
|
|
||||||
{
|
|
||||||
Expects(span_);
|
|
||||||
return (*span_)[index_];
|
|
||||||
}
|
|
||||||
|
|
||||||
constexpr pointer operator->() const
|
|
||||||
{
|
|
||||||
Expects(span_);
|
|
||||||
return &((*span_)[index_]);
|
|
||||||
}
|
|
||||||
|
|
||||||
constexpr const_span_iterator& operator++() noexcept
|
|
||||||
{
|
|
||||||
Expects(span_ && index_ >= 0 && index_ < span_->length());
|
|
||||||
++index_;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
|
|
||||||
constexpr const_span_iterator operator++(int) noexcept
|
|
||||||
{
|
|
||||||
auto ret = *this;
|
|
||||||
++(*this);
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
constexpr const_span_iterator& operator--() noexcept
|
|
||||||
{
|
|
||||||
Expects(span_ && index_ > 0 && index_ <= span_->length());
|
|
||||||
--index_;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
|
|
||||||
constexpr const_span_iterator operator--(int) noexcept
|
|
||||||
{
|
|
||||||
auto ret = *this;
|
|
||||||
--(*this);
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
constexpr const_span_iterator operator+(difference_type n) const noexcept
|
|
||||||
{
|
|
||||||
auto ret = *this;
|
|
||||||
return ret += n;
|
|
||||||
}
|
|
||||||
|
|
||||||
constexpr const_span_iterator& operator+=(difference_type n) noexcept
|
|
||||||
{
|
|
||||||
Expects(span_ && (index_ + n) >= 0 && (index_ + n) <= span_->length());
|
|
||||||
index_ += n;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
|
|
||||||
constexpr const_span_iterator operator-(difference_type n) const noexcept
|
|
||||||
{
|
|
||||||
auto ret = *this;
|
|
||||||
return ret -= n;
|
|
||||||
}
|
|
||||||
|
|
||||||
constexpr const_span_iterator& operator-=(difference_type n) noexcept
|
|
||||||
{
|
|
||||||
return *this += -n;
|
|
||||||
}
|
|
||||||
|
|
||||||
constexpr difference_type operator-(const const_span_iterator& rhs) const noexcept
|
|
||||||
{
|
|
||||||
Expects(span_ == rhs.span_);
|
|
||||||
return index_ - rhs.index_;
|
|
||||||
}
|
|
||||||
|
|
||||||
constexpr reference operator[](difference_type n) const noexcept { return *(*this + n); }
|
|
||||||
|
|
||||||
constexpr bool operator==(const const_span_iterator& rhs) const noexcept
|
|
||||||
{
|
|
||||||
return span_ == rhs.span_ && index_ == rhs.index_;
|
|
||||||
}
|
|
||||||
|
|
||||||
constexpr bool operator!=(const const_span_iterator& rhs) const noexcept
|
|
||||||
{
|
|
||||||
return !(*this == rhs);
|
|
||||||
}
|
|
||||||
|
|
||||||
constexpr bool operator<(const const_span_iterator& rhs) const noexcept
|
|
||||||
{
|
|
||||||
Expects(span_ == rhs.span_);
|
|
||||||
return index_ < rhs.index_;
|
|
||||||
}
|
|
||||||
|
|
||||||
constexpr bool operator<=(const const_span_iterator& rhs) const noexcept
|
|
||||||
{
|
|
||||||
return !(rhs < *this);
|
|
||||||
}
|
|
||||||
|
|
||||||
constexpr bool operator>(const const_span_iterator& rhs) const noexcept
|
|
||||||
{
|
|
||||||
return rhs < *this;
|
|
||||||
}
|
|
||||||
|
|
||||||
constexpr bool operator>=(const const_span_iterator& rhs) const noexcept
|
|
||||||
{
|
|
||||||
return !(rhs > *this);
|
|
||||||
}
|
|
||||||
|
|
||||||
void swap(const_span_iterator& rhs) noexcept
|
|
||||||
{
|
|
||||||
std::swap(index_, rhs.index_);
|
|
||||||
std::swap(span_, rhs.span_);
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
const Span* span_;
|
|
||||||
std::ptrdiff_t index_;
|
|
||||||
};
|
|
||||||
|
|
||||||
template <class Span>
|
|
||||||
class span_iterator
|
class span_iterator
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
using iterator_category = std::random_access_iterator_tag;
|
using iterator_category = std::random_access_iterator_tag;
|
||||||
using value_type = typename Span::element_type;
|
using value_type =
|
||||||
using difference_type = std::ptrdiff_t;
|
std::conditional_t<IsConst, std::add_const_t<typename Span::element_type>,
|
||||||
|
typename Span::element_type>;
|
||||||
|
using difference_type = typename Span::index_type;
|
||||||
|
|
||||||
using pointer = value_type*;
|
using pointer = std::add_pointer_t<value_type>;
|
||||||
using reference = value_type&;
|
using reference = std::add_lvalue_reference_t<value_type>;
|
||||||
|
|
||||||
|
constexpr span_iterator() : span_iterator(nullptr, 0) noexcept {}
|
||||||
|
|
||||||
constexpr span_iterator() : span_iterator(nullptr, 0) {}
|
|
||||||
constexpr span_iterator(const Span* span, typename Span::index_type index)
|
constexpr span_iterator(const Span* span, typename Span::index_type index)
|
||||||
: span_(span), index_(index)
|
: span_(span), index_(index)
|
||||||
{
|
{
|
||||||
Expects(span == nullptr || (index_ >= 0 && index <= span_->length()));
|
Expects(span == nullptr || (index_ >= 0 && index <= span_->length()));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
friend class span_iterator<Span, true>;
|
||||||
|
constexpr span_iterator(const span_iterator<Span, false>& other) noexcept
|
||||||
|
: span_iterator(other.span_, other.index_)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
constexpr reference operator*() const
|
constexpr reference operator*() const
|
||||||
{
|
{
|
||||||
Expects(span_);
|
Expects(span_);
|
||||||
@ -388,32 +262,39 @@ namespace details
|
|||||||
|
|
||||||
constexpr reference operator[](difference_type n) const noexcept { return *(*this + n); }
|
constexpr reference operator[](difference_type n) const noexcept { return *(*this + n); }
|
||||||
|
|
||||||
constexpr bool operator==(const span_iterator& rhs) const noexcept
|
constexpr friend bool operator==(const span_iterator& lhs,
|
||||||
|
const span_iterator& rhs) noexcept
|
||||||
{
|
{
|
||||||
return span_ == rhs.span_ && index_ == rhs.index_;
|
return lhs.span_ == rhs.span_ && lhs.index_ == rhs.index_;
|
||||||
}
|
}
|
||||||
|
|
||||||
constexpr bool operator!=(const span_iterator& rhs) const noexcept
|
constexpr friend bool operator!=(const span_iterator& lhs,
|
||||||
|
const span_iterator& rhs) noexcept
|
||||||
{
|
{
|
||||||
return !(*this == rhs);
|
return !(lhs == rhs);
|
||||||
}
|
}
|
||||||
|
|
||||||
constexpr bool operator<(const span_iterator& rhs) const noexcept
|
constexpr friend bool operator<(const span_iterator& lhs, const span_iterator& rhs) noexcept
|
||||||
{
|
{
|
||||||
Expects(span_ == rhs.span_);
|
Expects(lhs.span_ == rhs.span_);
|
||||||
return index_ < rhs.index_;
|
return lhs.index_ < rhs.index_;
|
||||||
}
|
}
|
||||||
|
|
||||||
constexpr bool operator<=(const span_iterator& rhs) const noexcept
|
constexpr friend bool operator<=(const span_iterator& lhs,
|
||||||
|
const span_iterator& rhs) noexcept
|
||||||
{
|
{
|
||||||
return !(rhs < *this);
|
return !(rhs < lhs);
|
||||||
}
|
}
|
||||||
|
|
||||||
constexpr bool operator>(const span_iterator& rhs) const noexcept { return rhs < *this; }
|
constexpr friend bool operator>(const span_iterator& lhs, const span_iterator& rhs) noexcept
|
||||||
|
|
||||||
constexpr bool operator>=(const span_iterator& rhs) const noexcept
|
|
||||||
{
|
{
|
||||||
return !(rhs > *this);
|
return rhs < lhs;
|
||||||
|
}
|
||||||
|
|
||||||
|
constexpr friend bool operator>=(const span_iterator& lhs,
|
||||||
|
const span_iterator& rhs) noexcept
|
||||||
|
{
|
||||||
|
return !(rhs > lhs);
|
||||||
}
|
}
|
||||||
|
|
||||||
void swap(span_iterator& rhs) noexcept
|
void swap(span_iterator& rhs) noexcept
|
||||||
@ -422,37 +303,23 @@ namespace details
|
|||||||
std::swap(span_, rhs.span_);
|
std::swap(span_, rhs.span_);
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
protected:
|
||||||
const Span* span_;
|
const Span* span_;
|
||||||
std::ptrdiff_t index_;
|
std::ptrdiff_t index_;
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename Span>
|
template <class Span, bool IsConst>
|
||||||
constexpr const_span_iterator<Span>
|
constexpr span_iterator<Span, IsConst>
|
||||||
operator+(typename const_span_iterator<Span>::difference_type n,
|
operator+(typename span_iterator<Span, IsConst>::difference_type n,
|
||||||
const const_span_iterator<Span>& rhs) noexcept
|
const span_iterator<Span, IsConst>& rhs) noexcept
|
||||||
{
|
{
|
||||||
return rhs + n;
|
return rhs + n;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename Span>
|
template <class Span, bool IsConst>
|
||||||
constexpr const_span_iterator<Span>
|
constexpr span_iterator<Span, IsConst>
|
||||||
operator-(typename const_span_iterator<Span>::difference_type n,
|
operator-(typename span_iterator<Span, IsConst>::difference_type n,
|
||||||
const const_span_iterator<Span>& rhs) noexcept
|
const span_iterator<Span, IsConst>& rhs) noexcept
|
||||||
{
|
|
||||||
return rhs - n;
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename Span>
|
|
||||||
constexpr span_iterator<Span> operator+(typename span_iterator<Span>::difference_type n,
|
|
||||||
const span_iterator<Span>& rhs) noexcept
|
|
||||||
{
|
|
||||||
return rhs + n;
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename Span>
|
|
||||||
constexpr span_iterator<Span> operator-(typename span_iterator<Span>::difference_type n,
|
|
||||||
const span_iterator<Span>& rhs) noexcept
|
|
||||||
{
|
{
|
||||||
return rhs - n;
|
return rhs - n;
|
||||||
}
|
}
|
||||||
@ -511,8 +378,8 @@ public:
|
|||||||
using pointer = element_type*;
|
using pointer = element_type*;
|
||||||
using reference = element_type&;
|
using reference = element_type&;
|
||||||
|
|
||||||
using iterator = details::span_iterator<span<ElementType, Extent>>;
|
using iterator = details::span_iterator<span<ElementType, Extent>, false>;
|
||||||
using const_iterator = details::const_span_iterator<span<ElementType, Extent>>;
|
using const_iterator = details::span_iterator<span<ElementType, Extent>, true>;
|
||||||
using reverse_iterator = std::reverse_iterator<iterator>;
|
using reverse_iterator = std::reverse_iterator<iterator>;
|
||||||
using const_reverse_iterator = std::reverse_iterator<const_iterator>;
|
using const_reverse_iterator = std::reverse_iterator<const_iterator>;
|
||||||
|
|
||||||
|
@ -787,22 +787,104 @@ SUITE(span_tests)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST(iterator)
|
TEST(iterator_default_init)
|
||||||
{
|
{
|
||||||
span<int>::iterator it1;
|
span<int>::iterator it1;
|
||||||
span<int>::iterator it2;
|
span<int>::iterator it2;
|
||||||
CHECK(it1 == it2);
|
CHECK(it1 == it2);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST(const_iterator)
|
TEST(const_iterator_default_init)
|
||||||
{
|
{
|
||||||
span<int>::const_iterator it1;
|
span<int>::const_iterator it1;
|
||||||
span<int>::const_iterator it2;
|
span<int>::const_iterator it2;
|
||||||
CHECK(it1 == it2);
|
CHECK(it1 == it2);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TEST(iterator_conversions)
|
||||||
|
{
|
||||||
|
span<int>::iterator badIt;
|
||||||
|
span<int>::const_iterator badConstIt;
|
||||||
|
CHECK(badIt == badConstIt);
|
||||||
|
|
||||||
|
int a[] = { 1, 2, 3, 4 };
|
||||||
|
span<int> s = a;
|
||||||
|
|
||||||
|
auto it = s.begin();
|
||||||
|
auto cit = s.cbegin();
|
||||||
|
|
||||||
|
CHECK(it == cit);
|
||||||
|
CHECK(cit == it);
|
||||||
|
|
||||||
|
span<int>::const_iterator cit2 = it;
|
||||||
|
CHECK(cit2 == cit);
|
||||||
|
|
||||||
|
span<int>::const_iterator cit3 = it + 4;
|
||||||
|
CHECK(cit3 == s.cend());
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(iterator_comparisons)
|
||||||
|
{
|
||||||
|
int a[] = { 1, 2, 3, 4 };
|
||||||
|
{
|
||||||
|
span<int> s = a;
|
||||||
|
span<int>::iterator it = s.begin();
|
||||||
|
auto it2 = it + 1;
|
||||||
|
span<int>::const_iterator cit = s.cbegin();
|
||||||
|
|
||||||
|
CHECK(it == cit);
|
||||||
|
CHECK(cit == it);
|
||||||
|
CHECK(it == it);
|
||||||
|
CHECK(cit == cit);
|
||||||
|
CHECK(cit == s.begin());
|
||||||
|
CHECK(s.begin() == cit);
|
||||||
|
CHECK(s.cbegin() == cit);
|
||||||
|
CHECK(it == s.begin());
|
||||||
|
CHECK(s.begin() == it);
|
||||||
|
|
||||||
|
CHECK(it != it2);
|
||||||
|
CHECK(it2 != it);
|
||||||
|
CHECK(it != s.end());
|
||||||
|
CHECK(it2 != s.end());
|
||||||
|
CHECK(s.end() != it);
|
||||||
|
CHECK(it2 != cit);
|
||||||
|
CHECK(cit != it2);
|
||||||
|
|
||||||
|
CHECK(it < it2);
|
||||||
|
CHECK(it <= it2);
|
||||||
|
CHECK(it2 <= s.end());
|
||||||
|
CHECK(it < s.end());
|
||||||
|
CHECK(it <= cit);
|
||||||
|
CHECK(cit <= it);
|
||||||
|
CHECK(cit < it2);
|
||||||
|
CHECK(cit <= it2);
|
||||||
|
CHECK(cit < s.end());
|
||||||
|
CHECK(cit <= s.end());
|
||||||
|
|
||||||
|
CHECK(it2 > it);
|
||||||
|
CHECK(it2 >= it);
|
||||||
|
CHECK(s.end() > it2);
|
||||||
|
CHECK(s.end() >= it2);
|
||||||
|
CHECK(it2 > cit);
|
||||||
|
CHECK(it2 >= cit);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
TEST(begin_end)
|
TEST(begin_end)
|
||||||
{
|
{
|
||||||
|
{
|
||||||
|
int a[] = { 1, 2, 3, 4 };
|
||||||
|
span<int> s = a;
|
||||||
|
|
||||||
|
span<int>::iterator it = s.begin();
|
||||||
|
span<int>::iterator it2 = std::begin(s);
|
||||||
|
CHECK(it == it2);
|
||||||
|
|
||||||
|
it = s.end();
|
||||||
|
it2 = std::end(s);
|
||||||
|
CHECK(it == it2);
|
||||||
|
}
|
||||||
|
|
||||||
{
|
{
|
||||||
int a[] = { 1, 2, 3, 4 };
|
int a[] = { 1, 2, 3, 4 };
|
||||||
span<int> s = a;
|
span<int> s = a;
|
||||||
@ -847,6 +929,19 @@ SUITE(span_tests)
|
|||||||
|
|
||||||
TEST(cbegin_cend)
|
TEST(cbegin_cend)
|
||||||
{
|
{
|
||||||
|
{
|
||||||
|
int a[] = { 1, 2, 3, 4 };
|
||||||
|
span<int> s = a;
|
||||||
|
|
||||||
|
span<int>::const_iterator cit = s.cbegin();
|
||||||
|
span<int>::const_iterator cit2 = std::cbegin(s);
|
||||||
|
CHECK(cit == cit2);
|
||||||
|
|
||||||
|
cit = s.cend();
|
||||||
|
cit2 = std::cend(s);
|
||||||
|
CHECK(cit == cit2);
|
||||||
|
}
|
||||||
|
|
||||||
{
|
{
|
||||||
int a[] = {1, 2, 3, 4};
|
int a[] = {1, 2, 3, 4};
|
||||||
span<int> s = a;
|
span<int> s = a;
|
||||||
@ -867,25 +962,21 @@ SUITE(span_tests)
|
|||||||
++it;
|
++it;
|
||||||
CHECK(it - first == 1);
|
CHECK(it - first == 1);
|
||||||
CHECK(*it == 2);
|
CHECK(*it == 2);
|
||||||
*it = 22;
|
|
||||||
CHECK(*it == 22);
|
|
||||||
CHECK(beyond - it == 3);
|
CHECK(beyond - it == 3);
|
||||||
|
|
||||||
|
int last = 0;
|
||||||
it = first;
|
it = first;
|
||||||
CHECK(it == first);
|
CHECK(it == first);
|
||||||
while (it != s.cend())
|
while (it != s.cend())
|
||||||
{
|
{
|
||||||
*it = 5;
|
CHECK(*it == last + 1);
|
||||||
|
|
||||||
|
last = *it;
|
||||||
++it;
|
++it;
|
||||||
}
|
}
|
||||||
|
|
||||||
CHECK(it == beyond);
|
CHECK(it == beyond);
|
||||||
CHECK(it - beyond == 0);
|
CHECK(it - beyond == 0);
|
||||||
|
|
||||||
for (auto& n : s)
|
|
||||||
{
|
|
||||||
CHECK(n == 5);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -955,25 +1046,21 @@ SUITE(span_tests)
|
|||||||
++it;
|
++it;
|
||||||
CHECK(it - first == 1);
|
CHECK(it - first == 1);
|
||||||
CHECK(*it == 3);
|
CHECK(*it == 3);
|
||||||
*it = 22;
|
|
||||||
CHECK(*it == 22);
|
|
||||||
CHECK(beyond - it == 3);
|
CHECK(beyond - it == 3);
|
||||||
|
|
||||||
it = first;
|
it = first;
|
||||||
CHECK(it == first);
|
CHECK(it == first);
|
||||||
|
int last = 5;
|
||||||
while (it != s.crend())
|
while (it != s.crend())
|
||||||
{
|
{
|
||||||
*it = 5;
|
CHECK(*it == last - 1);
|
||||||
|
last = *it;
|
||||||
|
|
||||||
++it;
|
++it;
|
||||||
}
|
}
|
||||||
|
|
||||||
CHECK(it == beyond);
|
CHECK(it == beyond);
|
||||||
CHECK(it - beyond == 0);
|
CHECK(it - beyond == 0);
|
||||||
|
|
||||||
for (auto& n : s)
|
|
||||||
{
|
|
||||||
CHECK(n == 5);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user