1*e4b17023SJohn Marino // Debugging iterator implementation (out of line) -*- C++ -*- 2*e4b17023SJohn Marino 3*e4b17023SJohn Marino // Copyright (C) 2003, 2004, 2005, 2006, 2007, 2009, 2010, 2011, 2012 4*e4b17023SJohn Marino // Free Software Foundation, Inc. 5*e4b17023SJohn Marino // 6*e4b17023SJohn Marino // This file is part of the GNU ISO C++ Library. This library is free 7*e4b17023SJohn Marino // software; you can redistribute it and/or modify it under the 8*e4b17023SJohn Marino // terms of the GNU General Public License as published by the 9*e4b17023SJohn Marino // Free Software Foundation; either version 3, or (at your option) 10*e4b17023SJohn Marino // any later version. 11*e4b17023SJohn Marino 12*e4b17023SJohn Marino // This library is distributed in the hope that it will be useful, 13*e4b17023SJohn Marino // but WITHOUT ANY WARRANTY; without even the implied warranty of 14*e4b17023SJohn Marino // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 15*e4b17023SJohn Marino // GNU General Public License for more details. 16*e4b17023SJohn Marino 17*e4b17023SJohn Marino // Under Section 7 of GPL version 3, you are granted additional 18*e4b17023SJohn Marino // permissions described in the GCC Runtime Library Exception, version 19*e4b17023SJohn Marino // 3.1, as published by the Free Software Foundation. 20*e4b17023SJohn Marino 21*e4b17023SJohn Marino // You should have received a copy of the GNU General Public License and 22*e4b17023SJohn Marino // a copy of the GCC Runtime Library Exception along with this program; 23*e4b17023SJohn Marino // see the files COPYING3 and COPYING.RUNTIME respectively. If not, see 24*e4b17023SJohn Marino // <http://www.gnu.org/licenses/>. 25*e4b17023SJohn Marino 26*e4b17023SJohn Marino /** @file debug/safe_iterator.tcc 27*e4b17023SJohn Marino * This file is a GNU debug extension to the Standard C++ Library. 28*e4b17023SJohn Marino */ 29*e4b17023SJohn Marino 30*e4b17023SJohn Marino #ifndef _GLIBCXX_DEBUG_SAFE_ITERATOR_TCC 31*e4b17023SJohn Marino #define _GLIBCXX_DEBUG_SAFE_ITERATOR_TCC 1 32*e4b17023SJohn Marino 33*e4b17023SJohn Marino namespace __gnu_debug 34*e4b17023SJohn Marino { 35*e4b17023SJohn Marino template<typename _Iterator, typename _Sequence> 36*e4b17023SJohn Marino bool 37*e4b17023SJohn Marino _Safe_iterator<_Iterator, _Sequence>:: _M_can_advance(const difference_type & __n) const38*e4b17023SJohn Marino _M_can_advance(const difference_type& __n) const 39*e4b17023SJohn Marino { 40*e4b17023SJohn Marino typedef typename _Sequence::const_iterator const_debug_iterator; 41*e4b17023SJohn Marino typedef typename const_debug_iterator::iterator_type const_iterator; 42*e4b17023SJohn Marino 43*e4b17023SJohn Marino if (this->_M_singular()) 44*e4b17023SJohn Marino return false; 45*e4b17023SJohn Marino if (__n == 0) 46*e4b17023SJohn Marino return true; 47*e4b17023SJohn Marino if (__n < 0) 48*e4b17023SJohn Marino { 49*e4b17023SJohn Marino const_iterator __begin = _M_get_sequence()->_M_base().begin(); 50*e4b17023SJohn Marino std::pair<difference_type, _Distance_precision> __dist = 51*e4b17023SJohn Marino __get_distance(__begin, base()); 52*e4b17023SJohn Marino bool __ok = ((__dist.second == __dp_exact && __dist.first >= -__n) 53*e4b17023SJohn Marino || (__dist.second != __dp_exact && __dist.first > 0)); 54*e4b17023SJohn Marino return __ok; 55*e4b17023SJohn Marino } 56*e4b17023SJohn Marino else 57*e4b17023SJohn Marino { 58*e4b17023SJohn Marino const_iterator __end = _M_get_sequence()->_M_base().end(); 59*e4b17023SJohn Marino std::pair<difference_type, _Distance_precision> __dist = 60*e4b17023SJohn Marino __get_distance(base(), __end); 61*e4b17023SJohn Marino bool __ok = ((__dist.second == __dp_exact && __dist.first >= __n) 62*e4b17023SJohn Marino || (__dist.second != __dp_exact && __dist.first > 0)); 63*e4b17023SJohn Marino return __ok; 64*e4b17023SJohn Marino } 65*e4b17023SJohn Marino } 66*e4b17023SJohn Marino 67*e4b17023SJohn Marino template<typename _Iterator, typename _Sequence> 68*e4b17023SJohn Marino template<typename _Other> 69*e4b17023SJohn Marino bool 70*e4b17023SJohn Marino _Safe_iterator<_Iterator, _Sequence>:: _M_valid_range(const _Safe_iterator<_Other,_Sequence> & __rhs) const71*e4b17023SJohn Marino _M_valid_range(const _Safe_iterator<_Other, _Sequence>& __rhs) const 72*e4b17023SJohn Marino { 73*e4b17023SJohn Marino if (!_M_can_compare(__rhs)) 74*e4b17023SJohn Marino return false; 75*e4b17023SJohn Marino 76*e4b17023SJohn Marino /* Determine if we can order the iterators without the help of 77*e4b17023SJohn Marino the container */ 78*e4b17023SJohn Marino std::pair<difference_type, _Distance_precision> __dist = 79*e4b17023SJohn Marino __get_distance(base(), __rhs.base()); 80*e4b17023SJohn Marino switch (__dist.second) { 81*e4b17023SJohn Marino case __dp_equality: 82*e4b17023SJohn Marino if (__dist.first == 0) 83*e4b17023SJohn Marino return true; 84*e4b17023SJohn Marino break; 85*e4b17023SJohn Marino 86*e4b17023SJohn Marino case __dp_sign: 87*e4b17023SJohn Marino case __dp_exact: 88*e4b17023SJohn Marino return __dist.first >= 0; 89*e4b17023SJohn Marino } 90*e4b17023SJohn Marino 91*e4b17023SJohn Marino /* We can only test for equality, but check if one of the 92*e4b17023SJohn Marino iterators is at an extreme. */ 93*e4b17023SJohn Marino /* Optim for classic [begin, it) or [it, end) ranges, limit checks 94*e4b17023SJohn Marino * when code is valid. Note, for the special case of forward_list, 95*e4b17023SJohn Marino * before_begin replaces the role of begin. */ 96*e4b17023SJohn Marino if (_M_is_beginnest() || __rhs._M_is_end()) 97*e4b17023SJohn Marino return true; 98*e4b17023SJohn Marino if (_M_is_end() || __rhs._M_is_beginnest()) 99*e4b17023SJohn Marino return false; 100*e4b17023SJohn Marino 101*e4b17023SJohn Marino // Assume that this is a valid range; we can't check anything else 102*e4b17023SJohn Marino return true; 103*e4b17023SJohn Marino } 104*e4b17023SJohn Marino } // namespace __gnu_debug 105*e4b17023SJohn Marino 106*e4b17023SJohn Marino #endif 107*e4b17023SJohn Marino 108