diff --git a/libcxx/include/filesystem b/libcxx/include/filesystem --- a/libcxx/include/filesystem +++ b/libcxx/include/filesystem @@ -791,7 +791,7 @@ template static typename enable_if<__is_cpp17_forward_iterator<_Iter>::value>::type __append_range(__path_string& __dest, _Iter __b, _Iter __e) { - __dest.__append_forward_unsafe(__b, __e); + __dest.append(__b, __e); } template diff --git a/libcxx/include/string b/libcxx/include/string --- a/libcxx/include/string +++ b/libcxx/include/string @@ -625,30 +625,6 @@ _LIBCPP_EXTERN_TEMPLATE(class _LIBCPP_EXTERN_TEMPLATE_TYPE_VIS __basic_string_common) -#ifdef _LIBCPP_NO_EXCEPTIONS -template -struct __libcpp_string_gets_noexcept_iterator_impl : public true_type {}; -#elif defined(_LIBCPP_HAS_NO_NOEXCEPT) -template -struct __libcpp_string_gets_noexcept_iterator_impl : public false_type {}; -#else -template ::value> -struct __libcpp_string_gets_noexcept_iterator_impl : public _LIBCPP_BOOL_CONSTANT(( - noexcept(++(declval<_Iter&>())) && - is_nothrow_assignable<_Iter&, _Iter>::value && - noexcept(declval<_Iter>() == declval<_Iter>()) && - noexcept(*declval<_Iter>()) -)) {}; - -template -struct __libcpp_string_gets_noexcept_iterator_impl<_Iter, false> : public false_type {}; -#endif - - -template -struct __libcpp_string_gets_noexcept_iterator - : public _LIBCPP_BOOL_CONSTANT(__libcpp_is_trivial_iterator<_Iter>::value || __libcpp_string_gets_noexcept_iterator_impl<_Iter>::value) {}; - template struct __can_be_converted_to_string_view : public _BoolConstant< is_convertible >::value && @@ -1041,15 +1017,11 @@ _LIBCPP_INLINE_VISIBILITY void __append_default_init(size_type __n); - template - _LIBCPP_METHOD_TEMPLATE_IMPLICIT_INSTANTIATION_VIS - basic_string& __append_forward_unsafe(_ForwardIterator, _ForwardIterator); template _LIBCPP_METHOD_TEMPLATE_IMPLICIT_INSTANTIATION_VIS _EnableIf < - __is_exactly_cpp17_input_iterator<_InputIterator>::value - || !__libcpp_string_gets_noexcept_iterator<_InputIterator>::value, + __is_exactly_cpp17_input_iterator<_InputIterator>::value, basic_string& > _LIBCPP_INLINE_VISIBILITY @@ -1062,14 +1034,11 @@ _LIBCPP_METHOD_TEMPLATE_IMPLICIT_INSTANTIATION_VIS _EnableIf < - __is_cpp17_forward_iterator<_ForwardIterator>::value - && __libcpp_string_gets_noexcept_iterator<_ForwardIterator>::value, + __is_cpp17_forward_iterator<_ForwardIterator>::value, basic_string& > _LIBCPP_INLINE_VISIBILITY - append(_ForwardIterator __first, _ForwardIterator __last) { - return __append_forward_unsafe(__first, __last); - } + append(_ForwardIterator __first, _ForwardIterator __last); #ifndef _LIBCPP_CXX03_LANG _LIBCPP_INLINE_VISIBILITY @@ -1117,8 +1086,8 @@ _LIBCPP_METHOD_TEMPLATE_IMPLICIT_INSTANTIATION_VIS _EnableIf < - __is_exactly_cpp17_input_iterator<_InputIterator>::value - || !__libcpp_string_gets_noexcept_iterator<_InputIterator>::value, + __is_exactly_cpp17_input_iterator<_InputIterator>::value + || !__libcpp_is_trivial_iterator<_InputIterator>::value, basic_string& > assign(_InputIterator __first, _InputIterator __last); @@ -1127,7 +1096,7 @@ _EnableIf < __is_cpp17_forward_iterator<_ForwardIterator>::value - && __libcpp_string_gets_noexcept_iterator<_ForwardIterator>::value, + && __libcpp_is_trivial_iterator<_ForwardIterator>::value, basic_string& > assign(_ForwardIterator __first, _ForwardIterator __last); @@ -1168,8 +1137,8 @@ _LIBCPP_METHOD_TEMPLATE_IMPLICIT_INSTANTIATION_VIS _EnableIf < - __is_exactly_cpp17_input_iterator<_InputIterator>::value - || !__libcpp_string_gets_noexcept_iterator<_InputIterator>::value, + __is_exactly_cpp17_input_iterator<_InputIterator>::value + || !__libcpp_is_trivial_iterator<_InputIterator>::value, iterator > insert(const_iterator __pos, _InputIterator __first, _InputIterator __last); @@ -1178,7 +1147,7 @@ _EnableIf < __is_cpp17_forward_iterator<_ForwardIterator>::value - && __libcpp_string_gets_noexcept_iterator<_ForwardIterator>::value, + && __libcpp_is_trivial_iterator<_ForwardIterator>::value, iterator > insert(const_iterator __pos, _ForwardIterator __first, _ForwardIterator __last); @@ -1714,6 +1683,22 @@ _LIBCPP_INLINE_VISIBILITY void __invalidate_all_iterators(); _LIBCPP_INLINE_VISIBILITY void __invalidate_iterators_past(size_type); + struct __null_terminator_guard { + explicit __null_terminator_guard(value_type *__p) : + __p_(__p) {} + + void __release_guard() { + __p_ = nullptr; + } + + ~__null_terminator_guard() { + if (__p_) + *__p_ = value_type(); + } + private: + value_type *__p_; + }; + friend basic_string operator+<>(const basic_string&, const basic_string&); friend basic_string operator+<>(const value_type*, const basic_string&); friend basic_string operator+<>(value_type, const basic_string&); @@ -2463,8 +2448,8 @@ template _EnableIf < - __is_exactly_cpp17_input_iterator <_InputIterator>::value - || !__libcpp_string_gets_noexcept_iterator<_InputIterator>::value, + __is_exactly_cpp17_input_iterator<_InputIterator>::value + || !__libcpp_is_trivial_iterator<_InputIterator>::value, basic_string<_CharT, _Traits, _Allocator>& > basic_string<_CharT, _Traits, _Allocator>::assign(_InputIterator __first, _InputIterator __last) @@ -2479,7 +2464,7 @@ _EnableIf < __is_cpp17_forward_iterator<_ForwardIterator>::value - && __libcpp_string_gets_noexcept_iterator<_ForwardIterator>::value, + && __libcpp_is_trivial_iterator<_ForwardIterator>::value, basic_string<_CharT, _Traits, _Allocator>& > basic_string<_CharT, _Traits, _Allocator>::assign(_ForwardIterator __first, _ForwardIterator __last) @@ -2658,12 +2643,14 @@ template template -basic_string<_CharT, _Traits, _Allocator>& -basic_string<_CharT, _Traits, _Allocator>::__append_forward_unsafe( +_EnableIf +< + __is_cpp17_forward_iterator<_ForwardIterator>::value, + basic_string<_CharT, _Traits, _Allocator>& +> +basic_string<_CharT, _Traits, _Allocator>::append( _ForwardIterator __first, _ForwardIterator __last) { - static_assert(__is_cpp17_forward_iterator<_ForwardIterator>::value, - "function requires a ForwardIterator"); size_type __sz = size(); size_type __cap = capacity(); size_type __n = static_cast(_VSTD::distance(__first, __last)); @@ -2681,8 +2668,10 @@ if (__cap - __sz < __n) __grow_by(__cap, __sz + __n - __cap, __sz, __sz, 0); pointer __p = __get_pointer() + __sz; + __null_terminator_guard __ntr(_VSTD::addressof(*__p)); for (; __first != __last; ++__p, ++__first) traits_type::assign(*__p, *__first); + __ntr.__release_guard(); traits_type::assign(*__p, value_type()); __set_size(__sz + __n); } @@ -2801,9 +2790,9 @@ template _EnableIf < - __is_exactly_cpp17_input_iterator<_InputIterator>::value - || !__libcpp_string_gets_noexcept_iterator<_InputIterator>::value, - typename basic_string<_CharT, _Traits, _Allocator>::iterator + __is_exactly_cpp17_input_iterator<_InputIterator>::value + || !__libcpp_is_trivial_iterator<_InputIterator>::value, + typename basic_string<_CharT, _Traits, _Allocator>::iterator > basic_string<_CharT, _Traits, _Allocator>::insert(const_iterator __pos, _InputIterator __first, _InputIterator __last) { @@ -2821,7 +2810,7 @@ _EnableIf < __is_cpp17_forward_iterator<_ForwardIterator>::value - && __libcpp_string_gets_noexcept_iterator<_ForwardIterator>::value, + && __libcpp_is_trivial_iterator<_ForwardIterator>::value, typename basic_string<_CharT, _Traits, _Allocator>::iterator > basic_string<_CharT, _Traits, _Allocator>::insert(const_iterator __pos, _ForwardIterator __first, _ForwardIterator __last) @@ -2832,6 +2821,12 @@ " referring to this string"); #endif size_type __ip = static_cast(__pos - begin()); + if (__ip == size()) + { + append(__first, __last); + return begin() + __ip; + } + size_type __n = static_cast(_VSTD::distance(__first, __last)); if (__n) { diff --git a/libcxx/test/libcxx/strings/iterators.exceptions.pass.cpp b/libcxx/test/libcxx/strings/iterators.exceptions.pass.cpp deleted file mode 100644 --- a/libcxx/test/libcxx/strings/iterators.exceptions.pass.cpp +++ /dev/null @@ -1,89 +0,0 @@ -//===----------------------------------------------------------------------===// -// -// 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 -// -//===----------------------------------------------------------------------===// -// -// - -// __libcpp_is_trivial_iterator - -// __libcpp_string_gets_noexcept_iterator determines if an iterator can be used -// w/o worrying about whether or not certain operations can throw. -// This gives us a "fast path for string operations" -// - -#include -#include -#include -#include -#include - -#include "test_macros.h" -#include "test_iterators.h" - -#ifndef TEST_HAS_NO_EXCEPTIONS -static const bool expected = false; -#else -// Under -fno-exceptions all noexcept expressions are trivially true, so -// any check for a noexcept returning false must actually check for it being -// true. -static const bool expected = true; -#endif - -int main(int, char**) -{ -// basic tests - static_assert(( std::__libcpp_string_gets_noexcept_iterator::value), ""); - static_assert(( std::__libcpp_string_gets_noexcept_iterator::value), ""); - - static_assert(( std::__libcpp_string_gets_noexcept_iterator > ::value), ""); - static_assert(( std::__libcpp_string_gets_noexcept_iterator >::value), ""); - static_assert(( std::__libcpp_string_gets_noexcept_iterator > ::value), ""); - static_assert(( std::__libcpp_string_gets_noexcept_iterator >::value), ""); - - static_assert(( std::__libcpp_string_gets_noexcept_iterator > ::value), ""); - static_assert(( std::__libcpp_string_gets_noexcept_iterator >::value), ""); - - static_assert(( std::__libcpp_string_gets_noexcept_iterator > > ::value), ""); - -// iterators in the libc++ test suite - static_assert(std::__libcpp_string_gets_noexcept_iterator >::value == expected, ""); - static_assert(std::__libcpp_string_gets_noexcept_iterator >::value == expected, ""); - static_assert(std::__libcpp_string_gets_noexcept_iterator >::value == expected, ""); - static_assert(std::__libcpp_string_gets_noexcept_iterator >::value == expected, ""); - static_assert(std::__libcpp_string_gets_noexcept_iterator >::value == expected, ""); - static_assert(std::__libcpp_string_gets_noexcept_iterator >::value == expected, ""); - -#if TEST_STD_VER >= 11 - static_assert(( std::__libcpp_string_gets_noexcept_iterator >::value), ""); -#else - static_assert(std::__libcpp_string_gets_noexcept_iterator >::value == expected, ""); -#endif - -// -// iterators from libc++'s containers -// - -// string - static_assert(( std::__libcpp_string_gets_noexcept_iterator::iterator> ::value), ""); - static_assert(( std::__libcpp_string_gets_noexcept_iterator::const_iterator> ::value), ""); - static_assert(( std::__libcpp_string_gets_noexcept_iterator::reverse_iterator> ::value), ""); - static_assert(( std::__libcpp_string_gets_noexcept_iterator::const_reverse_iterator>::value), ""); - -// vector - static_assert(( std::__libcpp_string_gets_noexcept_iterator::iterator> ::value), ""); - static_assert(( std::__libcpp_string_gets_noexcept_iterator::const_iterator> ::value), ""); - static_assert(( std::__libcpp_string_gets_noexcept_iterator::reverse_iterator> ::value), ""); - static_assert(( std::__libcpp_string_gets_noexcept_iterator::const_reverse_iterator>::value), ""); - -#if TEST_STD_VER >= 11 -// Initializer list (which has no reverse iterators) - static_assert(( std::__libcpp_string_gets_noexcept_iterator::iterator> ::value), ""); - static_assert(( std::__libcpp_string_gets_noexcept_iterator::const_iterator> ::value), ""); -#endif - - return 0; -} diff --git a/libcxx/test/libcxx/strings/iterators.noexcept.pass.cpp b/libcxx/test/libcxx/strings/iterators.noexcept.pass.cpp deleted file mode 100644 --- a/libcxx/test/libcxx/strings/iterators.noexcept.pass.cpp +++ /dev/null @@ -1,81 +0,0 @@ -//===----------------------------------------------------------------------===// -// -// 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 -// -//===----------------------------------------------------------------------===// -// - -// - -// __libcpp_is_trivial_iterator - -// __libcpp_string_gets_noexcept_iterator determines if an iterator can be used -// w/o worrying about whether or not certain operations can throw. -// This gives us a "fast path for string operations". -// -// When exceptions are disabled, all iterators should get this "fast path" -// - -// ADDITIONAL_COMPILE_FLAGS: -fno-exceptions - -#include -#include -#include -#include -#include - -#include "test_macros.h" -#include "test_iterators.h" - -int main(int, char**) -{ -// basic tests - static_assert(( std::__libcpp_string_gets_noexcept_iterator::value), ""); - static_assert(( std::__libcpp_string_gets_noexcept_iterator::value), ""); - - static_assert(( std::__libcpp_string_gets_noexcept_iterator > ::value), ""); - static_assert(( std::__libcpp_string_gets_noexcept_iterator >::value), ""); - static_assert(( std::__libcpp_string_gets_noexcept_iterator > ::value), ""); - static_assert(( std::__libcpp_string_gets_noexcept_iterator >::value), ""); - - static_assert(( std::__libcpp_string_gets_noexcept_iterator > ::value), ""); - static_assert(( std::__libcpp_string_gets_noexcept_iterator >::value), ""); - - static_assert(( std::__libcpp_string_gets_noexcept_iterator > > ::value), ""); - -// iterators in the libc++ test suite - static_assert(( std::__libcpp_string_gets_noexcept_iterator >::value), ""); - static_assert(( std::__libcpp_string_gets_noexcept_iterator >::value), ""); - static_assert(( std::__libcpp_string_gets_noexcept_iterator >::value), ""); - static_assert(( std::__libcpp_string_gets_noexcept_iterator >::value), ""); - static_assert(( std::__libcpp_string_gets_noexcept_iterator >::value), ""); - static_assert(( std::__libcpp_string_gets_noexcept_iterator >::value), ""); - - static_assert(( std::__libcpp_string_gets_noexcept_iterator >::value), ""); - -// -// iterators from libc++'s containers -// - -// string - static_assert(( std::__libcpp_string_gets_noexcept_iterator::iterator> ::value), ""); - static_assert(( std::__libcpp_string_gets_noexcept_iterator::const_iterator> ::value), ""); - static_assert(( std::__libcpp_string_gets_noexcept_iterator::reverse_iterator> ::value), ""); - static_assert(( std::__libcpp_string_gets_noexcept_iterator::const_reverse_iterator>::value), ""); - -// vector - static_assert(( std::__libcpp_string_gets_noexcept_iterator::iterator> ::value), ""); - static_assert(( std::__libcpp_string_gets_noexcept_iterator::const_iterator> ::value), ""); - static_assert(( std::__libcpp_string_gets_noexcept_iterator::reverse_iterator> ::value), ""); - static_assert(( std::__libcpp_string_gets_noexcept_iterator::const_reverse_iterator>::value), ""); - -#if TEST_STD_VER >= 11 -// Initializer list (which has no reverse iterators) - static_assert(( std::__libcpp_string_gets_noexcept_iterator::iterator> ::value), ""); - static_assert(( std::__libcpp_string_gets_noexcept_iterator::const_iterator> ::value), ""); -#endif - - return 0; -} diff --git a/libcxx/test/std/strings/basic.string/string.modifiers/string_insert/strong_guarantee.pass.cpp b/libcxx/test/std/strings/basic.string/string.modifiers/string_insert/strong_guarantee.pass.cpp new file mode 100644 --- /dev/null +++ b/libcxx/test/std/strings/basic.string/string.modifiers/string_insert/strong_guarantee.pass.cpp @@ -0,0 +1,86 @@ +//===----------------------------------------------------------------------===// +// +// 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 + +// + +// template +// iterator insert(const_iterator pos, InputIterator first, InputIterator last); + +#include +#include + +#include "test_macros.h" + +int count; + +struct Iterator { + using value_type = char; + using difference_type = int; + using iterator_category = std::forward_iterator_tag; + using reference = const char&; + using pointer = const char*; + + const char *ptr_ = nullptr; + + explicit Iterator() = default; + explicit Iterator(const char *p) : ptr_(p) {} + Iterator operator++(int) { Iterator copy(*this); ++*this; return copy; } + Iterator& operator++() noexcept { ++ptr_; return *this; } + const char& operator*() const noexcept { return *ptr_; } + friend bool operator==(Iterator a, Iterator b) noexcept { return a.ptr_ == b.ptr_; } + friend bool operator!=(Iterator a, Iterator b) noexcept(false) { + if (--count == 0) throw "oops"; + return a.ptr_ != b.ptr_; + } +}; + +static_assert(std::__libcpp_string_gets_noexcept_iterator::value, ""); + +void test_one_count(const std::string& original, const std::string& needle, int pos) +{ + std::string expected = original; + expected.insert(expected.begin() + pos, needle.begin(), needle.end()); + std::string s = original; + try { + Iterator first(needle.data()); + Iterator last(needle.data() + needle.size()); + s.insert(s.begin() + pos, first, last); + LIBCPP_ASSERT(s.__invariants()); + assert(s == expected); + } catch (...) { + LIBCPP_ASSERT(s.__invariants()); + assert(s == original); + } +} + +void test_all_counts(const std::string& original, const std::string& needle, int pos) +{ + for (int i=1; true; ++i) { + count = i; + test_one_count(original, needle, pos); + if (count != 0) break; // in this case no exception was thrown + } +} + +int main(int, char**) +{ + test_all_counts("", "xy", 0); + test_all_counts("aaa", "xy", 0); + test_all_counts("aaa", "xy", 1); + test_all_counts("aaa", "xy", 2); + test_all_counts("aaa", "xy", 3); + test_all_counts("a very long string that defies short string optimization", "short", 0); + test_all_counts("a very long string that defies short string optimization", "short", 2); + test_all_counts("a very long string that defies short string optimization", "short", 25); + test_all_counts("a very long string that defies short string optimization", "short", 53); + test_all_counts("a very long string that defies short string optimization", "short", 56); + + return 0; +}