When I commit the small series of patches on _Hashtable I realize that I miss a part on the one regarding reservation management on range insertion. I've added a comment saying that we consider that all initializer_list elements will be inserted.

For the moment it is true only for assignement operator, this patch makes it true for constructor and insert methods. In the insert method we "reserve" only if container is empty cause otherwise we can't be that sure that all elements will be inserted even if the users made its best for that.

The consequence is that silly initialization in the tests are not working anymore.

    libstdc++: Hashtable: Consider that all initializer_list elements are inserted

    libstdc++-v3/ChangeLog:

            * include/bits/hashtable_policy.h (_Insert_base<>::_M_initialize): New.
(_Insert_base<>::insert(initializer_list<value_type>)): Adapt, use
            latter.
            * include/bits/hashtable.h
            (_Hashtable(initializer_list<value_type>, size_type, const _H1&,
            const key_equal&, const allocator_type&)): Likewise.
(_Hashtable<>::operator=(initializer_list<value_type>)): Likewise.
            * testsuite/23_containers/unordered_set/cons/bucket_hint.cc (test02):
            New.
            * testsuite/23_containers/unordered_set/init-list.cc (test01): Remove.             * testsuite/23_containers/unordered_set/modifiers/insert.cc (test01):
            Remove.

Tested under Linux x86_64.

Ok to commit ?

François

diff --git a/libstdc++-v3/include/bits/hashtable.h b/libstdc++-v3/include/bits/hashtable.h
index 07a4abe5c33..d530005f05f 100644
--- a/libstdc++-v3/include/bits/hashtable.h
+++ b/libstdc++-v3/include/bits/hashtable.h
@@ -529,9 +529,11 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION
 		 const _Hash& __hf = _Hash(),
 		 const key_equal& __eql = key_equal(),
 		 const allocator_type& __a = allocator_type())
-      : _Hashtable(__l.begin(), __l.end(), __bkt_count_hint,
-		   __hf, __eql, __a, __unique_keys{})
-      { }
+      : _Hashtable(__bkt_count_hint, __hf, __eql, __a)
+      {
+	__alloc_node_gen_t __alloc_node_gen(*this);
+	this->_M_initialize(__l, __alloc_node_gen);
+      }
 
       _Hashtable&
       operator=(const _Hashtable& __ht);
@@ -556,14 +558,7 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION
 	_M_before_begin._M_nxt = nullptr;
 	clear();
 
-	// We consider that all elements of __l are going to be inserted.
-	auto __l_bkt_count = _M_rehash_policy._M_bkt_for_elements(__l.size());
-
-	// Do not shrink to keep potential user reservation.
-	if (_M_bucket_count < __l_bkt_count)
-	  rehash(__l_bkt_count);
-
-	this->_M_insert_range(__l.begin(), __l.end(), __roan, __unique_keys{});
+	this->_M_initialize(__l, __roan);
 	return *this;
       }
 
diff --git a/libstdc++-v3/include/bits/hashtable_policy.h b/libstdc++-v3/include/bits/hashtable_policy.h
index 0109ef86a7b..6ad09db10a0 100644
--- a/libstdc++-v3/include/bits/hashtable_policy.h
+++ b/libstdc++-v3/include/bits/hashtable_policy.h
@@ -825,6 +825,25 @@ namespace __detail
 	_M_insert_range(_InputIterator __first, _InputIterator __last,
 			const _NodeGetter&, false_type __uks);
 
+      template<typename _NodeGetter>
+	void
+	_M_initialize(initializer_list<value_type> __l,
+		      const _NodeGetter& __node_gen)
+	{
+	  __hashtable& __h = _M_conjure_hashtable();
+
+	  // We consider that all elements of __l are going to be inserted.
+	  auto __l_bkt_count
+	    = __h._M_rehash_policy._M_bkt_for_elements(__l.size());
+
+	  // Do not shrink to keep potential user reservation.
+	  if (__h._M_bucket_count < __l_bkt_count)
+	    __h.rehash(__l_bkt_count);
+
+	  this->_M_insert_range(__l.begin(), __l.end(), __node_gen,
+				__unique_keys{});
+	}
+
     public:
       __ireturn_type
       insert(const value_type& __v)
