123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433 |
- //===----------------------------------------------------------------------===//
- //
- // 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
- //
- //===----------------------------------------------------------------------===//
- #ifndef ANY_HELPERS_H
- #define ANY_HELPERS_H
- #include <typeinfo>
- #include <type_traits>
- #include <cassert>
- namespace std { namespace experimental {} }
- #include "test_macros.h"
- #include "type_id.h"
- #if !defined(TEST_HAS_NO_RTTI)
- #define RTTI_ASSERT(X) assert(X)
- #else
- #define RTTI_ASSERT(X)
- #endif
- template <class T>
- struct IsSmallObject
- : public std::integral_constant<bool
- , sizeof(T) <= sizeof(std::any) - sizeof(void*)
- && std::alignment_of<void*>::value
- % std::alignment_of<T>::value == 0
- && std::is_nothrow_move_constructible<T>::value
- >
- {};
- template <class T>
- bool containsType(std::any const& a) {
- #if !defined(TEST_HAS_NO_RTTI)
- return a.type() == typeid(T);
- #else
- return a.has_value() && std::any_cast<T>(&a) != nullptr;
- #endif
- }
- // Return 'true' if 'Type' will be considered a small type by 'any'
- template <class Type>
- bool isSmallType() {
- return IsSmallObject<Type>::value;
- }
- // Assert that an object is empty. If the object used to contain an object
- // of type 'LastType' check that it can no longer be accessed.
- template <class LastType = int>
- void assertEmpty(std::any const& a) {
- using namespace std;
- assert(!a.has_value());
- RTTI_ASSERT(a.type() == typeid(void));
- assert(any_cast<LastType const>(&a) == nullptr);
- }
- template <class Type>
- constexpr auto has_value_member(int) -> decltype(std::declval<Type&>().value, true)
- { return true; }
- template <class> constexpr bool has_value_member(long) { return false; }
- // Assert that an 'any' object stores the specified 'Type' and 'value'.
- template <class Type>
- std::enable_if_t<has_value_member<Type>(0)>
- _LIBCPP_AVAILABILITY_THROW_BAD_ANY_CAST
- assertContains(std::any const& a, int value) {
- assert(a.has_value());
- assert(containsType<Type>(a));
- assert(std::any_cast<Type const &>(a).value == value);
- }
- template <class Type, class Value>
- std::enable_if_t<!has_value_member<Type>(0)>
- _LIBCPP_AVAILABILITY_THROW_BAD_ANY_CAST
- assertContains(std::any const& a, Value value) {
- assert(a.has_value());
- assert(containsType<Type>(a));
- assert(std::any_cast<Type const &>(a) == value);
- }
- // Modify the value of a "test type" stored within an any to the specified
- // 'value'.
- template <class Type>
- _LIBCPP_AVAILABILITY_THROW_BAD_ANY_CAST
- void modifyValue(std::any& a, int value) {
- using namespace std;
- using namespace std::experimental;
- assert(a.has_value());
- assert(containsType<Type>(a));
- any_cast<Type&>(a).value = value;
- }
- // A test type that will trigger the small object optimization within 'any'.
- template <int Dummy = 0>
- struct small_type
- {
- static int count;
- static int copied;
- static int moved;
- static int const_copied;
- static int non_const_copied;
- static void reset() {
- small_type::copied = 0;
- small_type::moved = 0;
- small_type::const_copied = 0;
- small_type::non_const_copied = 0;
- }
- int value;
- explicit small_type(int val = 0) : value(val) {
- ++count;
- }
- explicit small_type(int, int val, int) : value(val) {
- ++count;
- }
- small_type(std::initializer_list<int> il) : value(*il.begin()) {
- ++count;
- }
- small_type(small_type const & other) noexcept {
- value = other.value;
- ++count;
- ++copied;
- ++const_copied;
- }
- small_type(small_type& other) noexcept {
- value = other.value;
- ++count;
- ++copied;
- ++non_const_copied;
- }
- small_type(small_type && other) noexcept {
- value = other.value;
- other.value = 0;
- ++count;
- ++moved;
- }
- ~small_type() {
- value = -1;
- --count;
- }
- private:
- small_type& operator=(small_type const&) = delete;
- small_type& operator=(small_type&&) = delete;
- };
- template <int Dummy>
- int small_type<Dummy>::count = 0;
- template <int Dummy>
- int small_type<Dummy>::copied = 0;
- template <int Dummy>
- int small_type<Dummy>::moved = 0;
- template <int Dummy>
- int small_type<Dummy>::const_copied = 0;
- template <int Dummy>
- int small_type<Dummy>::non_const_copied = 0;
- typedef small_type<> small;
- typedef small_type<1> small1;
- typedef small_type<2> small2;
- // A test type that will NOT trigger the small object optimization in any.
- template <int Dummy = 0>
- struct large_type
- {
- static int count;
- static int copied;
- static int moved;
- static int const_copied;
- static int non_const_copied;
- static void reset() {
- large_type::copied = 0;
- large_type::moved = 0;
- large_type::const_copied = 0;
- large_type::non_const_copied = 0;
- }
- int value;
- large_type(int val = 0) : value(val) {
- ++count;
- data[0] = 0;
- }
- large_type(int, int val, int) : value(val) {
- ++count;
- data[0] = 0;
- }
- large_type(std::initializer_list<int> il) : value(*il.begin()) {
- ++count;
- }
- large_type(large_type const & other) {
- value = other.value;
- ++count;
- ++copied;
- ++const_copied;
- }
- large_type(large_type & other) {
- value = other.value;
- ++count;
- ++copied;
- ++non_const_copied;
- }
- large_type(large_type && other) {
- value = other.value;
- other.value = 0;
- ++count;
- ++moved;
- }
- ~large_type() {
- value = 0;
- --count;
- }
- private:
- large_type& operator=(large_type const&) = delete;
- large_type& operator=(large_type &&) = delete;
- int data[10];
- };
- template <int Dummy>
- int large_type<Dummy>::count = 0;
- template <int Dummy>
- int large_type<Dummy>::copied = 0;
- template <int Dummy>
- int large_type<Dummy>::moved = 0;
- template <int Dummy>
- int large_type<Dummy>::const_copied = 0;
- template <int Dummy>
- int large_type<Dummy>::non_const_copied = 0;
- typedef large_type<> large;
- typedef large_type<1> large1;
- typedef large_type<2> large2;
- // The exception type thrown by 'small_throws_on_copy', 'large_throws_on_copy'
- // and 'throws_on_move'.
- struct my_any_exception {};
- void throwMyAnyExpression() {
- #if !defined(TEST_HAS_NO_EXCEPTIONS)
- throw my_any_exception();
- #else
- assert(false && "Exceptions are disabled");
- #endif
- }
- // A test type that will trigger the small object optimization within 'any'.
- // this type throws if it is copied.
- struct small_throws_on_copy
- {
- static int count;
- static int copied;
- static int moved;
- static void reset() { count = copied = moved = 0; }
- int value;
- explicit small_throws_on_copy(int val = 0) : value(val) {
- ++count;
- }
- explicit small_throws_on_copy(int, int val, int) : value(val) {
- ++count;
- }
- small_throws_on_copy(small_throws_on_copy const &) {
- throwMyAnyExpression();
- }
- small_throws_on_copy(small_throws_on_copy && other) throw() {
- value = other.value;
- ++count; ++moved;
- }
- ~small_throws_on_copy() {
- --count;
- }
- private:
- small_throws_on_copy& operator=(small_throws_on_copy const&) = delete;
- small_throws_on_copy& operator=(small_throws_on_copy &&) = delete;
- };
- int small_throws_on_copy::count = 0;
- int small_throws_on_copy::copied = 0;
- int small_throws_on_copy::moved = 0;
- // A test type that will NOT trigger the small object optimization within 'any'.
- // this type throws if it is copied.
- struct large_throws_on_copy
- {
- static int count;
- static int copied;
- static int moved;
- static void reset() { count = copied = moved = 0; }
- int value = 0;
- explicit large_throws_on_copy(int val = 0) : value(val) {
- data[0] = 0;
- ++count;
- }
- explicit large_throws_on_copy(int, int val, int) : value(val) {
- data[0] = 0;
- ++count;
- }
- large_throws_on_copy(large_throws_on_copy const &) {
- throwMyAnyExpression();
- }
- large_throws_on_copy(large_throws_on_copy && other) throw() {
- value = other.value;
- ++count; ++moved;
- }
- ~large_throws_on_copy() {
- --count;
- }
- private:
- large_throws_on_copy& operator=(large_throws_on_copy const&) = delete;
- large_throws_on_copy& operator=(large_throws_on_copy &&) = delete;
- int data[10];
- };
- int large_throws_on_copy::count = 0;
- int large_throws_on_copy::copied = 0;
- int large_throws_on_copy::moved = 0;
- // A test type that throws when it is moved. This object will NOT trigger
- // the small object optimization in 'any'.
- struct throws_on_move
- {
- static int count;
- static int copied;
- static int moved;
- static void reset() { count = copied = moved = 0; }
- int value;
- explicit throws_on_move(int val = 0) : value(val) { ++count; }
- explicit throws_on_move(int, int val, int) : value(val) { ++count; }
- throws_on_move(throws_on_move const & other) {
- value = other.value;
- ++count; ++copied;
- }
- throws_on_move(throws_on_move &&) {
- throwMyAnyExpression();
- }
- ~throws_on_move() {
- --count;
- }
- private:
- throws_on_move& operator=(throws_on_move const&) = delete;
- throws_on_move& operator=(throws_on_move &&) = delete;
- };
- int throws_on_move::count = 0;
- int throws_on_move::copied = 0;
- int throws_on_move::moved = 0;
- struct small_tracked_t {
- small_tracked_t()
- : arg_types(&makeArgumentID<>()) {}
- small_tracked_t(small_tracked_t const&) noexcept
- : arg_types(&makeArgumentID<small_tracked_t const&>()) {}
- small_tracked_t(small_tracked_t &&) noexcept
- : arg_types(&makeArgumentID<small_tracked_t &&>()) {}
- template <class ...Args>
- explicit small_tracked_t(Args&&...)
- : arg_types(&makeArgumentID<Args...>()) {}
- template <class ...Args>
- explicit small_tracked_t(std::initializer_list<int>, Args&&...)
- : arg_types(&makeArgumentID<std::initializer_list<int>, Args...>()) {}
- TypeID const* arg_types;
- };
- static_assert(IsSmallObject<small_tracked_t>::value, "must be small");
- struct large_tracked_t {
- large_tracked_t()
- : arg_types(&makeArgumentID<>()) { dummy[0] = 42; }
- large_tracked_t(large_tracked_t const&) noexcept
- : arg_types(&makeArgumentID<large_tracked_t const&>()) {}
- large_tracked_t(large_tracked_t &&) noexcept
- : arg_types(&makeArgumentID<large_tracked_t &&>()) {}
- template <class ...Args>
- explicit large_tracked_t(Args&&...)
- : arg_types(&makeArgumentID<Args...>()) {}
- template <class ...Args>
- explicit large_tracked_t(std::initializer_list<int>, Args&&...)
- : arg_types(&makeArgumentID<std::initializer_list<int>, Args...>()) {}
- TypeID const* arg_types;
- int dummy[sizeof(std::any) / sizeof(int) + 1];
- };
- static_assert(!IsSmallObject<large_tracked_t>::value, "must not be small");
- template <class Type, class ...Args>
- void assertArgsMatch(std::any const& a) {
- using namespace std;
- using namespace std::experimental;
- assert(a.has_value());
- assert(containsType<Type>(a));
- assert(any_cast<Type const &>(a).arg_types == &makeArgumentID<Args...>());
- };
- #endif
|