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 void assign_range(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 assignments (assigning an {empty/one-element/mid-sized/long range} to an 2317bbb224Svarconst // {empty/one-element/full} container); 2417bbb224Svarconst // - an exception is thrown when allocating new elements. 2517bbb224Svarconst constexpr bool test() { 2617bbb224Svarconst static_assert(test_constraints_assign_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_assign_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 assignment -- make sure to test both cases. 3617bbb224Svarconst { // Ensure reallocation happens. Note that `vector<bool>` typically reserves a lot of capacity. 3717bbb224Svarconst constexpr int N = 255; 3817bbb224Svarconst bool in[N] = {}; 3917bbb224Svarconst std::vector<bool> v = {0, 0, 0, 1, 1, 0, 0, 0}; 4017bbb224Svarconst assert(v.capacity() < v.size() + std::ranges::size(in)); 4117bbb224Svarconst 4217bbb224Svarconst v.assign_range(in); 4317bbb224Svarconst assert(std::ranges::equal(v, in)); 4417bbb224Svarconst } 4517bbb224Svarconst 4617bbb224Svarconst { // Ensure no reallocation happens. 4717bbb224Svarconst bool in[] = {1, 1, 0, 1, 1}; 4817bbb224Svarconst std::vector<bool> v = {0, 0, 0, 1, 1, 0, 0, 0}; 4917bbb224Svarconst 5017bbb224Svarconst v.assign_range(in); 5117bbb224Svarconst assert(std::ranges::equal(v, in)); 5217bbb224Svarconst } 53*733a98dbSA. Jiang 54*733a98dbSA. Jiang { // Ensure input-only sized ranges are accepted. 55*733a98dbSA. Jiang using input_iter = cpp20_input_iterator<const bool*>; 56*733a98dbSA. Jiang const bool in[]{true, true, false, true}; 57*733a98dbSA. Jiang std::vector<bool> v; 58*733a98dbSA. Jiang v.assign_range(std::views::counted(input_iter{std::ranges::begin(in)}, std::ranges::ssize(in))); 59*733a98dbSA. Jiang assert(std::ranges::equal(v, std::vector<bool>{true, true, false, true})); 60*733a98dbSA. Jiang } 6117bbb224Svarconst } 6217bbb224Svarconst 6317bbb224Svarconst return true; 6417bbb224Svarconst } 6517bbb224Svarconst 66*733a98dbSA. Jiang #ifndef TEST_HAS_NO_LOCALIZATION 67*733a98dbSA. Jiang void test_counted_istream_view() { 68*733a98dbSA. Jiang std::istringstream is{"1 1 0 1"}; 69*733a98dbSA. Jiang auto vals = std::views::istream<bool>(is); 70*733a98dbSA. Jiang std::vector<bool> v; 71*733a98dbSA. Jiang v.assign_range(std::views::counted(vals.begin(), 3)); 72*733a98dbSA. Jiang assert(v == (std::vector{true, true, false})); 73*733a98dbSA. Jiang } 74*733a98dbSA. Jiang #endif 75*733a98dbSA. Jiang 7617bbb224Svarconst int main(int, char**) { 7717bbb224Svarconst test(); 7817bbb224Svarconst static_assert(test()); 7917bbb224Svarconst 8017bbb224Svarconst // Note: `test_assign_range_exception_safety_throwing_copy` doesn't apply because copying booleans cannot throw. 8117bbb224Svarconst test_assign_range_exception_safety_throwing_allocator<std::vector, bool>(); 8217bbb224Svarconst 83*733a98dbSA. Jiang #ifndef TEST_HAS_NO_LOCALIZATION 84*733a98dbSA. Jiang test_counted_istream_view(); 85*733a98dbSA. Jiang #endif 86*733a98dbSA. Jiang 8717bbb224Svarconst return 0; 8817bbb224Svarconst } 89