Home | History | Annotate | Download | only in any.cons
      1 //===----------------------------------------------------------------------===//
      2 //
      3 //                     The LLVM Compiler Infrastructure
      4 //
      5 // This file is dual licensed under the MIT and the University of Illinois Open
      6 // Source Licenses. See LICENSE.TXT for details.
      7 //
      8 //===----------------------------------------------------------------------===//
      9 
     10 // UNSUPPORTED: c++98, c++03, c++11, c++14
     11 
     12 // <any>
     13 
     14 // any(any &&) noexcept;
     15 
     16 #include <any>
     17 #include <utility>
     18 #include <type_traits>
     19 #include <cassert>
     20 
     21 #include "any_helpers.h"
     22 #include "count_new.hpp"
     23 #include "test_macros.h"
     24 
     25 using std::any;
     26 using std::any_cast;
     27 
     28 // Moves are always noexcept. The throws_on_move object
     29 // must be stored dynamically so the pointer is moved and
     30 // not the stored object.
     31 void test_move_does_not_throw()
     32 {
     33 #if !defined(TEST_HAS_NO_EXCEPTIONS)
     34     assert(throws_on_move::count == 0);
     35     {
     36         throws_on_move v(42);
     37         any a(v);
     38         assert(throws_on_move::count == 2);
     39         // No allocations should be performed after this point.
     40         DisableAllocationGuard g; ((void)g);
     41         try {
     42             any const a2(std::move(a));
     43             assertEmpty(a);
     44             assertContains<throws_on_move>(a2, 42);
     45         } catch (...) {
     46             assert(false);
     47         }
     48         assert(throws_on_move::count == 1);
     49         assertEmpty(a);
     50     }
     51     assert(throws_on_move::count == 0);
     52 #endif
     53 }
     54 
     55 void test_move_empty() {
     56     DisableAllocationGuard g; ((void)g); // no allocations should be performed.
     57 
     58     any a1;
     59     any a2(std::move(a1));
     60 
     61     assertEmpty(a1);
     62     assertEmpty(a2);
     63 }
     64 
     65 template <class Type>
     66 void test_move() {
     67     assert(Type::count == 0);
     68     Type::reset();
     69     {
     70         any a((Type(42)));
     71         assert(Type::count == 1);
     72         assert(Type::copied == 0);
     73         assert(Type::moved == 1);
     74 
     75         // Moving should not perform allocations since it must be noexcept.
     76         DisableAllocationGuard g; ((void)g);
     77 
     78         any a2(std::move(a));
     79 
     80         assert(Type::moved == 1 || Type::moved == 2); // zero or more move operations can be performed.
     81         assert(Type::copied == 0); // no copies can be performed.
     82         assert(Type::count == 1 + a.has_value());
     83         assertContains<Type>(a2, 42);
     84         LIBCPP_ASSERT(!a.has_value()); // Moves are always destructive.
     85         if (a.has_value())
     86             assertContains<Type>(a, 0);
     87     }
     88     assert(Type::count == 0);
     89 }
     90 
     91 int main()
     92 {
     93     // noexcept test
     94     {
     95         static_assert(
     96             std::is_nothrow_move_constructible<any>::value
     97           , "any must be nothrow move constructible"
     98           );
     99     }
    100     test_move<small>();
    101     test_move<large>();
    102     test_move_empty();
    103     test_move_does_not_throw();
    104 }
    105