//===----------------------------------------------------------------------===// // // 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 SUPPORT_FROM_RANGE_HELPERS_H #define SUPPORT_FROM_RANGE_HELPERS_H #include #include #include #include "min_allocator.h" #include "test_allocator.h" #include "test_iterators.h" #include "test_macros.h" #include "type_algorithms.h" struct Empty {}; template struct InputRange { cpp20_input_iterator begin(); sentinel_wrapper> end(); }; template constexpr auto wrap_input(Range&& input) { auto b = Iter(std::ranges::begin(input)); auto e = Sent(Iter(std::ranges::end(input))); return std::ranges::subrange(std::move(b), std::move(e)); } template constexpr auto wrap_input(std::vector& input) { auto b = Iter(input.data()); auto e = Sent(Iter(input.data() + input.size())); return std::ranges::subrange(std::move(b), std::move(e)); } struct KeyValue { int key; // Only the key is considered for equality comparison. char value; // Allows distinguishing equivalent instances. bool operator<(const KeyValue& other) const { return key < other.key; } bool operator==(const KeyValue& other) const { return key == other.key; } }; template <> struct std::hash { std::size_t operator()(const KeyValue& kv) const { return kv.key; } }; #if !defined(TEST_HAS_NO_EXCEPTIONS) template struct ThrowingCopy { static bool throwing_enabled; static int created_by_copying; static int destroyed; int x = 0; // Allows distinguishing between different instances. ThrowingCopy() = default; ThrowingCopy(int value) : x(value) {} ~ThrowingCopy() { ++destroyed; } ThrowingCopy(const ThrowingCopy& other) : x(other.x) { ++created_by_copying; if (throwing_enabled && created_by_copying == N) { throw -1; } } // Defined to silence GCC warnings. For test purposes, only copy construction is considered `created_by_copying`. ThrowingCopy& operator=(const ThrowingCopy& other) { x = other.x; return *this; } friend auto operator<=>(const ThrowingCopy&, const ThrowingCopy&) = default; static void reset() { created_by_copying = destroyed = 0; } }; template struct std::hash> { std::size_t operator()(const ThrowingCopy& value) const { return value.x; } }; template bool ThrowingCopy::throwing_enabled = true; template int ThrowingCopy::created_by_copying = 0; template int ThrowingCopy::destroyed = 0; template struct ThrowingAllocator { using value_type = T; using char_type = T; using is_always_equal = std::false_type; ThrowingAllocator() = default; template ThrowingAllocator(const ThrowingAllocator&) {} T* allocate(std::size_t) { throw 1; } void deallocate(T*, std::size_t) {} template friend bool operator==(const ThrowingAllocator&, const ThrowingAllocator&) { return true; } }; #endif template constexpr void for_all_iterators_and_allocators(Func f) { using Iterators = types::type_list< cpp20_input_iterator, forward_iterator, bidirectional_iterator, random_access_iterator, contiguous_iterator, T* >; types::for_each(Iterators{}, [=]() { f.template operator(), std::allocator>(); f.template operator(), test_allocator>(); f.template operator(), min_allocator>(); f.template operator(), safe_allocator>(); if constexpr (std::sentinel_for) { f.template operator()>(); f.template operator()>(); f.template operator()>(); f.template operator()>(); } }); } #endif // SUPPORT_FROM_RANGE_HELPERS_H