@@ -866,7 +885,16 @@ namespace __detail
 
       void
       insert(initializer_list<value_type> __l)
-      { this->insert(__l.begin(), __l.end()); }
+      {
+	__hashtable& __h = _M_conjure_hashtable();
+	if (__h.empty())
+	  {
+	    __node_gen_type __node_gen(__h);
+	    _M_initialize(__l, __node_gen);
+	  }
+	else
+	  this->insert(__l.begin(), __l.end());
+      }
 
       template<typename _InputIterator>
 	void
diff --git a/libstdc++-v3/testsuite/23_containers/unordered_set/cons/bucket_hint.cc b/libstdc++-v3/testsuite/23_containers/unordered_set/cons/bucket_hint.cc
index 100eb5a27df..424e8f68c28 100644
--- a/libstdc++-v3/testsuite/23_containers/unordered_set/cons/bucket_hint.cc
+++ b/libstdc++-v3/testsuite/23_containers/unordered_set/cons/bucket_hint.cc
@@ -23,15 +23,6 @@
 
 #include <testsuite_hooks.h>
 
-void test01()
-{
-  std::unordered_set<int> a;
-  a.reserve(2);
-
-  std::unordered_set<int> b({ 0, 1, 0, 1, 0, 1, 0, 1 }, a.bucket_count());
-  VERIFY( b.bucket_count() == a.bucket_count() );
-}
-
 void test02()
 {
   std::unordered_set<int> a;
@@ -56,7 +47,6 @@ void test03()
 
 int main()
 {
-  test01();
   test02();
   test03();
   return 0;
diff --git a/libstdc++-v3/testsuite/23_containers/unordered_set/init-list.cc b/libstdc++-v3/testsuite/23_containers/unordered_set/init-list.cc
index 66efa72f262..e86f90b1361 100644
--- a/libstdc++-v3/testsuite/23_containers/unordered_set/init-list.cc
+++ b/libstdc++-v3/testsuite/23_containers/unordered_set/init-list.cc
@@ -48,8 +48,27 @@ void test01()
   VERIFY(m.count(1) == 0);
 }
 
+void test02()
+{
+  unordered_set<int> u({ 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12 });
+  VERIFY( u.size() == 13 );
+  VERIFY( u.count(0) == 1 );
+  VERIFY( u.count(13) == 0 );
+
+  auto bkt_count = u.bucket_count();
+  u.insert({ 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12 });
+  VERIFY( u.size() == 13 );
+  VERIFY( u.bucket_count() == bkt_count );
+
+  u.clear();
+  u.insert({ 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12 });
+  VERIFY( u.size() == 13 );
+  VERIFY( u.bucket_count() == bkt_count );
+}
+
 int main()
 {
   __gnu_test::set_memory_limits();
   test01();
+  test02();
 }
diff --git a/libstdc++-v3/testsuite/23_containers/unordered_set/modifiers/insert.cc b/libstdc++-v3/testsuite/23_containers/unordered_set/modifiers/insert.cc
index acf01c73b1b..7e4d3c06cb9 100644
--- a/libstdc++-v3/testsuite/23_containers/unordered_set/modifiers/insert.cc
+++ b/libstdc++-v3/testsuite/23_containers/unordered_set/modifiers/insert.cc
@@ -23,16 +23,6 @@
 
 #include <testsuite_hooks.h>
 
-void test01()
-{
-  std::unordered_set<int> a;
-  a.reserve(2);
-
-  auto bkt_count = a.bucket_count();
-  a.insert({ 0, 1, 0, 1, 0, 1, 0, 1 });
-  VERIFY( a.bucket_count() == bkt_count );
-}
-
 void test02()
 {
   std::unordered_set<int> a;
@@ -59,7 +49,6 @@ void test03()
 
 int main()
 {
-  test01();
   test02();
   test03();
   return 0;

Reply via email to