//===----------------------------------------------------------------------===// // // 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 // //===----------------------------------------------------------------------===// // UNSUPPORTED: c++03, c++11, c++14 // UNSUPPORTED: availability-filesystem-missing // These tests require locale for non-char paths // UNSUPPORTED: no-localization // // class path // template // path& operator=(Source const&); // path& operator=(string_type&&); // template // path& assign(Source const&); // template // path& assign(InputIterator first, InputIterator last); #include #include #include #include // On Windows, charset conversions cause allocations in the path class in // cases where no allocations are done on other platforms. #include "../../path_helper.h" #include "count_new.h" #include "make_string.h" #include "test_iterators.h" #include "test_macros.h" namespace fs = std::filesystem; template void RunTestCase(MultiStringType const& MS) { using namespace fs; const fs::path::value_type* Expect = MS; const CharT* TestPath = MS; const CharT* TestPathEnd = StrEnd(TestPath); const std::size_t Size = TestPathEnd - TestPath; const std::size_t SSize = StrEnd(Expect) - Expect; assert(Size == SSize); ////////////////////////////////////////////////////////////////////////////// // basic_string { const std::basic_string S(TestPath); path p; PathReserve(p, S.length() + 1); { // string provides a contiguous iterator. No allocation needed. TEST_NOT_WIN32(DisableAllocationGuard g); path& pref = (p = S); assert(&pref == &p); } assert(p.native() == Expect); assert(p.string() == TestPath); assert(p.string() == S); } { const std::basic_string S(TestPath); path p; PathReserve(p, S.length() + 1); { TEST_NOT_WIN32(DisableAllocationGuard g); path& pref = p.assign(S); assert(&pref == &p); } assert(p.native() == Expect); assert(p.string() == TestPath); assert(p.string() == S); } // basic_string { const std::basic_string_view S(TestPath); path p; PathReserve(p, S.length() + 1); { // string provides a contiguous iterator. No allocation needed. TEST_NOT_WIN32(DisableAllocationGuard g); path& pref = (p = S); assert(&pref == &p); } assert(p.native() == Expect); assert(p.string() == TestPath); assert(p.string() == S); } { const std::basic_string_view S(TestPath); path p; PathReserve(p, S.length() + 1); { TEST_NOT_WIN32(DisableAllocationGuard g); path& pref = p.assign(S); assert(&pref == &p); } assert(p.native() == Expect); assert(p.string() == TestPath); assert(p.string() == S); } ////////////////////////////////////////////////////////////////////////////// // Char* pointers { path p; PathReserve(p, Size + 1); { // char* pointers are contiguous and can be used with code_cvt directly. // no allocations needed. TEST_NOT_WIN32(DisableAllocationGuard g); path& pref = (p = TestPath); assert(&pref == &p); } assert(p.native() == Expect); assert(p.string() == TestPath); } { path p; PathReserve(p, Size + 1); { TEST_NOT_WIN32(DisableAllocationGuard g); path& pref = p.assign(TestPath); assert(&pref == &p); } assert(p.native() == Expect); assert(p.string() == TestPath); } { path p; PathReserve(p, Size + 1); { TEST_NOT_WIN32(DisableAllocationGuard g); path& pref = p.assign(TestPath, TestPathEnd); assert(&pref == &p); } assert(p.native() == Expect); assert(p.string() == TestPath); } ////////////////////////////////////////////////////////////////////////////// // Iterators { using It = cpp17_input_iterator; path p; PathReserve(p, Size + 1); It it(TestPath); { // Iterators cannot be used with code_cvt directly. This assignment // may allocate if it's larger than a "short-string". path& pref = (p = it); assert(&pref == &p); } assert(p.native() == Expect); assert(p.string() == TestPath); } { using It = cpp17_input_iterator; path p; PathReserve(p, Size + 1); It it(TestPath); { path& pref = p.assign(it); assert(&pref == &p); } assert(p.native() == Expect); assert(p.string() == TestPath); } { using It = cpp17_input_iterator; path p; PathReserve(p, Size + 1); It it(TestPath); It e(TestPathEnd); { path& pref = p.assign(it, e); assert(&pref == &p); } assert(p.native() == Expect); assert(p.string() == TestPath); } } template ()))> constexpr bool has_assign(int) { return true; } template constexpr bool has_assign(long) { return false; } template constexpr bool has_assign() { return has_assign(0); } void test_sfinae() { using namespace fs; { using It = const char* const; static_assert(std::is_assignable::value, ""); static_assert(has_assign(), ""); } { using It = cpp17_input_iterator; static_assert(std::is_assignable::value, ""); static_assert(has_assign(), ""); } { struct Traits { using iterator_category = std::input_iterator_tag; using value_type = const char; using pointer = const char*; using reference = const char&; using difference_type = std::ptrdiff_t; }; using It = cpp17_input_iterator; static_assert(std::is_assignable::value, ""); static_assert(has_assign(), ""); } { using It = cpp17_output_iterator; static_assert(!std::is_assignable::value, ""); static_assert(!has_assign(), ""); } { static_assert(!std::is_assignable::value, ""); static_assert(!has_assign(), ""); } } void RunStringMoveTest(const fs::path::value_type* Expect) { using namespace fs; fs::path::string_type ss(Expect); path p; { DisableAllocationGuard g; ((void)g); path& pr = (p = std::move(ss)); assert(&pr == &p); } assert(p == Expect); { // Signature test LIBCPP_ASSERT_NOEXCEPT(p = std::move(ss)); } } int main(int, char**) { for (auto const& MS : PathList) { RunTestCase(MS); #ifndef TEST_HAS_NO_WIDE_CHARACTERS RunTestCase(MS); #endif RunTestCase(MS); RunTestCase(MS); RunStringMoveTest(MS); } test_sfinae(); return 0; }