mirror of git://gcc.gnu.org/git/gcc.git
				
				
				
			Fix incorrect results from std::boyer_moore_searcher
* include/std/functional (default_searcher, __boyer_moore_array_base) (__is_std_equal_to, __boyer_moore_base_t, boyer_moore_searcher) (boyer_moore_horspool_searcher): Remove redundant namespace qualification. (default_searcher::operator()): Construct return value early and advance second member in-place. (boyer_moore_horspool_searcher::operator()): Increment random access iterator directly instead of using std::next. (boyer_moore_searcher::operator()): Fix return value. * testsuite/20_util/function_objects/searchers.cc: Check both parts of return values. From-SVN: r246964
This commit is contained in:
		
							parent
							
								
									d6e3e8a54c
								
							
						
					
					
						commit
						38d68cf0f5
					
				|  | @ -1,3 +1,17 @@ | |||
| 2017-04-18  Jonathan Wakely  <jwakely@redhat.com> | ||||
| 
 | ||||
| 	* include/std/functional (default_searcher, __boyer_moore_array_base) | ||||
| 	(__is_std_equal_to, __boyer_moore_base_t, boyer_moore_searcher) | ||||
| 	(boyer_moore_horspool_searcher): Remove redundant namespace | ||||
| 	qualification. | ||||
| 	(default_searcher::operator()): Construct return value early and | ||||
| 	advance second member in-place. | ||||
| 	(boyer_moore_horspool_searcher::operator()): Increment random access | ||||
| 	iterator directly instead of using std::next. | ||||
| 	(boyer_moore_searcher::operator()): Fix return value. | ||||
| 	* testsuite/20_util/function_objects/searchers.cc: Check both parts | ||||
| 	of return values. | ||||
| 
 | ||||
| 2017-04-12  Gerald Pfeifer  <gerald@pfeifer.com> | ||||
| 
 | ||||
| 	* doc/xml/faq.xml: Update reference link to C++ ABI for Itanium. | ||||
|  |  | |||
|  | @ -969,17 +969,17 @@ _GLIBCXX_MEM_FN_TRAITS(&&, false_type, true_type) | |||
| 	operator()(_ForwardIterator2 __first, _ForwardIterator2 __last) const | ||||
| 	{ | ||||
| 	  _ForwardIterator2 __first_ret = | ||||
| 	    std::search(__first, __last, | ||||
| 			std::get<0>(_M_m), std::get<1>(_M_m), | ||||
| 	    std::search(__first, __last, std::get<0>(_M_m), std::get<1>(_M_m), | ||||
| 			std::get<2>(_M_m)); | ||||
| 	  _ForwardIterator2 __second_ret = __first_ret == __last ? | ||||
| 	    __last :  std::next(__first_ret, std::distance(std::get<0>(_M_m), | ||||
| 							   std::get<1>(_M_m))); | ||||
| 	  return std::make_pair(__first_ret, __second_ret); | ||||
| 	  auto __ret = std::make_pair(__first_ret, __first_ret); | ||||
| 	  if (__ret.first != __last) | ||||
| 	    std::advance(__ret.second, std::distance(std::get<0>(_M_m), | ||||
| 						     std::get<1>(_M_m))); | ||||
| 	  return __ret; | ||||
| 	} | ||||
| 
 | ||||
|     private: | ||||
|       std::tuple<_ForwardIterator1, _ForwardIterator1, _BinaryPredicate> _M_m; | ||||
|       tuple<_ForwardIterator1, _ForwardIterator1, _BinaryPredicate> _M_m; | ||||
|     }; | ||||
| 
 | ||||
|   template<typename _Key, typename _Tp, typename _Hash, typename _Pred> | ||||
|  | @ -1025,7 +1025,7 @@ _GLIBCXX_MEM_FN_TRAITS(&&, false_type, true_type) | |||
| 	    for (__diff_type __i = 0; __i < __patlen - 1; ++__i) | ||||
| 	      { | ||||
| 		auto __ch = __pat[__i]; | ||||
| 		using _UCh = std::make_unsigned_t<decltype(__ch)>; | ||||
| 		using _UCh = make_unsigned_t<decltype(__ch)>; | ||||
| 		auto __uch = static_cast<_UCh>(__ch); | ||||
| 		std::get<0>(_M_bad_char)[__uch] = __patlen - 1 - __i; | ||||
| 	      } | ||||
|  | @ -1037,7 +1037,7 @@ _GLIBCXX_MEM_FN_TRAITS(&&, false_type, true_type) | |||
| 	__diff_type | ||||
| 	_M_lookup(_Key __key, __diff_type __not_found) const | ||||
| 	{ | ||||
| 	  auto __ukey = static_cast<std::make_unsigned_t<_Key>>(__key); | ||||
| 	  auto __ukey = static_cast<make_unsigned_t<_Key>>(__key); | ||||
| 	  if (__ukey >= _Len) | ||||
| 	    return __not_found; | ||||
| 	  return std::get<0>(_M_bad_char)[__ukey]; | ||||
|  | @ -1046,14 +1046,14 @@ _GLIBCXX_MEM_FN_TRAITS(&&, false_type, true_type) | |||
|       const _Pred& | ||||
|       _M_pred() const { return std::get<1>(_M_bad_char); } | ||||
| 
 | ||||
|       std::tuple<_GLIBCXX_STD_C::array<_Tp, _Len>, _Pred> _M_bad_char; | ||||
|       tuple<_GLIBCXX_STD_C::array<_Tp, _Len>, _Pred> _M_bad_char; | ||||
|     }; | ||||
| 
 | ||||
