mirror of git://gcc.gnu.org/git/gcc.git
				
				
				
			PR libstdc++/86734 make reverse_iterator::operator-> more robust
Implement the proposed resolution from LWG 1052, which also resolves DR 2118 by avoiding taking the address in the first place. PR libstdc++/86734 * include/bits/stl_iterator.h (reverse_iterator::operator->): Call _S_to_pointer (LWG 1052, LWG 2118). (reverse_iterator::_S_to_pointer): Define overloaded helper functions. * testsuite/24_iterators/reverse_iterator/dr1052.cc: New test. * testsuite/24_iterators/reverse_iterator/dr2188.cc: New test. From-SVN: r263074
This commit is contained in:
		
							parent
							
								
									1b3b888d11
								
							
						
					
					
						commit
						a64ede727f
					
				|  | @ -1,5 +1,12 @@ | |||
| 2018-07-30  Jonathan Wakely  <jwakely@redhat.com> | ||||
| 
 | ||||
| 	PR libstdc++/86734 | ||||
| 	* include/bits/stl_iterator.h (reverse_iterator::operator->): Call | ||||
| 	_S_to_pointer (LWG 1052, LWG 2118). | ||||
| 	(reverse_iterator::_S_to_pointer): Define overloaded helper functions. | ||||
| 	* testsuite/24_iterators/reverse_iterator/dr1052.cc: New test. | ||||
| 	* testsuite/24_iterators/reverse_iterator/dr2188.cc: New test. | ||||
| 
 | ||||
| 	* libsupc++/new_opa.cc (operator new(size_t, align_val_t)): Add | ||||
| 	workaround for aligned_alloc bug on AIX. | ||||
| 	* testsuite/18_support/new_aligned.cc: New test. | ||||
|  |  | |||
|  | @ -122,6 +122,7 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION | |||
|       */ | ||||
|       // _GLIBCXX_RESOLVE_LIB_DEFECTS
 | ||||
|       // 235 No specification of default ctor for reverse_iterator
 | ||||
|       // 1012. reverse_iterator default ctor should value initialize
 | ||||
|       _GLIBCXX17_CONSTEXPR | ||||
|       reverse_iterator() : current() { } | ||||
| 
 | ||||
|  | @ -182,7 +183,13 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION | |||
|       */ | ||||
|       _GLIBCXX17_CONSTEXPR pointer | ||||
|       operator->() const | ||||
|       { return &(operator*()); } | ||||
|       { | ||||
| 	// _GLIBCXX_RESOLVE_LIB_DEFECTS
 | ||||
| 	// 1052. operator-> should also support smart pointers
 | ||||
| 	_Iterator __tmp = current; | ||||
| 	--__tmp; | ||||
| 	return _S_to_pointer(__tmp); | ||||
|       } | ||||
| 
 | ||||
