__glibcxx_function_requires(_InputIteratorConcept<_InputIterator>)
__glibcxx_function_requires(_OutputIteratorConcept<_OutputIterator,
typename iterator_traits<_InputIterator>::value_type>)
- __glibcxx_requires_can_increment(__first, __n);
- __glibcxx_requires_can_increment(__result, __n);
- return std::__copy_n(__first, __n, __result,
+ const auto __n2 = std::__size_to_integer(__n);
+ if (__n2 <= 0)
+ return __result;
+
+ __glibcxx_requires_can_increment(__first, __n2);
+ __glibcxx_requires_can_increment(__result, __n2);
+
+ return std::__copy_n(__first, __n2, __result,
std::__iterator_category(__first));
}
--- /dev/null
+// Copyright (C) 2020 Free Software Foundation, Inc.
+//
+// This file is part of the GNU ISO C++ Library. This library is free
+// software; you can redistribute it and/or modify it under the
+// terms of the GNU General Public License as published by the
+// Free Software Foundation; either version 3, or (at your option)
+// any later version.
+
+// This library is distributed in the hope that it will be useful,
+// but WITHOUT ANY WARRANTY; without even the implied warranty of
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+// GNU General Public License for more details.
+
+// You should have received a copy of the GNU General Public License along
+// with this library; see the file COPYING3. If not see
+// <http://www.gnu.org/licenses/>.
+
+// { dg-do run { target c++11 } }
+
+#include <algorithm>
+#include <testsuite_iterators.h>
+
+void
+test01()
+{
+ // Negative sizes should be a no-op
+
+ const int from[2] = { 1, 2 };
+ __gnu_test::input_container<const int> f(from);
+ int* to = nullptr;
+ std::copy_n(f.begin(), -1, to);
+
+ std::copy_n(from, -20000, to); // random access
+
+ __gnu_test::random_access_container<const int> f2(from);
+ std::copy_n(f2.end(), -1, to);
+ std::copy_n(f2.begin(), -1, to);
+}
+
+struct Size
+{
+ operator long() const { return 2L; }
+
+ void operator++() = delete;
+ void operator--() = delete;
+ void operator++(int) = delete;
+ void operator--(int) = delete;
+
+ template<typename T> friend void operator+(Size, T) = delete;
+ template<typename T> friend void operator+(T, Size) = delete;
+ template<typename T> friend void operator-(Size, T) = delete;
+ template<typename T> friend void operator-(T, Size) = delete;
+ template<typename T> friend void operator==(Size, T) = delete;
+ template<typename T> friend void operator==(T, Size) = delete;
+ template<typename T> friend void operator!=(Size, T) = delete;
+ template<typename T> friend void operator!=(T, Size) = delete;
+ template<typename T> friend void operator<(Size, T) = delete;
+ template<typename T> friend void operator<(T, Size) = delete;
+ template<typename T> friend void operator<=(Size, T) = delete;
+ template<typename T> friend void operator<=(T, Size) = delete;
+ template<typename T> friend void operator>(Size, T) = delete;
+ template<typename T> friend void operator>(T, Size) = delete;
+ template<typename T> friend void operator>=(Size, T) = delete;
+ template<typename T> friend void operator>=(T, Size) = delete;
+};
+
+void
+test02()
+{
+ // C++20 only requires that Size is convertible to an integral type,
+ // it doesn't need to support any arithmetic or relational expressions.
+
+ const int from[3] = { 1, 2, 3 };
+ __gnu_test::input_container<const int> f(from);
+ int to[3] = { };
+ __gnu_test::output_container<int> t(to);
+ Size s;
+ std::copy_n(f.begin(), s, t.begin());
+ VERIFY( to[0] == 1 );
+ VERIFY( to[1] == 2 );
+ VERIFY( to[2] == 0 );
+
+ const int from2[3] = { 11, 22, 33 };
+ __gnu_test::random_access_container<const int> f2(from2);
+ __gnu_test::output_container<int> t2(to);
+ std::copy_n(f2.begin(), s, t2.begin());
+ VERIFY( to[0] == 11 );
+ VERIFY( to[1] == 22 );
+ VERIFY( to[2] == 0 );
+}
+
+int
+main()
+{
+ test01();
+ test02();
+}
}
else
{
- ITERATOR_VERIFY(n <= this->ptr - this->SharedInfo->first);
+ ITERATOR_VERIFY(-n <= this->ptr - this->SharedInfo->first);
this->ptr += n;
}
return *this;
{ return bounds.size(); }
};
+#if __cplusplus >= 201103L
+ template<typename T>
+ using output_container
+ = test_container<T, output_iterator_wrapper>;
+
+ template<typename T>
+ using input_container
+ = test_container<T, input_iterator_wrapper>;
+
+ template<typename T>
+ using forward_container
+ = test_container<T, forward_iterator_wrapper>;
+
+ template<typename T>
+ using bidirectional_container
+ = test_container<T, bidirectional_iterator_wrapper>;
+
+ template<typename T>
+ using random_access_container
+ = test_container<T, random_access_iterator_wrapper>;
+#endif
+
#if __cplusplus > 201703L
template<typename T>
struct contiguous_iterator_wrapper
{ return iter -= n; }
};
+ template<typename T>
+ using contiguous_container
+ = test_container<T, contiguous_iterator_wrapper>;
+
// A move-only input iterator type.
template<typename T>
struct input_iterator_wrapper_nocopy : input_iterator_wrapper<T>