mirror of git://gcc.gnu.org/git/gcc.git
				
				
				
			re PR libstdc++/88199 (memory leak on unordered container move assignment)
2018-11-27 François Dumont <fdumont@gcc.gnu.org> PR libstdc++/88199 * include/bits/hashtable.h (_Hashtable<>::_M_assign_elements): New. (_Hashtable<>::operator=(const _Hashtable&)): Use latter. (_Hashtable<>::_M_move_assign(_Hashtable&&, false_type)): Likewise. * testsuite/23_containers/unordered_set/allocator/move_assign.cc (test03): New. From-SVN: r266528
This commit is contained in:
		
							parent
							
								
									d15a796967
								
							
						
					
					
						commit
						010211394d
					
				|  | @ -1,3 +1,12 @@ | |||
| 2018-11-27  François Dumont  <fdumont@gcc.gnu.org> | ||||
| 
 | ||||
| 	PR libstdc++/88199 | ||||
| 	* include/bits/hashtable.h (_Hashtable<>::_M_assign_elements): New. | ||||
| 	(_Hashtable<>::operator=(const _Hashtable&)): Use latter. | ||||
| 	(_Hashtable<>::_M_move_assign(_Hashtable&&, false_type)): Likewise. | ||||
| 	* testsuite/23_containers/unordered_set/allocator/move_assign.cc | ||||
| 	(test03): New. | ||||
| 
 | ||||
| 2018-11-26  Jonathan Wakely  <jwakely@redhat.com> | ||||
| 
 | ||||
| 	* testsuite/26_numerics/complex/requirements/more_constexpr.cc: Fix | ||||
|  |  | |||
|  | @ -388,6 +388,12 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION | |||
|       _M_begin() const | ||||
|       { return static_cast<__node_type*>(_M_before_begin._M_nxt); } | ||||
| 
 | ||||
|       // Assign *this using another _Hashtable instance. Either elements
 | ||||
|       // are copy or move depends on the _NodeGenerator.
 | ||||
|       template<typename _Ht, typename _NodeGenerator> | ||||
| 	void | ||||
| 	_M_assign_elements(_Ht&&, const _NodeGenerator&); | ||||
| 
 | ||||
|       template<typename _NodeGenerator> | ||||
| 	void | ||||
| 	_M_assign(const _Hashtable&, const _NodeGenerator&); | ||||
|  | @ -1042,6 +1048,22 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION | |||
| 	} | ||||
| 
 | ||||
|       // Reuse allocated buckets and nodes.
 | ||||
|       _M_assign_elements(__ht, | ||||
| 	[](const __reuse_or_alloc_node_type& __roan, const __node_type* __n) | ||||
| 	{ return __roan(__n->_M_v()); }); | ||||
|       return *this; | ||||
|     } | ||||
| 
 | ||||
|   template<typename _Key, typename _Value, | ||||
| 	   typename _Alloc, typename _ExtractKey, typename _Equal, | ||||
| 	   typename _H1, typename _H2, typename _Hash, typename _RehashPolicy, | ||||
| 	   typename _Traits> | ||||
|     template<typename _Ht, typename _NodeGenerator> | ||||
|       void | ||||
|       _Hashtable<_Key, _Value, _Alloc, _ExtractKey, _Equal, | ||||
| 		 _H1, _H2, _Hash, _RehashPolicy, _Traits>:: | ||||
|       _M_assign_elements(_Ht&& __ht, const _NodeGenerator& __node_gen) | ||||
|       { | ||||
| 	__bucket_type* __former_buckets = nullptr; | ||||
| 	std::size_t __former_bucket_count = _M_bucket_count; | ||||
| 	const __rehash_state& __former_state = _M_rehash_policy._M_state(); | ||||
|  | @ -1058,14 +1080,14 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION | |||
| 
 | ||||
| 	__try | ||||
| 	  { | ||||
| 	  __hashtable_base::operator=(__ht); | ||||
| 	    __hashtable_base::operator=(std::forward<_Ht>(__ht)); | ||||
| 	    _M_element_count = __ht._M_element_count; | ||||
| 	    _M_rehash_policy = __ht._M_rehash_policy; | ||||
| 	    __reuse_or_alloc_node_type __roan(_M_begin(), *this); | ||||
| 	    _M_before_begin._M_nxt = nullptr; | ||||
| 	    _M_assign(__ht, | ||||
| 		    [&__roan](const __node_type* __n) | ||||
| 		    { return __roan(__n->_M_v()); }); | ||||
| 		      [&__node_gen, &__roan](__node_type* __n) | ||||
| 		      { return __node_gen(__roan, __n); }); | ||||
| 	    if (__former_buckets) | ||||
| 	      _M_deallocate_buckets(__former_buckets, __former_bucket_count); | ||||
| 	  } | ||||
|  | @ -1083,7 +1105,6 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION | |||
| 			     _M_bucket_count * sizeof(__bucket_type)); | ||||
| 	    __throw_exception_again; | ||||
| 	  } | ||||
|       return *this; | ||||
|       } | ||||
| 
 | ||||
