//===----------------------------------------------------------------------===// // // 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 #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::array& input) { auto b = Iter(input.data()); auto e = Sent(Iter(input.data() + input.size())); 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 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