summaryrefslogtreecommitdiffstats
path: root/libcxx/test/std/containers/container.requirements
diff options
context:
space:
mode:
authorEric Fiselier <eric@efcs.ca>2018-06-05 22:32:52 +0000
committerEric Fiselier <eric@efcs.ca>2018-06-05 22:32:52 +0000
commit8cef7fd75ac3cbeb8efb767e5ba2dcdb76bf4454 (patch)
tree1ba50be1285e73602ab0fd69f0c89eef23062683 /libcxx/test/std/containers/container.requirements
parenta79b6b3ef0d814080680a84202570acf4425ee12 (diff)
downloadbcm5719-llvm-8cef7fd75ac3cbeb8efb767e5ba2dcdb76bf4454.tar.gz
bcm5719-llvm-8cef7fd75ac3cbeb8efb767e5ba2dcdb76bf4454.zip
Fix PR37694 - std::vector doesn't correctly move construct allocators.
C++2a[container.requirements.general]p8 states that when move constructing a container, the allocator is move constructed. Vector previously copy constructed these allocators. This patch fixes that bug. Additionally it cleans up some unnecessary allocator conversions when copy constructing containers. Libc++ uses __internal_allocator_traits::select_on_copy_construction to select the correct allocator during copy construction, but it unnecessarily converted the resulting allocator to the user specified allocator type and back. After this patch list and forward_list no longer do that. Technically we're supposed to be using allocator_traits<allocator_type>::select_on_copy_construction, but that should seemingly be addressed as a separate patch, if at all. llvm-svn: 334053
Diffstat (limited to 'libcxx/test/std/containers/container.requirements')
-rw-r--r--libcxx/test/std/containers/container.requirements/container.requirements.general/allocator_move.pass.cpp97
1 files changed, 97 insertions, 0 deletions
diff --git a/libcxx/test/std/containers/container.requirements/container.requirements.general/allocator_move.pass.cpp b/libcxx/test/std/containers/container.requirements/container.requirements.general/allocator_move.pass.cpp
new file mode 100644
index 00000000000..da9b7e646cc
--- /dev/null
+++ b/libcxx/test/std/containers/container.requirements/container.requirements.general/allocator_move.pass.cpp
@@ -0,0 +1,97 @@
+//===----------------------------------------------------------------------===//
+//
+// The LLVM Compiler Infrastructure
+//
+// This file is dual licensed under the MIT and the University of Illinois Open
+// Source Licenses. See LICENSE.TXT for details.
+//
+//===----------------------------------------------------------------------===//
+
+// UNSUPPORTED: c++98, c++03
+
+// C++2a[container.requirements.general]p8
+// Move constructors obtain an allocator by move construction from the allocator
+// belonging to the container being moved. Such move construction of the
+// allocator shall not exit via an exception.
+
+#include <vector>
+#include <list>
+#include <forward_list>
+#include <set>
+#include <map>
+#include <unordered_map>
+#include <unordered_set>
+
+#include "test_macros.h"
+#include "test_allocator.h"
+
+template <class C>
+void test(int expected_num_allocs = 1) {
+ {
+ test_alloc_base::clear();
+ using AllocT = typename C::allocator_type;
+ C v(AllocT(42, 101));
+
+ assert(test_alloc_base::count == expected_num_allocs);
+
+ const int num_stored_allocs = test_alloc_base::count;
+ {
+ const AllocT& a = v.get_allocator();
+ assert(test_alloc_base::count == 1 + num_stored_allocs);
+ assert(a.get_data() == 42);
+ assert(a.get_id() == 101);
+ }
+ assert(test_alloc_base::count == num_stored_allocs);
+ test_alloc_base::clear_ctor_counters();
+
+ C v2 = std::move(v);
+ assert(test_alloc_base::count == num_stored_allocs * 2);
+ assert(test_alloc_base::copied == 0);
+ assert(test_alloc_base::moved == num_stored_allocs);
+ {
+ const AllocT& a = v.get_allocator();
+ assert(a.get_id() == test_alloc_base::moved_value);
+ assert(a.get_data() == test_alloc_base::moved_value);
+ }
+ {
+ const AllocT& a = v2.get_allocator();
+ assert(a.get_id() == 101);
+ assert(a.get_data() == 42);
+ }
+ }
+}
+
+int main() {
+ { // test sequence containers
+ test<std::vector<int, test_allocator<int> > >();
+ test<std::vector<bool, test_allocator<bool> > >();
+ test<std::list<int, test_allocator<int> > >();
+ test<std::forward_list<int, test_allocator<int> > >();
+ }
+ { // test associative containers
+ test<std::set<int, std::less<int>, test_allocator<int> > >();
+ test<std::multiset<int, std::less<int>, test_allocator<int> > >();
+
+ using KV = std::pair<const int, int>;
+ test<std::map<int, int, std::less<int>, test_allocator<KV> > >();
+ test<std::multimap<int, int, std::less<int>, test_allocator<KV> > >();
+ }
+ { // test unordered containers
+ // libc++ stores two allocators in the unordered containers.
+#ifdef _LIBCPP_VERSION
+ int stored_allocators = 2;
+#else
+ int stored_allocators = 1;
+#endif
+ test<std::unordered_set<int, std::hash<int>, std::equal_to<int>,
+ test_allocator<int> > >(stored_allocators);
+ test<std::unordered_multiset<int, std::hash<int>, std::equal_to<int>,
+ test_allocator<int> > >(stored_allocators);
+
+ using KV = std::pair<const int, int>;
+ test<std::unordered_map<int, int, std::hash<int>, std::equal_to<int>,
+ test_allocator<KV> > >(stored_allocators);
+ test<std::unordered_multimap<int, int, std::hash<int>, std::equal_to<int>,
+ test_allocator<KV> > >(stored_allocators);
+ }
+}
OpenPOWER on IntegriCloud