mirror of
				https://github.com/llvm-mirror/libcxx.git
				synced 2025-10-23 01:18:52 +08:00 
			
		
		
		
	Fix bug 20740 - std::set/std::map don't support heterogeneous lookup for count(). Thanks to Jim Porter for the bug report
git-svn-id: https://llvm.org/svn/llvm-project/libcxx/trunk@216353 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
		
							
								
								
									
										16
									
								
								include/map
									
									
									
									
									
								
							
							
						
						
									
										16
									
								
								include/map
									
									
									
									
									
								
							| @@ -159,7 +159,7 @@ public: | ||||
|     template<typename K> | ||||
|         const_iterator find(const K& x) const;  // C++14 | ||||
|     template<typename K> | ||||
|       size_type count(const K& x) const; | ||||
|       size_type count(const K& x) const;        // C++14 | ||||
|  | ||||
|     size_type      count(const key_type& k) const; | ||||
|           iterator lower_bound(const key_type& k); | ||||
| @@ -353,7 +353,7 @@ public: | ||||
|     template<typename K> | ||||
|         const_iterator find(const K& x) const;  // C++14 | ||||
|     template<typename K> | ||||
|       size_type count(const K& x) const; | ||||
|       size_type count(const K& x) const;        // C++14 | ||||
|  | ||||
|     size_type      count(const key_type& k) const; | ||||
|           iterator lower_bound(const key_type& k); | ||||
| @@ -1104,6 +1104,12 @@ public: | ||||
|     _LIBCPP_INLINE_VISIBILITY | ||||
|     size_type      count(const key_type& __k) const | ||||
|         {return __tree_.__count_unique(__k);} | ||||
| #if _LIBCPP_STD_VER > 11 | ||||
|     template <typename _K2> | ||||
|     _LIBCPP_INLINE_VISIBILITY | ||||
|     typename enable_if<__is_transparent<_Compare, _K2>::value,size_type>::type | ||||
|     count(const _K2& __k)                  {return __tree_.__count_unique(__k);} | ||||
| #endif | ||||
|     _LIBCPP_INLINE_VISIBILITY | ||||
|     iterator lower_bound(const key_type& __k) | ||||
|         {return __tree_.lower_bound(__k);} | ||||
| @@ -1834,6 +1840,12 @@ public: | ||||
|     _LIBCPP_INLINE_VISIBILITY | ||||
|     size_type      count(const key_type& __k) const | ||||
|         {return __tree_.__count_multi(__k);} | ||||
| #if _LIBCPP_STD_VER > 11 | ||||
|     template <typename _K2> | ||||
|     _LIBCPP_INLINE_VISIBILITY | ||||
|     typename enable_if<__is_transparent<_Compare, _K2>::value,size_type>::type | ||||
|     count(const _K2& __k)                   {return __tree_.__count_multi(__k);} | ||||
| #endif | ||||
|     _LIBCPP_INLINE_VISIBILITY | ||||
|     iterator lower_bound(const key_type& __k) | ||||
|         {return __tree_.lower_bound(__k);} | ||||
|   | ||||
							
								
								
									
										12
									
								
								include/set
									
									
									
									
									
								
							
							
						
						
									
										12
									
								
								include/set
									
									
									
									
									
								
							| @@ -663,6 +663,12 @@ public: | ||||