|   template<typename _Pred> | ||||
|     struct __is_std_equal_to : std::false_type { }; | ||||
|     struct __is_std_equal_to : false_type { }; | ||||
| 
 | ||||
|   template<> | ||||
|     struct __is_std_equal_to<std::equal_to<void>> : std::true_type { }; | ||||
|     struct __is_std_equal_to<equal_to<void>> : true_type { }; | ||||
| 
 | ||||
|   // Use __boyer_moore_array_base when pattern consists of narrow characters | ||||
|   // and uses std::equal_to as the predicate. | ||||
|  | @ -1061,14 +1061,14 @@ _GLIBCXX_MEM_FN_TRAITS(&&, false_type, true_type) | |||
|            typename _Val = typename iterator_traits<_RAIter>::value_type, | ||||
| 	   typename _Diff = typename iterator_traits<_RAIter>::difference_type> | ||||
|     using __boyer_moore_base_t | ||||
|       = std::conditional_t<sizeof(_Val) == 1 && is_integral<_Val>::value | ||||
| 			   && __is_std_equal_to<_Pred>::value, | ||||
| 			   __boyer_moore_array_base<_Diff, 256, _Pred>, | ||||
| 			   __boyer_moore_map_base<_Val, _Diff, _Hash, _Pred>>; | ||||
|       = conditional_t<sizeof(_Val) == 1 && is_integral<_Val>::value | ||||
| 		      && __is_std_equal_to<_Pred>::value, | ||||
| 		      __boyer_moore_array_base<_Diff, 256, _Pred>, | ||||
| 		      __boyer_moore_map_base<_Val, _Diff, _Hash, _Pred>>; | ||||
| 
 | ||||
|   template<typename _RAIter, typename _Hash | ||||
| 	     = std::hash<typename std::iterator_traits<_RAIter>::value_type>, | ||||
| 	   typename _BinaryPredicate = std::equal_to<>> | ||||
| 	     = hash<typename iterator_traits<_RAIter>::value_type>, | ||||
| 	   typename _BinaryPredicate = equal_to<>> | ||||
|     class boyer_moore_searcher | ||||
|     : __boyer_moore_base_t<_RAIter, _Hash, _BinaryPredicate> | ||||
|     { | ||||
|  | @ -1123,8 +1123,8 @@ _GLIBCXX_MEM_FN_TRAITS(&&, false_type, true_type) | |||
|     }; | ||||
| 
 | ||||
|   template<typename _RAIter, typename _Hash | ||||
| 	     = std::hash<typename std::iterator_traits<_RAIter>::value_type>, | ||||
| 	   typename _BinaryPredicate = std::equal_to<>> | ||||
| 	     = hash<typename iterator_traits<_RAIter>::value_type>, | ||||
| 	   typename _BinaryPredicate = equal_to<>> | ||||
|     class boyer_moore_horspool_searcher | ||||
|     : __boyer_moore_base_t<_RAIter, _Hash, _BinaryPredicate> | ||||
|     { | ||||
|  | @ -1156,8 +1156,7 @@ _GLIBCXX_MEM_FN_TRAITS(&&, false_type, true_type) | |||
| 	      for (auto __scan = __patlen - 1; | ||||
| 		   __pred(__first[__scan], _M_pat[__scan]); --__scan) | ||||
| 		if (__scan == 0) | ||||
| 		  return std::make_pair(__first, | ||||
| 					std::next(__first, __patlen)); | ||||
| 		  return std::make_pair(__first, __first + __patlen); | ||||
| 	      auto __shift = _M_bad_char_shift(__first[__patlen - 1]); | ||||
| 	      __len -= __shift; | ||||
| 	      __first += __shift; | ||||
|  | @ -1223,8 +1222,10 @@ _GLIBCXX_MEM_FN_TRAITS(&&, false_type, true_type) | |||
| 	      --__j; | ||||
| 	    } | ||||
| 	  if (__j < 0) | ||||
| 	    return std::make_pair(__first + __i + 1, std::next(__first, | ||||
| 							       __patlen)); | ||||
| 	    { | ||||
| 	      const auto __match = __first + __i + 1; | ||||
| 	      return std::make_pair(__match, __match + __patlen); | ||||
| 	    } | ||||
| 	  __i += std::max(_M_bad_char_shift(__first[__i]), | ||||
| 			  _M_good_suffix[__j]); | ||||
| 	} | ||||
|  |  | |||
|  | @ -49,7 +49,8 @@ test01() | |||
| 
 | ||||