|   template<typename _Key, typename _Value, | ||||
|  | @ -1198,46 +1219,11 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION | |||
|       else | ||||
| 	{ | ||||
| 	  // Can't move memory, move elements then.
 | ||||
| 	  __bucket_type* __former_buckets = nullptr; | ||||
| 	  size_type __former_bucket_count = _M_bucket_count; | ||||
| 	  const __rehash_state& __former_state = _M_rehash_policy._M_state(); | ||||
| 
 | ||||
| 	  if (_M_bucket_count != __ht._M_bucket_count) | ||||
| 	    { | ||||
| 	      __former_buckets = _M_buckets; | ||||
| 	      _M_buckets = _M_allocate_buckets(__ht._M_bucket_count); | ||||
| 	      _M_bucket_count = __ht._M_bucket_count; | ||||
| 	    } | ||||
| 	  else | ||||
| 	    __builtin_memset(_M_buckets, 0, | ||||
| 			     _M_bucket_count * sizeof(__bucket_type)); | ||||
| 
 | ||||
| 	  __try | ||||
| 	    { | ||||
| 	      __hashtable_base::operator=(std::move(__ht)); | ||||
| 	      _M_element_count = __ht._M_element_count; | ||||
| 	      _M_rehash_policy = __ht._M_rehash_policy; | ||||
| 	      __reuse_or_alloc_node_type __roan(_M_begin(), *this); | ||||
| 	      _M_before_begin._M_nxt = nullptr; | ||||
| 	      _M_assign(__ht, | ||||
| 			[&__roan](__node_type* __n) | ||||
| 	  _M_assign_elements(std::move(__ht), | ||||
| 		[](const __reuse_or_alloc_node_type& __roan, __node_type* __n) | ||||
| 		{ return __roan(std::move_if_noexcept(__n->_M_v())); }); | ||||
| 	  __ht.clear(); | ||||
| 	} | ||||
| 	  __catch(...) | ||||
| 	    { | ||||
| 	      if (__former_buckets) | ||||
| 		{ | ||||
| 		  _M_deallocate_buckets(); | ||||
| 		  _M_rehash_policy._M_reset(__former_state); | ||||
| 		  _M_buckets = __former_buckets; | ||||
| 		  _M_bucket_count = __former_bucket_count; | ||||
| 		} | ||||
| 	      __builtin_memset(_M_buckets, 0, | ||||
| 			       _M_bucket_count * sizeof(__bucket_type)); | ||||
| 	      __throw_exception_again; | ||||
| 	    } | ||||
| 	} | ||||
|     } | ||||
| 
 | ||||
|   template<typename _Key, typename _Value, | ||||
|  |  | |||
|  | @ -18,16 +18,22 @@ | |||
| // { dg-do run { target c++11 } }
 | ||||
| 
 | ||||
| #include <unordered_set> | ||||
| 
 | ||||
| #include <testsuite_hooks.h> | ||||
| #include <testsuite_allocator.h> | ||||
| #include <testsuite_counter_type.h> | ||||
| 
 | ||||
| using __gnu_test::propagating_allocator; | ||||
| using __gnu_test::counter_type; | ||||
| using __gnu_test::tracker_allocator; | ||||
| using __gnu_test::tracker_allocator_counter; | ||||
| 
 | ||||