|     _LIBCPP_INLINE_VISIBILITY | ||||
|     size_type      count(const key_type& __k) const | ||||
|         {return __tree_.__count_unique(__k);} | ||||
| #if _LIBCPP_STD_VER > 11 | ||||
|     template <typename _K2> | ||||
|     _LIBCPP_INLINE_VISIBILITY | ||||
|     typename enable_if<__is_transparent<_Compare, _K2>::value,size_type>::type | ||||
|     count(const _K2& __k)                  {return __tree_.__count_unique(__k);} | ||||
| #endif | ||||
|     _LIBCPP_INLINE_VISIBILITY | ||||
|     iterator lower_bound(const key_type& __k) | ||||
|         {return __tree_.lower_bound(__k);} | ||||
| @@ -1066,6 +1072,12 @@ public: | ||||
|     _LIBCPP_INLINE_VISIBILITY | ||||
|     size_type      count(const key_type& __k) const | ||||
|         {return __tree_.__count_multi(__k);} | ||||
| #if _LIBCPP_STD_VER > 11 | ||||
|     template <typename _K2> | ||||
|     _LIBCPP_INLINE_VISIBILITY | ||||
|     typename enable_if<__is_transparent<_Compare, _K2>::value,size_type>::type | ||||
|     count(const _K2& __k)                  {return __tree_.__count_multi(__k);} | ||||
| #endif | ||||
|  | ||||
|     _LIBCPP_INLINE_VISIBILITY | ||||
|     iterator lower_bound(const key_type& __k) | ||||
|   | ||||
| @@ -17,6 +17,7 @@ | ||||
| #include <cassert> | ||||
|  | ||||
| #include "min_allocator.h" | ||||
| #include "private_constructor.hpp" | ||||
|  | ||||
| int main() | ||||
| { | ||||
| @@ -96,4 +97,77 @@ int main() | ||||
|     } | ||||
|     } | ||||
| #endif | ||||
| #if _LIBCPP_STD_VER > 11 | ||||
|     { | ||||
|     typedef std::pair<const int, double> V; | ||||
|     typedef std::map<int, double, std::less <>> M; | ||||
|     typedef M::size_type R; | ||||
|  | ||||
|     V ar[] = | ||||
|     { | ||||
|         V(5, 5), | ||||
|         V(6, 6), | ||||
|         V(7, 7), | ||||
|         V(8, 8), | ||||
|         V(9, 9), | ||||
|         V(10, 10), | ||||
|         V(11, 11), | ||||
|         V(12, 12) | ||||
|     }; | ||||
|     const M m(ar, ar+sizeof(ar)/sizeof(ar[0])); | ||||
|     R r = m.count(5); | ||||
|     assert(r == 1); | ||||
|     r = m.count(6); | ||||
|     assert(r == 1); | ||||
|     r = m.count(7); | ||||
|     assert(r == 1); | ||||
|     r = m.count(8); | ||||
|     assert(r == 1); | ||||
|     r = m.count(9); | ||||
|     assert(r == 1); | ||||
|     r = m.count(10); | ||||
|     assert(r == 1); | ||||
|     r = m.count(11); | ||||
|     assert(r == 1); | ||||
|     r = m.count(12); | ||||
|     assert(r == 1); | ||||
|     r = m.count(4); | ||||
|     assert(r == 0); | ||||
|     } | ||||
|  | ||||
|     { | ||||
|     typedef PrivateConstructor PC; | ||||
|     typedef std::map<PC, double, std::less<>> M; | ||||
|     typedef M::size_type R; | ||||
|  | ||||
|     M m; | ||||
|     m [ PC::make(5)  ] = 5; | ||||
|     m [ PC::make(6)  ] = 6; | ||||
|     m [ PC::make(7)  ] = 7; | ||||
|     m [ PC::make(8)  ] = 8; | ||||
|     m [ PC::make(9)  ] = 9; | ||||
|     m [ PC::make(10) ] = 10; | ||||
|     m [ PC::make(11) ] = 11; | ||||
|     m [ PC::make(12) ] = 12; | ||||
|  | ||||
|     R r = m.count(5); | ||||
|     assert(r == 1); | ||||
|     r = m.count(6); | ||||
|     assert(r == 1); | ||||
|     r = m.count(7); | ||||
|     assert(r == 1); | ||||
|     r = m.count(8); | ||||
|     assert(r == 1); | ||||
|     r = m.count(9); | ||||
|     assert(r == 1); | ||||
|     r = m.count(10); | ||||
|     assert(r == 1); | ||||
|     r = m.count(11); | ||||
|     assert(r == 1); | ||||
|     r = m.count(12); | ||||
|     assert(r == 1); | ||||
|     r = m.count(4); | ||||
|     assert(r == 0); | ||||
|     } | ||||
| #endif | ||||
| } | ||||
|   | ||||
| @@ -17,6 +17,7 @@ | ||||
| #include <cassert> | ||||
|  | ||||
| #include "min_allocator.h" | ||||
| #include "private_constructor.hpp" | ||||
|  | ||||
| int main() | ||||
| { | ||||
| @@ -89,4 +90,70 @@ int main() | ||||
|     } | ||||
|     } | ||||
| #endif | ||||
|  | ||||
| #if _LIBCPP_STD_VER > 11 | ||||
|     { | ||||
|     typedef std::multimap<int, double, std::less<>> M; | ||||
|     typedef M::size_type R; | ||||
|     V ar[] = | ||||
|     { | ||||
|         V(5, 1), | ||||
|         V(5, 2), | ||||
|         V(5, 3), | ||||
|         V(7, 1), | ||||
|         V(7, 2), | ||||
|         V(7, 3), | ||||
|         V(9, 1), | ||||
|         V(9, 2), | ||||
|         V(9, 3) | ||||
|     }; | ||||
|     const M m(ar, ar+sizeof(ar)/sizeof(ar[0])); | ||||
|     R r = m.count(4); | ||||
|     assert(r == 0); | ||||
|     r = m.count(5); | ||||
|     assert(r == 3); | ||||
|     r = m.count(6); | ||||
|     assert(r == 0); | ||||
|     r = m.count(7); | ||||
|     assert(r == 3); | ||||
|     r = m.count(8); | ||||
|     assert(r == 0); | ||||
|     r = m.count(9); | ||||
|     assert(r == 3); | ||||
|     r = m.count(10); | ||||
|     assert(r == 0); | ||||
|     } | ||||
|  | ||||
|     { | ||||
|     typedef PrivateConstructor PC; | ||||
|     typedef std::multimap<PC, double, std::less<>> M; | ||||
|     typedef M::size_type R; | ||||
|  | ||||
|     M m; | ||||
|     m.insert ( std::make_pair<PC, double> ( PC::make(5), 1 )); | ||||
|     m.insert ( std::make_pair<PC, double> ( PC::make(5), 2 )); | ||||
|     m.insert ( std::make_pair<PC, double> ( PC::make(5), 3 )); | ||||
|     m.insert ( std::make_pair<PC, double> ( PC::make(7), 1 )); | ||||
|     m.insert ( std::make_pair<PC, double> ( PC::make(7), 2 )); | ||||
|     m.insert ( std::make_pair<PC, double> ( PC::make(7), 3 )); | ||||
|     m.insert ( std::make_pair<PC, double> ( PC::make(9), 1 )); | ||||
|     m.insert ( std::make_pair<PC, double> ( PC::make(9), 2 )); | ||||
|     m.insert ( std::make_pair<PC, double> ( PC::make(9), 3 )); | ||||
|  | ||||
|     R r = m.count(4); | ||||
|     assert(r == 0); | ||||
|     r = m.count(5); | ||||
|     assert(r == 3); | ||||
|     r = m.count(6); | ||||
|     assert(r == 0); | ||||
|     r = m.count(7); | ||||
|     assert(r == 3); | ||||
|     r = m.count(8); | ||||
|     assert(r == 0); | ||||
|     r = m.count(9); | ||||
|     assert(r == 3); | ||||
|     r = m.count(10); | ||||
|     assert(r == 0); | ||||
|     } | ||||
| #endif | ||||
| } | ||||
|   | ||||
| @@ -17,6 +17,7 @@ | ||||
| #include <cassert> | ||||
|  | ||||
| #include "min_allocator.h" | ||||
| #include "private_constructor.hpp" | ||||
|  | ||||
| int main() | ||||
| { | ||||
| @@ -90,4 +91,70 @@ int main() | ||||
|     } | ||||
|     } | ||||
| #endif | ||||
| #if _LIBCPP_STD_VER > 11 | ||||
|     { | ||||
|     typedef int V; | ||||
|     typedef std::multiset<int, std::less<>> M; | ||||
|     typedef M::size_type R; | ||||
|     V ar[] = | ||||
|     { | ||||
|         5, | ||||
|         5, | ||||
|         5, | ||||
|         5, | ||||
|         7, | ||||
|         7, | ||||
|         7, | ||||
|         9, | ||||
|         9 | ||||
|     }; | ||||
|     const M m(ar, ar+sizeof(ar)/sizeof(ar[0])); | ||||
|     R r = m.count(4); | ||||
|     assert(r == 0); | ||||
|     r = m.count(5); | ||||
|     assert(r == 4); | ||||
|     r = m.count(6); | ||||
|     assert(r == 0); | ||||
|     r = m.count(7); | ||||
|     assert(r == 3); | ||||
|     r = m.count(8); | ||||
|     assert(r == 0); | ||||
|     r = m.count(9); | ||||
|     assert(r == 2); | ||||
|     r = m.count(10); | ||||
|     assert(r == 0); | ||||
|     } | ||||
|  | ||||
|     { | ||||
|     typedef PrivateConstructor V; | ||||
|     typedef std::multiset<V, std::less<>> M; | ||||
|     typedef M::size_type R; | ||||
|  | ||||
|     M m; | ||||
|     m.insert ( V::make ( 5 )); | ||||
|     m.insert ( V::make ( 5 )); | ||||
|     m.insert ( V::make ( 5 )); | ||||
|     m.insert ( V::make ( 5 )); | ||||
|     m.insert ( V::make ( 7 )); | ||||
|     m.insert ( V::make ( 7 )); | ||||
|     m.insert ( V::make ( 7 )); | ||||
|     m.insert ( V::make ( 9 )); | ||||
|     m.insert ( V::make ( 9 )); | ||||
|  | ||||
|     R r = m.count(4); | ||||
|     assert(r == 0); | ||||
|     r = m.count(5); | ||||
|     assert(r == 4); | ||||
|     r = m.count(6); | ||||
|     assert(r == 0); | ||||
|     r = m.count(7); | ||||
|     assert(r == 3); | ||||
|     r = m.count(8); | ||||
|     assert(r == 0); | ||||
|     r = m.count(9); | ||||
|     assert(r == 2); | ||||
|     r = m.count(10); | ||||
|     assert(r == 0); | ||||
|     } | ||||
| #endif | ||||
| } | ||||
|   | ||||
| @@ -17,6 +17,7 @@ | ||||
| #include <cassert> | ||||
|  | ||||
| #include "min_allocator.h" | ||||
| #include "private_constructor.hpp" | ||||
|  | ||||
| int main() | ||||
| { | ||||
| @@ -92,4 +93,76 @@ int main() | ||||
|         assert(r == 0); | ||||
|     } | ||||
| #endif | ||||
| #if _LIBCPP_STD_VER > 11 | ||||
|     { | ||||
|         typedef int V; | ||||
|         typedef std::set<int, std::less<>> M; | ||||
|         typedef M::size_type R; | ||||
|         V ar[] = | ||||
|         { | ||||
|             5, | ||||
|             6, | ||||
|             7, | ||||
|             8, | ||||
|             9, | ||||
|             10, | ||||
|             11, | ||||
|             12 | ||||
|         }; | ||||
|         const M m(ar, ar+sizeof(ar)/sizeof(ar[0])); | ||||
|         R r = m.count(5); | ||||
|         assert(r == 1); | ||||
|         r = m.count(6); | ||||
|         assert(r == 1); | ||||
|         r = m.count(7); | ||||
|         assert(r == 1); | ||||
|         r = m.count(8); | ||||
|         assert(r == 1); | ||||
|         r = m.count(9); | ||||
|         assert(r == 1); | ||||
|         r = m.count(10); | ||||
|         assert(r == 1); | ||||
|         r = m.count(11); | ||||
|         assert(r == 1); | ||||
|         r = m.count(12); | ||||
|         assert(r == 1); | ||||
|         r = m.count(4); | ||||
|         assert(r == 0); | ||||
|     } | ||||
|     { | ||||
|     typedef PrivateConstructor V; | ||||
|     typedef std::set<V, std::less<>> M; | ||||
|         typedef M::size_type R; | ||||
|  | ||||
|     M m; | ||||
|     m.insert ( V::make ( 5 )); | ||||
|     m.insert ( V::make ( 6 )); | ||||
|     m.insert ( V::make ( 7 )); | ||||
|     m.insert ( V::make ( 8 )); | ||||
|     m.insert ( V::make ( 9 )); | ||||
|     m.insert ( V::make ( 10 )); | ||||
|     m.insert ( V::make ( 11 )); | ||||
|     m.insert ( V::make ( 12 )); | ||||
|  | ||||
|     R r = m.count(5); | ||||
|     assert(r == 1); | ||||
|     r = m.count(6); | ||||
|     assert(r == 1); | ||||
|     r = m.count(7); | ||||
|     assert(r == 1); | ||||
|     r = m.count(8); | ||||
|     assert(r == 1); | ||||
|     r = m.count(9); | ||||
|     assert(r == 1); | ||||
|     r = m.count(10); | ||||
|     assert(r == 1); | ||||
|     r = m.count(11); | ||||
|     assert(r == 1); | ||||
|     r = m.count(12); | ||||
|     assert(r == 1); | ||||
|     r = m.count(4); | ||||
|     assert(r == 0); | ||||
|     } | ||||
| #endif | ||||
|  | ||||
| } | ||||
|   | ||||
		Reference in New Issue
	
	Block a user
	 Marshall Clow
					Marshall Clow