123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115 |
- //===----------------------------------------------------------------------===//
- //
- // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
- // See https://llvm.org/LICENSE.txt for license information.
- // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
- //
- //===----------------------------------------------------------------------===//
- // UNSUPPORTED: c++98, c++03, c++11, c++14
- // <memory>
- // template <class InputIt, class ForwardIt>
- // ForwardIt uninitialized_move(InputIt, InputIt, ForwardIt);
- #include <memory>
- #include <cstdlib>
- #include <cassert>
- #include "test_macros.h"
- #include "test_iterators.h"
- struct Counted {
- static int count;
- static int constructed;
- static void reset() { count = constructed = 0; }
- explicit Counted(int&& x) : value(x) { x = 0; ++count; ++constructed; }
- Counted(Counted const&) { assert(false); }
- ~Counted() { assert(count > 0); --count; }
- friend void operator&(Counted) = delete;
- int value;
- };
- int Counted::count = 0;
- int Counted::constructed = 0;
- struct ThrowsCounted {
- static int count;
- static int constructed;
- static int throw_after;
- static void reset() { throw_after = count = constructed = 0; }
- explicit ThrowsCounted(int&& x) {
- ++constructed;
- if (throw_after > 0 && --throw_after == 0) {
- TEST_THROW(1);
- }
- ++count;
- x = 0;
- }
- ThrowsCounted(ThrowsCounted const&) { assert(false); }
- ~ThrowsCounted() { assert(count > 0); --count; }
- friend void operator&(ThrowsCounted) = delete;
- };
- int ThrowsCounted::count = 0;
- int ThrowsCounted::constructed = 0;
- int ThrowsCounted::throw_after = 0;
- void test_ctor_throws()
- {
- #ifndef TEST_HAS_NO_EXCEPTIONS
- using It = forward_iterator<ThrowsCounted*>;
- const int N = 5;
- int values[N] = {1, 2, 3, 4, 5};
- alignas(ThrowsCounted) char pool[sizeof(ThrowsCounted)*N] = {};
- ThrowsCounted* p = (ThrowsCounted*)pool;
- try {
- ThrowsCounted::throw_after = 4;
- std::uninitialized_move(values, values + N, It(p));
- assert(false);
- } catch (...) {}
- assert(ThrowsCounted::count == 0);
- assert(ThrowsCounted::constructed == 4); // forth construction throws
- assert(values[0] == 0);
- assert(values[1] == 0);
- assert(values[2] == 0);
- assert(values[3] == 4);
- assert(values[4] == 5);
- #endif
- }
- void test_counted()
- {
- using It = input_iterator<int*>;
- using FIt = forward_iterator<Counted*>;
- const int N = 5;
- int values[N] = {1, 2, 3, 4, 5};
- alignas(Counted) char pool[sizeof(Counted)*N] = {};
- Counted* p = (Counted*)pool;
- auto ret = std::uninitialized_move(It(values), It(values + 1), FIt(p));
- assert(ret == FIt(p +1));
- assert(Counted::constructed == 1);
- assert(Counted::count == 1);
- assert(p[0].value == 1);
- assert(values[0] == 0);
- ret = std::uninitialized_move(It(values+1), It(values+N), FIt(p+1));
- assert(ret == FIt(p + N));
- assert(Counted::count == 5);
- assert(Counted::constructed == 5);
- assert(p[1].value == 2);
- assert(p[2].value == 3);
- assert(p[3].value == 4);
- assert(p[4].value == 5);
- assert(values[1] == 0);
- assert(values[2] == 0);
- assert(values[3] == 0);
- assert(values[4] == 0);
- std::destroy(p, p+N);
- assert(Counted::count == 0);
- }
- int main(int, char**) {
- test_counted();
- test_ctor_throws();
- return 0;
- }
|