| void test01() | ||||
| { | ||||
|   typedef propagating_allocator<counter_type, false> alloc_type; | ||||
|   tracker_allocator_counter::reset(); | ||||
|   { | ||||
|     typedef propagating_allocator<counter_type, false, | ||||
| 				  tracker_allocator<counter_type>> alloc_type; | ||||
|     typedef __gnu_test::counter_type_hasher hash; | ||||
|     typedef std::unordered_set<counter_type, hash, | ||||
| 			       std::equal_to<counter_type>, | ||||
|  | @ -50,9 +56,19 @@ void test01() | |||
|     VERIFY( counter_type::destructor_count == 2 ); | ||||
|   } | ||||
| 
 | ||||
|   // Check there's nothing left allocated or constructed.
 | ||||
|   VERIFY( tracker_allocator_counter::get_construct_count() | ||||
| 	  == tracker_allocator_counter::get_destruct_count() ); | ||||
|   VERIFY( tracker_allocator_counter::get_allocation_count() | ||||
| 	  == tracker_allocator_counter::get_deallocation_count() ); | ||||
| } | ||||
| 
 | ||||
| void test02() | ||||
| { | ||||
|   typedef propagating_allocator<counter_type, true> alloc_type; | ||||
|   tracker_allocator_counter::reset(); | ||||
|   { | ||||
|     typedef propagating_allocator<counter_type, true, | ||||
| 				  tracker_allocator<counter_type>> alloc_type; | ||||
|     typedef __gnu_test::counter_type_hasher hash; | ||||
|     typedef std::unordered_set<counter_type, hash, | ||||
| 			       std::equal_to<counter_type>, | ||||
|  | @ -80,9 +96,55 @@ void test02() | |||
|     VERIFY( it == v2.begin() ); | ||||
|   } | ||||
| 
 | ||||
|   // Check there's nothing left allocated or constructed.
 | ||||
|   VERIFY( tracker_allocator_counter::get_construct_count() | ||||
| 	  == tracker_allocator_counter::get_destruct_count() ); | ||||
|   VERIFY( tracker_allocator_counter::get_allocation_count() | ||||
| 	  == tracker_allocator_counter::get_deallocation_count() ); | ||||
| } | ||||
| 
 | ||||
| void test03() | ||||
| { | ||||
|   tracker_allocator_counter::reset(); | ||||
|   { | ||||
|     typedef propagating_allocator<counter_type, false, | ||||
| 				  tracker_allocator<counter_type>> alloc_type; | ||||
|     typedef __gnu_test::counter_type_hasher hash; | ||||
|     typedef std::unordered_set<counter_type, hash, | ||||
| 			       std::equal_to<counter_type>, | ||||
| 			       alloc_type> test_type; | ||||
| 
 | ||||
|     test_type v1(alloc_type(1)); | ||||
|     v1.emplace(0); | ||||
| 
 | ||||
|     test_type v2(alloc_type(2)); | ||||
|     int i = 0; | ||||
|     v2.emplace(i++); | ||||
|     for (; v2.bucket_count() == v1.bucket_count(); ++i) | ||||
|       v2.emplace(i); | ||||
| 
 | ||||
|     counter_type::reset(); | ||||
| 
 | ||||
|     v2 = std::move(v1); | ||||
| 
 | ||||
|     VERIFY( 1 == v1.get_allocator().get_personality() ); | ||||
|     VERIFY( 2 == v2.get_allocator().get_personality() ); | ||||
| 
 | ||||
|     VERIFY( counter_type::move_count == 1  ); | ||||
|     VERIFY( counter_type::destructor_count == i + 1 ); | ||||
|   } | ||||
| 
 | ||||
|   // Check there's nothing left allocated or constructed.
 | ||||
|   VERIFY( tracker_allocator_counter::get_construct_count() | ||||
| 	  == tracker_allocator_counter::get_destruct_count() ); | ||||
|   VERIFY( tracker_allocator_counter::get_allocation_count() | ||||
| 	  == tracker_allocator_counter::get_deallocation_count() ); | ||||
| } | ||||
| 
 | ||||
| int main() | ||||
| { | ||||
|   test01(); | ||||
|   test02(); | ||||
|   test03(); | ||||
|   return 0; | ||||
| } | ||||
|  |  | |||
		Loading…
	
		Reference in New Issue
	
	 François Dumont
						François Dumont