From: Benjamin Kramer Date: Tue, 4 Aug 2015 15:52:56 +0000 (+0000) Subject: [ArrayRef] Make copy use std::uninitialized_copy. X-Git-Url: http://demsky.eecs.uci.edu/git/?a=commitdiff_plain;h=0d2030d07ecd6fe7662a988e7fafacf43a2e8185;p=oota-llvm.git [ArrayRef] Make copy use std::uninitialized_copy. std::copy does not work for non-trivially copyable classes when we're copying into uninitialized memory. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@243995 91177308-0d34-0410-b5e6-96231b3b80d8 --- diff --git a/include/llvm/ADT/ArrayRef.h b/include/llvm/ADT/ArrayRef.h index f6cc4f4e7c3..e8063305591 100644 --- a/include/llvm/ADT/ArrayRef.h +++ b/include/llvm/ADT/ArrayRef.h @@ -148,7 +148,7 @@ namespace llvm { // copy - Allocate copy in Allocator and return ArrayRef to it. template ArrayRef copy(Allocator &A) { T *Buff = A.template Allocate(Length); - std::copy(begin(), end(), Buff); + std::uninitialized_copy(begin(), end(), Buff); return ArrayRef(Buff, Length); } diff --git a/unittests/ADT/ArrayRefTest.cpp b/unittests/ADT/ArrayRefTest.cpp index 9ad32d54d36..064024140c4 100644 --- a/unittests/ADT/ArrayRefTest.cpp +++ b/unittests/ADT/ArrayRefTest.cpp @@ -31,7 +31,7 @@ static_assert( !std::is_convertible, ArrayRef>::value, "Removing volatile"); -namespace llvm { +namespace { TEST(ArrayRefTest, AllocatorCopy) { BumpPtrAllocator Alloc; @@ -45,6 +45,18 @@ TEST(ArrayRefTest, AllocatorCopy) { EXPECT_NE(Array1.data(), Array1c.data()); EXPECT_TRUE(Array2.equals(Array2c)); EXPECT_NE(Array2.data(), Array2c.data()); + + // Check that copy can cope with uninitialized memory. + struct NonAssignable { + const char *Ptr; + + NonAssignable(const NonAssignable &RHS) = default; + void operator=(const NonAssignable &RHS) { assert(RHS.Ptr != nullptr); } + bool operator==(const NonAssignable &RHS) const { return Ptr == RHS.Ptr; } + } Array3Src[] = {{"hello"}, {"world"}}; + ArrayRef Array3Copy = makeArrayRef(Array3Src).copy(Alloc); + EXPECT_EQ(makeArrayRef(Array3Src), Array3Copy); + EXPECT_NE(makeArrayRef(Array3Src).data(), Array3Copy.data()); } TEST(ArrayRefTest, DropBack) {