forked from OSchip/llvm-project
Add llvm::enumerate() range adapter.
This allows you to enumerate over a range using a range-based for while the return type contains the index of the enumeration. Differential revision: https://reviews.llvm.org/D25124 llvm-svn: 283337
This commit is contained in:
parent
24db10d8e1
commit
aa0a562bd7
|
@ -35,7 +35,7 @@ namespace llvm {
|
|||
namespace detail {
|
||||
|
||||
template <typename RangeT>
|
||||
using IterOfRange = decltype(std::begin(std::declval<RangeT>()));
|
||||
using IterOfRange = decltype(std::begin(std::declval<RangeT &>()));
|
||||
|
||||
} // End detail namespace
|
||||
|
||||
|
@ -627,7 +627,7 @@ template <typename T> struct deref {
|
|||
};
|
||||
|
||||
namespace detail {
|
||||
template <typename I, typename V> class enumerator_impl {
|
||||
template <typename R> class enumerator_impl {
|
||||
public:
|
||||
template <typename X> struct result_pair {
|
||||
result_pair(std::size_t Index, X Value) : Index(Index), Value(Value) {}
|
||||
|
@ -636,13 +636,16 @@ public:
|
|||
X Value;
|
||||
};
|
||||
|
||||
struct iterator {
|
||||
iterator(I Iter, std::size_t Index) : Iter(Iter), Index(Index) {}
|
||||
class iterator {
|
||||
typedef
|
||||
typename std::iterator_traits<IterOfRange<R>>::reference iter_reference;
|
||||
typedef result_pair<iter_reference> result_type;
|
||||
|
||||
result_pair<const V> operator*() const {
|
||||
return result_pair<const V>(Index, *Iter);
|
||||
}
|
||||
result_pair<V> operator*() { return result_pair<V>(Index, *Iter); }
|
||||
public:
|
||||
iterator(IterOfRange<R> &&Iter, std::size_t Index)
|
||||
: Iter(Iter), Index(Index) {}
|
||||
|
||||
result_type operator*() const { return result_type(Index, *Iter); }
|
||||
|
||||
iterator &operator++() {
|
||||
++Iter;
|
||||
|
@ -653,28 +656,19 @@ public:
|
|||
bool operator!=(const iterator &RHS) const { return Iter != RHS.Iter; }
|
||||
|
||||
private:
|
||||
I Iter;
|
||||
IterOfRange<R> Iter;
|
||||
std::size_t Index;
|
||||
};
|
||||
|
||||
enumerator_impl(I Begin, I End)
|
||||
: Begin(std::move(Begin)), End(std::move(End)) {}
|
||||
public:
|
||||
explicit enumerator_impl(R &&Range) : Range(std::forward<R>(Range)) {}
|
||||
|
||||
iterator begin() { return iterator(Begin, 0); }
|
||||
iterator end() { return iterator(End, std::size_t(-1)); }
|
||||
|
||||
iterator begin() const { return iterator(Begin, 0); }
|
||||
iterator end() const { return iterator(End, std::size_t(-1)); }
|
||||
iterator begin() { return iterator(std::begin(Range), 0); }
|
||||
iterator end() { return iterator(std::end(Range), std::size_t(-1)); }
|
||||
|
||||
private:
|
||||
I Begin;
|
||||
I End;
|
||||
R Range;
|
||||
};
|
||||
|
||||
template <typename I>
|
||||
auto make_enumerator(I Begin, I End) -> enumerator_impl<I, decltype(*Begin)> {
|
||||
return enumerator_impl<I, decltype(*Begin)>(std::move(Begin), std::move(End));
|
||||
}
|
||||
}
|
||||
|
||||
/// Given an input range, returns a new range whose values are are pair (A,B)
|
||||
|
@ -692,10 +686,8 @@ auto make_enumerator(I Begin, I End) -> enumerator_impl<I, decltype(*Begin)> {
|
|||
/// Item 2 - C
|
||||
/// Item 3 - D
|
||||
///
|
||||
template <typename R>
|
||||
auto enumerate(R &&Range)
|
||||
-> decltype(detail::make_enumerator(std::begin(Range), std::end(Range))) {
|
||||
return detail::make_enumerator(std::begin(Range), std::end(Range));
|
||||
template <typename R> detail::enumerator_impl<R> enumerate(R &&Range) {
|
||||
return detail::enumerator_impl<R>(std::forward<R>(Range));
|
||||
}
|
||||
|
||||
} // End llvm namespace
|
||||
|
|
|
@ -39,51 +39,152 @@ TEST(STLExtrasTest, Rank) {
|
|||
EXPECT_EQ(4, f(rank<6>()));
|
||||
}
|
||||
|
||||
TEST(STLExtrasTest, Enumerate) {
|
||||
TEST(STLExtrasTest, EnumerateLValue) {
|
||||
// Test that a simple LValue can be enumerated and gives correct results with
|
||||
// multiple types, including the empty container.
|
||||
std::vector<char> foo = {'a', 'b', 'c'};
|
||||
|
||||
std::vector<std::pair<std::size_t, char>> results;
|
||||
std::vector<std::pair<std::size_t, char>> CharResults;
|
||||
|
||||
for (auto X : llvm::enumerate(foo)) {
|
||||
results.push_back(std::make_pair(X.Index, X.Value));
|
||||
CharResults.emplace_back(X.Index, X.Value);
|
||||
}
|
||||
ASSERT_EQ(3u, results.size());
|
||||
EXPECT_EQ(0u, results[0].first);
|
||||
EXPECT_EQ('a', results[0].second);
|
||||
EXPECT_EQ(1u, results[1].first);
|
||||
EXPECT_EQ('b', results[1].second);
|
||||
EXPECT_EQ(2u, results[2].first);
|
||||
EXPECT_EQ('c', results[2].second);
|
||||
ASSERT_EQ(3u, CharResults.size());
|
||||
EXPECT_EQ(std::make_pair(0u, 'a'), CharResults[0]);
|
||||
EXPECT_EQ(std::make_pair(1u, 'b'), CharResults[1]);
|
||||
EXPECT_EQ(std::make_pair(2u, 'c'), CharResults[2]);
|
||||
|
||||
results.clear();
|
||||
const std::vector<int> bar = {'1', '2', '3'};
|
||||
// Test a const range of a different type.
|
||||
std::vector<std::pair<std::size_t, int>> IntResults;
|
||||
const std::vector<int> bar = {1, 2, 3};
|
||||
for (auto X : llvm::enumerate(bar)) {
|
||||
results.push_back(std::make_pair(X.Index, X.Value));
|
||||
IntResults.emplace_back(X.Index, X.Value);
|
||||
}
|
||||
EXPECT_EQ(0u, results[0].first);
|
||||
EXPECT_EQ('1', results[0].second);
|
||||
EXPECT_EQ(1u, results[1].first);
|
||||
EXPECT_EQ('2', results[1].second);
|
||||
EXPECT_EQ(2u, results[2].first);
|
||||
EXPECT_EQ('3', results[2].second);
|
||||
ASSERT_EQ(3u, IntResults.size());
|
||||
EXPECT_EQ(std::make_pair(0u, 1), IntResults[0]);
|
||||
EXPECT_EQ(std::make_pair(1u, 2), IntResults[1]);
|
||||
EXPECT_EQ(std::make_pair(2u, 3), IntResults[2]);
|
||||
|
||||
results.clear();
|
||||
// Test an empty range.
|
||||
IntResults.clear();
|
||||
const std::vector<int> baz;
|
||||
for (auto X : llvm::enumerate(baz)) {
|
||||
results.push_back(std::make_pair(X.Index, X.Value));
|
||||
IntResults.emplace_back(X.Index, X.Value);
|
||||
}
|
||||
EXPECT_TRUE(baz.empty());
|
||||
EXPECT_TRUE(IntResults.empty());
|
||||
}
|
||||
|
||||
TEST(STLExtrasTest, EnumerateModify) {
|
||||
TEST(STLExtrasTest, EnumerateModifyLValue) {
|
||||
// Test that you can modify the underlying entries of an lvalue range through
|
||||
// the enumeration iterator.
|
||||
std::vector<char> foo = {'a', 'b', 'c'};
|
||||
|
||||
for (auto X : llvm::enumerate(foo)) {
|
||||
++X.Value;
|
||||
}
|
||||
|
||||
EXPECT_EQ('b', foo[0]);
|
||||
EXPECT_EQ('c', foo[1]);
|
||||
EXPECT_EQ('d', foo[2]);
|
||||
}
|
||||
|
||||
TEST(STLExtrasTest, EnumerateRValueRef) {
|
||||
// Test that an rvalue can be enumerated.
|
||||
std::vector<std::pair<std::size_t, int>> Results;
|
||||
|
||||
auto Enumerator = llvm::enumerate(std::vector<int>{1, 2, 3});
|
||||
|
||||
for (auto X : llvm::enumerate(std::vector<int>{1, 2, 3})) {
|
||||
Results.emplace_back(X.Index, X.Value);
|
||||
}
|
||||
|
||||
ASSERT_EQ(3u, Results.size());
|
||||
EXPECT_EQ(std::make_pair(0u, 1), Results[0]);
|
||||
EXPECT_EQ(std::make_pair(1u, 2), Results[1]);
|
||||
EXPECT_EQ(std::make_pair(2u, 3), Results[2]);
|
||||
}
|
||||
|
||||
TEST(STLExtrasTest, EnumerateModifyRValue) {
|
||||
// Test that when enumerating an rvalue, modification still works (even if
|
||||
// this isn't terribly useful, it at least shows that we haven't snuck an
|
||||
// extra const in there somewhere.
|
||||
std::vector<std::pair<std::size_t, char>> Results;
|
||||
|
||||
for (auto X : llvm::enumerate(std::vector<char>{'1', '2', '3'})) {
|
||||
++X.Value;
|
||||
Results.emplace_back(X.Index, X.Value);
|
||||
}
|
||||
|
||||
ASSERT_EQ(3u, Results.size());
|
||||
EXPECT_EQ(std::make_pair(0u, '2'), Results[0]);
|
||||
EXPECT_EQ(std::make_pair(1u, '3'), Results[1]);
|
||||
EXPECT_EQ(std::make_pair(2u, '4'), Results[2]);
|
||||
}
|
||||
|
||||
template <bool B> struct CanMove {};
|
||||
template <> struct CanMove<false> {
|
||||
CanMove(CanMove &&) = delete;
|
||||
|
||||
CanMove() = default;
|
||||
CanMove(const CanMove &) = default;
|
||||
};
|
||||
|
||||
template <bool B> struct CanCopy {};
|
||||
template <> struct CanCopy<false> {
|
||||
CanCopy(const CanCopy &) = delete;
|
||||
|
||||
CanCopy() = default;
|
||||
CanCopy(CanCopy &&) = default;
|
||||
};
|
||||
|
||||
template <bool Moveable, bool Copyable>
|
||||
struct Range : CanMove<Moveable>, CanCopy<Copyable> {
|
||||
explicit Range(int &C, int &M, int &D) : C(C), M(M), D(D) {}
|
||||
Range(const Range &R) : CanCopy<Copyable>(R), C(R.C), M(R.M), D(R.D) { ++C; }
|
||||
Range(Range &&R) : CanMove<Moveable>(std::move(R)), C(R.C), M(R.M), D(R.D) {
|
||||
++M;
|
||||
}
|
||||
~Range() { ++D; }
|
||||
|
||||
int &C;
|
||||
int &M;
|
||||
int &D;
|
||||
|
||||
int *begin() { return nullptr; }
|
||||
int *end() { return nullptr; }
|
||||
};
|
||||
|
||||
TEST(STLExtrasTest, EnumerateLifetimeSemantics) {
|
||||
// Test that when enumerating lvalues and rvalues, there are no surprise
|
||||
// copies or moves.
|
||||
|
||||
// With an rvalue, it should not be destroyed until the end of the scope.
|
||||
int Copies = 0;
|
||||
int Moves = 0;
|
||||
int Destructors = 0;
|
||||
{
|
||||
auto E1 = enumerate(Range<true, false>(Copies, Moves, Destructors));
|
||||
// Doesn't compile. rvalue ranges must be moveable.
|
||||
// auto E2 = enumerate(Range<false, true>(Copies, Moves, Destructors));
|
||||
EXPECT_EQ(0, Copies);
|
||||
EXPECT_EQ(1, Moves);
|
||||
EXPECT_EQ(1, Destructors);
|
||||
}
|
||||
EXPECT_EQ(0, Copies);
|
||||
EXPECT_EQ(1, Moves);
|
||||
EXPECT_EQ(2, Destructors);
|
||||
|
||||
Copies = Moves = Destructors = 0;
|
||||
// With an lvalue, it should not be destroyed even after the end of the scope.
|
||||
// lvalue ranges need be neither copyable nor moveable.
|
||||
Range<false, false> R(Copies, Moves, Destructors);
|
||||
{
|
||||
auto Enumerator = enumerate(R);
|
||||
(void)Enumerator;
|
||||
EXPECT_EQ(0, Copies);
|
||||
EXPECT_EQ(0, Moves);
|
||||
EXPECT_EQ(0, Destructors);
|
||||
}
|
||||
EXPECT_EQ(0, Copies);
|
||||
EXPECT_EQ(0, Moves);
|
||||
EXPECT_EQ(0, Destructors);
|
||||
}
|
||||
}
|
||||
|
|
Loading…
Reference in New Issue