|   for (auto n : needles) | ||||
|   { | ||||
|     auto ne = n + std::strlen(n); | ||||
|     auto nlen = std::strlen(n); | ||||
|     auto ne = n + nlen; | ||||
|     default_searcher d(n, ne); | ||||
|     boyer_moore_searcher bm(n, ne); | ||||
|     boyer_moore_horspool_searcher bmh(n, ne); | ||||
|  | @ -59,10 +60,22 @@ test01() | |||
|       auto res = std::search(h, he, n, ne); | ||||
|       auto d_res = d(h, he); | ||||
|       VERIFY( d_res.first == res ); | ||||
|       if (res == he) | ||||
| 	VERIFY( d_res.second == d_res.first ); | ||||
|       else | ||||
| 	VERIFY( d_res.second == (d_res.first + nlen) ); | ||||
|       auto bm_res = bm(h, he); | ||||
|       VERIFY( bm_res.first == res ); | ||||
|       if (res == he) | ||||
| 	VERIFY( bm_res.second == bm_res.first ); | ||||
|       else | ||||
| 	VERIFY( bm_res.second == (bm_res.first + nlen) ); | ||||
|       auto bmh_res = bmh(h, he); | ||||
|       VERIFY( bmh_res.first == res ); | ||||
|       if (res == he) | ||||
| 	VERIFY( bmh_res.second == bmh_res.first ); | ||||
|       else | ||||
| 	VERIFY( bmh_res.second == (bmh_res.first + nlen) ); | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | @ -82,7 +95,8 @@ test02() | |||
| 
 | ||||
|   for (auto n : needles) | ||||
|   { | ||||
|     auto ne = n + std::wcslen(n); | ||||
|     auto nlen = std::wcslen(n); | ||||
|     auto ne = n + nlen; | ||||
|     default_searcher d(n, ne); | ||||
|     boyer_moore_searcher bm(n, ne); | ||||
|     boyer_moore_horspool_searcher bmh(n, ne); | ||||
|  | @ -92,10 +106,22 @@ test02() | |||
|       auto res = std::search(h, he, n, ne); | ||||
|       auto d_res = d(h, he); | ||||
|       VERIFY( d_res.first == res ); | ||||
|       if (res == he) | ||||
| 	VERIFY( d_res.second == d_res.first ); | ||||
|       else | ||||
| 	VERIFY( d_res.second == (d_res.first + nlen) ); | ||||
|       auto bm_res = bm(h, he); | ||||
|       VERIFY( bm_res.first == res ); | ||||
|       if (res == he) | ||||
| 	VERIFY( bm_res.second == bm_res.first ); | ||||
|       else | ||||
| 	VERIFY( bm_res.second == (bm_res.first + nlen) ); | ||||
|       auto bmh_res = bmh(h, he); | ||||
|       VERIFY( bmh_res.first == res ); | ||||
|       if (res == he) | ||||
| 	VERIFY( bmh_res.second == bmh_res.first ); | ||||
|       else | ||||
| 	VERIFY( bmh_res.second == (bmh_res.first + nlen) ); | ||||
|     } | ||||
|   } | ||||
| #endif | ||||
|  | @ -119,7 +145,8 @@ test03() | |||
| 
 | ||||
|   const char* needle = " foo 123 "; | ||||
|   const char* haystack = "*****foo*123******"; | ||||
|   const char* ne = needle + std::strlen(needle); | ||||
|   auto nlen = std::strlen(needle); | ||||
|   const char* ne = needle + nlen; | ||||
|   const char* he = haystack + std::strlen(haystack); | ||||
| 
 | ||||
|   default_searcher d(needle, ne, eq); | ||||
|  | @ -129,10 +156,22 @@ test03() | |||
|   auto res = std::search(haystack, he, needle, ne, eq); | ||||
|   auto d_res = d(haystack, he); | ||||
|   VERIFY( d_res.first == res ); | ||||
|   if (res == he) | ||||
|     VERIFY( d_res.second == d_res.first ); | ||||
|   else | ||||
|     VERIFY( d_res.second == (d_res.first + nlen) ); | ||||
|   auto bm_res = bm(haystack, he); | ||||
|   VERIFY( bm_res.first == res ); | ||||
|   if (res == he) | ||||
|     VERIFY( bm_res.second == bm_res.first ); | ||||
|   else | ||||
|     VERIFY( bm_res.second == (bm_res.first + nlen) ); | ||||
|   auto bmh_res = bmh(haystack, he); | ||||
|   VERIFY( bmh_res.first == res ); | ||||
|   if (res == he) | ||||
|     VERIFY( bmh_res.second == bmh_res.first ); | ||||
|   else | ||||
|     VERIFY( bmh_res.second == (bmh_res.first + nlen) ); | ||||
| } | ||||
| 
 | ||||
| int | ||||
|  |  | |||
		Loading…
	
		Reference in New Issue
	
	 Jonathan Wakely
						Jonathan Wakely