Skip to content

Commit

Permalink
Add an ArrayRef upcasting constructor from ArrayRef<U*> -> ArrayRef<T…
Browse files Browse the repository at this point in the history
…*> where T is a base of U.

git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@225053 91177308-0d34-0410-b5e6-96231b3b80d8
  • Loading branch information
gottesmm committed Dec 31, 2014
1 parent 28650b8 commit 735f1df
Show file tree
Hide file tree
Showing 2 changed files with 45 additions and 0 deletions.
10 changes: 10 additions & 0 deletions include/llvm/ADT/ArrayRef.h
Original file line number Diff line number Diff line change
Expand Up @@ -112,6 +112,16 @@ namespace llvm {
std::is_convertible<U *const *, T const *>::value>::type* = 0)
: Data(A.data()), Length(A.size()) {}

/// Construct an ArrayRef<T*> from an ArrayRef<U*> where T is a super class
/// of U. This uses SFINAE to ensure that only ArrayRefs with this property
/// can be converted. This is an upcasting constructor.
template <typename U>
ArrayRef(const ArrayRef<U> &A,
typename std::enable_if<std::is_base_of<
typename std::remove_pointer<T>::type,
typename std::remove_pointer<U>::type>::value>::type * = 0)
: Data(reinterpret_cast<T const *>(A.data())), Length(A.size()) {}

/// @}
/// @name Simple Operations
/// @{
Expand Down
35 changes: 35 additions & 0 deletions unittests/ADT/ArrayRefTest.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -90,4 +90,39 @@ TEST(ArrayRefTest, ConstConvert) {
a = ArrayRef<int *>(A);
}

struct A {
int data;

A() : data(0) {}
};

struct B : A {
int data2;

B() : A(), data2(0) {}
};

TEST(ArrayRefTest, UpcastConvert) {
B Data[5];

for (unsigned i = 0, e = 5; i != e; ++i) {
Data[i].data = i + 5;
Data[i].data2 = i + 30;
}

B *DataPtrs[5];
for (unsigned i = 0, e = 5; i != e; ++i) {
DataPtrs[i] = &Data[i];
}

ArrayRef<B *> BArray(DataPtrs, 5);
ArrayRef<A *> AArray(BArray);

EXPECT_TRUE(AArray.size() == 5);
for (unsigned i = 0, e = 5; i != e; ++i) {
A *a = AArray[i];
EXPECT_TRUE(a->data == int(i + 5));
}
}

} // end anonymous namespace

0 comments on commit 735f1df

Please sign in to comment.