117bbb224Svarconst //===----------------------------------------------------------------------===// 217bbb224Svarconst // 317bbb224Svarconst // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 417bbb224Svarconst // See https://llvm.org/LICENSE.txt for license information. 517bbb224Svarconst // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 617bbb224Svarconst // 717bbb224Svarconst //===----------------------------------------------------------------------===// 817bbb224Svarconst 917bbb224Svarconst // UNSUPPORTED: c++03, c++11, c++14, c++17, c++20 1017bbb224Svarconst // ADDITIONAL_COMPILE_FLAGS(has-fconstexpr-steps): -fconstexpr-steps=2000000 1117bbb224Svarconst 1217bbb224Svarconst // template<container-compatible-range<bool> R> 1317bbb224Svarconst // constexpr iterator insert_range(const_iterator position, R&& rg); // C++23 1417bbb224Svarconst 15*733a98dbSA. Jiang #include <sstream> 1617bbb224Svarconst #include <vector> 1717bbb224Svarconst 1817bbb224Svarconst #include "../insert_range_sequence_containers.h" 1917bbb224Svarconst #include "test_macros.h" 2017bbb224Svarconst 2117bbb224Svarconst // Tested cases: 2217bbb224Svarconst // - different kinds of insertions (inserting an {empty/one-element/mid-sized/long range} into an 2317bbb224Svarconst // {empty/one-element/full} container at the {beginning/middle/end}); 2417bbb224Svarconst // - an exception is thrown when allocating new elements. 2517bbb224Svarconst constexpr bool test() { 2617bbb224Svarconst static_assert(test_constraints_insert_range<std::vector, bool, char>()); 2717bbb224Svarconst 2817bbb224Svarconst for_all_iterators_and_allocators<bool, const int*>([]<class Iter, class Sent, class Alloc>() { 29346a2990SStephan T. Lavavej test_sequence_insert_range<std::vector<bool, Alloc>, Iter, Sent>([]([[maybe_unused]] auto&& c) { 3017bbb224Svarconst LIBCPP_ASSERT(c.__invariants()); 3117bbb224Svarconst // `is_contiguous_container_asan_correct` doesn't work on `vector<bool>`. 3217bbb224Svarconst }); 3317bbb224Svarconst }); 3417bbb224Svarconst 3517bbb224Svarconst { // Vector may or may not need to reallocate because of the insertion -- make sure to test both cases. 3617bbb224Svarconst { // Ensure reallocation happens. 3717bbb224Svarconst constexpr int N = 255; 3817bbb224Svarconst bool in[N] = {}; 3917bbb224Svarconst std::vector<bool> v = {0, 0, 0, 1, 1, 0, 0, 0}; 4017bbb224Svarconst auto initial = v; 4117bbb224Svarconst assert(v.capacity() < v.size() + std::ranges::size(in)); 4217bbb224Svarconst 4317bbb224Svarconst v.insert_range(v.end(), in); 4417bbb224Svarconst // Because `in` is very large (it has to be to exceed the large capacity that `vector<bool>` allocates), it is 4517bbb224Svarconst // impractical to have the expected value as a literal. 4617bbb224Svarconst assert(v.size() == initial.size() + N); 4717bbb224Svarconst assert(std::ranges::equal(v.begin(), v.begin() + initial.size(), initial.begin(), initial.end())); 4817bbb224Svarconst assert(std::ranges::equal(v.begin() + initial.size(), v.end(), std::ranges::begin(in), std::ranges::end(in))); 4917bbb224Svarconst } 5017bbb224Svarconst 5117bbb224Svarconst { // Ensure no reallocation happens. 5217bbb224Svarconst bool in[] = {1, 1, 1, 1, 0, 0, 1, 1, 1, 1}; 5317bbb224Svarconst std::vector<bool> v = {0, 0, 0, 1, 1, 0, 0, 0}; 5417bbb224Svarconst v.reserve(v.size() + std::ranges::size(in)); 5517bbb224Svarconst assert(v.capacity() >= v.size() + std::ranges::size(in)); 5617bbb224Svarconst 5717bbb224Svarconst v.insert_range(v.end(), in); 5817bbb224Svarconst assert(std::ranges::equal(v, std::vector<bool>{0, 0, 0, 1, 1, 0, 0, 0, 1, 1, 1, 1, 0, 0, 1, 1, 1, 1})); 5917bbb224Svarconst } 60*733a98dbSA. Jiang 61*733a98dbSA. Jiang { // Ensure input-only sized ranges are accepted. 62*733a98dbSA. Jiang using input_iter = cpp20_input_iterator<const bool*>; 63*733a98dbSA. Jiang const bool in[]{true, true, false, true}; 64*733a98dbSA. Jiang std::vector<bool> v{true, false}; 65*733a98dbSA. Jiang v.insert_range(v.begin(), std::views::counted(input_iter{std::ranges::begin(in)}, std::ranges::ssize(in))); 66*733a98dbSA. Jiang assert(std::ranges::equal(v, std::vector<bool>{true, true, false, true, true, false})); 67*733a98dbSA. Jiang } 6817bbb224Svarconst } 6917bbb224Svarconst 7017bbb224Svarconst return true; 7117bbb224Svarconst } 7217bbb224Svarconst 73*733a98dbSA. Jiang #ifndef TEST_HAS_NO_LOCALIZATION 74*733a98dbSA. Jiang void test_counted_istream_view() { 75*733a98dbSA. Jiang std::istringstream is{"1 1 0 1"}; 76*733a98dbSA. Jiang auto vals = std::views::istream<bool>(is); 77*733a98dbSA. Jiang std::vector<bool> v; 78*733a98dbSA. Jiang v.insert_range(v.end(), std::views::counted(vals.begin(), 3)); 79*733a98dbSA. Jiang assert(v == (std::vector{true, true, false})); 80*733a98dbSA. Jiang } 81*733a98dbSA. Jiang #endif 82*733a98dbSA. Jiang 8317bbb224Svarconst int main(int, char**) { 8417bbb224Svarconst test(); 8517bbb224Svarconst static_assert(test()); 8617bbb224Svarconst 8717bbb224Svarconst // Note: `test_insert_range_exception_safety_throwing_copy` doesn't apply because copying booleans cannot throw. 8817bbb224Svarconst test_insert_range_exception_safety_throwing_allocator<std::vector, bool>(); 8917bbb224Svarconst 90*733a98dbSA. Jiang #ifndef TEST_HAS_NO_LOCALIZATION 91*733a98dbSA. Jiang test_counted_istream_view(); 92*733a98dbSA. Jiang #endif 93*733a98dbSA. Jiang 9417bbb224Svarconst return 0; 9517bbb224Svarconst } 96