|       /**
 | ||||
|        *  @return  @c *this | ||||
|  | @ -286,6 +293,17 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION | |||
|       _GLIBCXX17_CONSTEXPR reference | ||||
|       operator[](difference_type __n) const | ||||
|       { return *(*this + __n); } | ||||
| 
 | ||||
|     private: | ||||
|       template<typename _Tp> | ||||
| 	static _GLIBCXX17_CONSTEXPR _Tp* | ||||
| 	_S_to_pointer(_Tp* __p) | ||||
|         { return __p; } | ||||
| 
 | ||||
|       template<typename _Tp> | ||||
| 	static _GLIBCXX17_CONSTEXPR pointer | ||||
| 	_S_to_pointer(_Tp __t) | ||||
|         { return __t.operator->(); } | ||||
|     }; | ||||
| 
 | ||||
|   //@{
 | ||||
|  |  | |||
|  | @ -0,0 +1,82 @@ | |||
| // Copyright (C) 2018 Free Software Foundation, Inc.
 | ||||
| //
 | ||||
| // This file is part of the GNU ISO C++ Library.  This library is free
 | ||||
| // software; you can redistribute it and/or modify it under the
 | ||||
| // terms of the GNU General Public License as published by the
 | ||||
| // Free Software Foundation; either version 3, or (at your option)
 | ||||
| // any later version.
 | ||||
| 
 | ||||
| // This library is distributed in the hope that it will be useful,
 | ||||
| // but WITHOUT ANY WARRANTY; without even the implied warranty of
 | ||||
| // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | ||||
| // GNU General Public License for more details.
 | ||||
| 
 | ||||
| // You should have received a copy of the GNU General Public License along
 | ||||
| // with this library; see the file COPYING3.  If not see
 | ||||
| // <http://www.gnu.org/licenses/>.
 | ||||
| 
 | ||||
| // { dg-do run { target c++11 } }
 | ||||
| 
 | ||||
| // PR libstdc++/86734
 | ||||
| // LWG 1052. reverse_iterator::operator-> should also support smart pointers
 | ||||
| // LWG 2775. reverse_iterator is does not compile for fancy pointers
 | ||||
| 
 | ||||
| #include <iterator> | ||||
| #include <testsuite_hooks.h> | ||||
| 
 | ||||
| void | ||||
| test01() | ||||
| { | ||||
|   // Example 1 from LWG 1052
 | ||||
| 
 | ||||
|   struct X { int m; }; | ||||
| 
 | ||||
|   static X x; | ||||
| 
 | ||||
|   struct IterX { | ||||
|     typedef std::bidirectional_iterator_tag iterator_category; | ||||
|     typedef X& reference; | ||||
|     struct pointer | ||||
|     { | ||||
|       pointer(X& v) : value(v) {} | ||||
|       X& value; | ||||
|       X* operator->() const {return &value;} | ||||
|     }; | ||||
|     typedef std::ptrdiff_t difference_type; | ||||
|     typedef X value_type; | ||||
|     // additional iterator requirements not important for this issue
 | ||||
| 
 | ||||
|     reference operator*() const { return x; } | ||||
|     pointer operator->() const { return pointer(x); } | ||||
|     IterX& operator--() {return *this;} | ||||
| 
 | ||||
|   }; | ||||
| 
 | ||||
|   std::reverse_iterator<IterX> ix; | ||||
|   VERIFY( &ix->m == &(*ix).m ); | ||||
| } | ||||
| 
 | ||||
| void | ||||
| test02() | ||||
| { | ||||
|   // Example 2 from LWG 1052
 | ||||
| 
 | ||||
|   struct P { | ||||
|     P() : first(10), second(20.0) { } | ||||
|     int first; | ||||
|     double second; | ||||
|   }; | ||||
|   P op; | ||||
|   std::reverse_iterator<P*> ri(&op + 1); | ||||
|   VERIFY( ri->first == 10 ); | ||||
| } | ||||
| 
 | ||||
| // N.B. Example 3 from LWG 1052 isn't expected to work,
 | ||||
| // because a caching iterator like IterX is not a forward iterator.
 | ||||
| 
 | ||||
| int | ||||
| main() | ||||
| { | ||||
|   test01(); | ||||
|   test02(); | ||||
| } | ||||
|  | @ -0,0 +1,47 @@ | |||
| // Copyright (C) 2018 Free Software Foundation, Inc.
 | ||||
| //
 | ||||
| // This file is part of the GNU ISO C++ Library.  This library is free
 | ||||
| // software; you can redistribute it and/or modify it under the
 | ||||
| // terms of the GNU General Public License as published by the
 | ||||
| // Free Software Foundation; either version 3, or (at your option)
 | ||||
| // any later version.
 | ||||
| 
 | ||||
| // This library is distributed in the hope that it will be useful,
 | ||||
| // but WITHOUT ANY WARRANTY; without even the implied warranty of
 | ||||
| // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | ||||
| // GNU General Public License for more details.
 | ||||
| 
 | ||||
| // You should have received a copy of the GNU General Public License along
 | ||||
| // with this library; see the file COPYING3.  If not see
 | ||||
| // <http://www.gnu.org/licenses/>.
 | ||||
| 
 | ||||
| // { dg-do run { target c++11 } }
 | ||||
| 
 | ||||
| // PR libstdc++/86734
 | ||||
| 
 | ||||
| #include <iterator> | ||||
| #include <testsuite_hooks.h> | ||||
| 
 | ||||
| void | ||||
| test01() | ||||
| { | ||||
|   // LWG DR 2188
 | ||||
|   // Reverse iterator does not fully support targets that overload operator&
 | ||||
|   struct X { | ||||
|     int val; | ||||
|     int* operator&() { return &val; } | ||||
|     const int* operator&() const { return &val; } | ||||
|   }; | ||||
| 
 | ||||
|   X x[2] = { {1}, {2} }; | ||||
|   std::reverse_iterator<X*> rev(x+2); | ||||
|   VERIFY( rev->val == 2 ); | ||||
|   ++rev; | ||||
|   VERIFY( rev->val == 1 ); | ||||
| } | ||||
| 
 | ||||
| int | ||||
| main() | ||||
| { | ||||
|   test01(); | ||||
| } | ||||
		Loading…
	
		Reference in New Issue
	
	 Jonathan Wakely
						Jonathan Wakely