1*f8a1b7d9SAlexander Kabaev // Debugging support implementation -*- C++ -*- 2*f8a1b7d9SAlexander Kabaev 3*f8a1b7d9SAlexander Kabaev // Copyright (C) 2003, 2005, 2006 4*f8a1b7d9SAlexander Kabaev // Free Software Foundation, Inc. 5*f8a1b7d9SAlexander Kabaev // 6*f8a1b7d9SAlexander Kabaev // This file is part of the GNU ISO C++ Library. This library is free 7*f8a1b7d9SAlexander Kabaev // software; you can redistribute it and/or modify it under the 8*f8a1b7d9SAlexander Kabaev // terms of the GNU General Public License as published by the 9*f8a1b7d9SAlexander Kabaev // Free Software Foundation; either version 2, or (at your option) 10*f8a1b7d9SAlexander Kabaev // any later version. 11*f8a1b7d9SAlexander Kabaev 12*f8a1b7d9SAlexander Kabaev // This library is distributed in the hope that it will be useful, 13*f8a1b7d9SAlexander Kabaev // but WITHOUT ANY WARRANTY; without even the implied warranty of 14*f8a1b7d9SAlexander Kabaev // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 15*f8a1b7d9SAlexander Kabaev // GNU General Public License for more details. 16*f8a1b7d9SAlexander Kabaev 17*f8a1b7d9SAlexander Kabaev // You should have received a copy of the GNU General Public License along 18*f8a1b7d9SAlexander Kabaev // with this library; see the file COPYING. If not, write to the Free 19*f8a1b7d9SAlexander Kabaev // Software Foundation, 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, 20*f8a1b7d9SAlexander Kabaev // USA. 21*f8a1b7d9SAlexander Kabaev 22*f8a1b7d9SAlexander Kabaev // As a special exception, you may use this file as part of a free software 23*f8a1b7d9SAlexander Kabaev // library without restriction. Specifically, if other files instantiate 24*f8a1b7d9SAlexander Kabaev // templates or use macros or inline functions from this file, or you compile 25*f8a1b7d9SAlexander Kabaev // this file and link it with other files to produce an executable, this 26*f8a1b7d9SAlexander Kabaev // file does not by itself cause the resulting executable to be covered by 27*f8a1b7d9SAlexander Kabaev // the GNU General Public License. This exception does not however 28*f8a1b7d9SAlexander Kabaev // invalidate any other reasons why the executable file might be covered by 29*f8a1b7d9SAlexander Kabaev // the GNU General Public License. 30*f8a1b7d9SAlexander Kabaev 31*f8a1b7d9SAlexander Kabaev /** @file debug/functions.h 32*f8a1b7d9SAlexander Kabaev * This file is a GNU debug extension to the Standard C++ Library. 33*f8a1b7d9SAlexander Kabaev */ 34*f8a1b7d9SAlexander Kabaev 35*f8a1b7d9SAlexander Kabaev #ifndef _GLIBCXX_DEBUG_FUNCTIONS_H 36*f8a1b7d9SAlexander Kabaev #define _GLIBCXX_DEBUG_FUNCTIONS_H 1 37*f8a1b7d9SAlexander Kabaev 38*f8a1b7d9SAlexander Kabaev #include <bits/c++config.h> 39*f8a1b7d9SAlexander Kabaev #include <stddef.h> // for ptrdiff_t 40*f8a1b7d9SAlexander Kabaev #include <bits/stl_iterator_base_types.h> // for iterator_traits, categories 41*f8a1b7d9SAlexander Kabaev #include <bits/cpp_type_traits.h> // for __is_integer 42*f8a1b7d9SAlexander Kabaev 43*f8a1b7d9SAlexander Kabaev namespace __gnu_debug 44*f8a1b7d9SAlexander Kabaev { 45*f8a1b7d9SAlexander Kabaev template<typename _Iterator, typename _Sequence> 46*f8a1b7d9SAlexander Kabaev class _Safe_iterator; 47*f8a1b7d9SAlexander Kabaev 48*f8a1b7d9SAlexander Kabaev // An arbitrary iterator pointer is not singular. 49*f8a1b7d9SAlexander Kabaev inline bool __check_singular_aux(const void *)50*f8a1b7d9SAlexander Kabaev __check_singular_aux(const void*) { return false; } 51*f8a1b7d9SAlexander Kabaev 52*f8a1b7d9SAlexander Kabaev // We may have an iterator that derives from _Safe_iterator_base but isn't 53*f8a1b7d9SAlexander Kabaev // a _Safe_iterator. 54*f8a1b7d9SAlexander Kabaev template<typename _Iterator> 55*f8a1b7d9SAlexander Kabaev inline bool __check_singular(_Iterator & __x)56*f8a1b7d9SAlexander Kabaev __check_singular(_Iterator& __x) 57*f8a1b7d9SAlexander Kabaev { return __check_singular_aux(&__x); } 58*f8a1b7d9SAlexander Kabaev 59*f8a1b7d9SAlexander Kabaev /** Non-NULL pointers are nonsingular. */ 60*f8a1b7d9SAlexander Kabaev template<typename _Tp> 61*f8a1b7d9SAlexander Kabaev inline bool __check_singular(const _Tp * __ptr)62*f8a1b7d9SAlexander Kabaev __check_singular(const _Tp* __ptr) 63*f8a1b7d9SAlexander Kabaev { return __ptr == 0; } 64*f8a1b7d9SAlexander Kabaev 65*f8a1b7d9SAlexander Kabaev /** Safe iterators know if they are singular. */ 66*f8a1b7d9SAlexander Kabaev template<typename _Iterator, typename _Sequence> 67*f8a1b7d9SAlexander Kabaev inline bool __check_singular(const _Safe_iterator<_Iterator,_Sequence> & __x)68*f8a1b7d9SAlexander Kabaev __check_singular(const _Safe_iterator<_Iterator, _Sequence>& __x) 69*f8a1b7d9SAlexander Kabaev { return __x._M_singular(); } 70*f8a1b7d9SAlexander Kabaev 71*f8a1b7d9SAlexander Kabaev /** Assume that some arbitrary iterator is dereferenceable, because we 72*f8a1b7d9SAlexander Kabaev can't prove that it isn't. */ 73*f8a1b7d9SAlexander Kabaev template<typename _Iterator> 74*f8a1b7d9SAlexander Kabaev inline bool __check_dereferenceable(_Iterator &)75*f8a1b7d9SAlexander Kabaev __check_dereferenceable(_Iterator&) 76*f8a1b7d9SAlexander Kabaev { return true; } 77*f8a1b7d9SAlexander Kabaev 78*f8a1b7d9SAlexander Kabaev /** Non-NULL pointers are dereferenceable. */ 79*f8a1b7d9SAlexander Kabaev template<typename _Tp> 80*f8a1b7d9SAlexander Kabaev inline bool __check_dereferenceable(const _Tp * __ptr)81*f8a1b7d9SAlexander Kabaev __check_dereferenceable(const _Tp* __ptr) 82*f8a1b7d9SAlexander Kabaev { return __ptr; } 83*f8a1b7d9SAlexander Kabaev 84*f8a1b7d9SAlexander Kabaev /** Safe iterators know if they are singular. */ 85*f8a1b7d9SAlexander Kabaev template<typename _Iterator, typename _Sequence> 86*f8a1b7d9SAlexander Kabaev inline bool __check_dereferenceable(const _Safe_iterator<_Iterator,_Sequence> & __x)87*f8a1b7d9SAlexander Kabaev __check_dereferenceable(const _Safe_iterator<_Iterator, _Sequence>& __x) 88*f8a1b7d9SAlexander Kabaev { return __x._M_dereferenceable(); } 89*f8a1b7d9SAlexander Kabaev 90*f8a1b7d9SAlexander Kabaev /** If the distance between two random access iterators is 91*f8a1b7d9SAlexander Kabaev * nonnegative, assume the range is valid. 92*f8a1b7d9SAlexander Kabaev */ 93*f8a1b7d9SAlexander Kabaev template<typename _RandomAccessIterator> 94*f8a1b7d9SAlexander Kabaev inline bool __valid_range_aux2(const _RandomAccessIterator & __first,const _RandomAccessIterator & __last,std::random_access_iterator_tag)95*f8a1b7d9SAlexander Kabaev __valid_range_aux2(const _RandomAccessIterator& __first, 96*f8a1b7d9SAlexander Kabaev const _RandomAccessIterator& __last, 97*f8a1b7d9SAlexander Kabaev std::random_access_iterator_tag) 98*f8a1b7d9SAlexander Kabaev { return __last - __first >= 0; } 99*f8a1b7d9SAlexander Kabaev 100*f8a1b7d9SAlexander Kabaev /** Can't test for a valid range with input iterators, because 101*f8a1b7d9SAlexander Kabaev * iteration may be destructive. So we just assume that the range 102*f8a1b7d9SAlexander Kabaev * is valid. 103*f8a1b7d9SAlexander Kabaev */ 104*f8a1b7d9SAlexander Kabaev template<typename _InputIterator> 105*f8a1b7d9SAlexander Kabaev inline bool __valid_range_aux2(const _InputIterator &,const _InputIterator &,std::input_iterator_tag)106*f8a1b7d9SAlexander Kabaev __valid_range_aux2(const _InputIterator&, const _InputIterator&, 107*f8a1b7d9SAlexander Kabaev std::input_iterator_tag) 108*f8a1b7d9SAlexander Kabaev { return true; } 109*f8a1b7d9SAlexander Kabaev 110*f8a1b7d9SAlexander Kabaev /** We say that integral types for a valid range, and defer to other 111*f8a1b7d9SAlexander Kabaev * routines to realize what to do with integral types instead of 112*f8a1b7d9SAlexander Kabaev * iterators. 113*f8a1b7d9SAlexander Kabaev */ 114*f8a1b7d9SAlexander Kabaev template<typename _Integral> 115*f8a1b7d9SAlexander Kabaev inline bool __valid_range_aux(const _Integral &,const _Integral &,std::__true_type)116*f8a1b7d9SAlexander Kabaev __valid_range_aux(const _Integral&, const _Integral&, std::__true_type) 117*f8a1b7d9SAlexander Kabaev { return true; } 118*f8a1b7d9SAlexander Kabaev 119*f8a1b7d9SAlexander Kabaev /** We have iterators, so figure out what kind of iterators that are 120*f8a1b7d9SAlexander Kabaev * to see if we can check the range ahead of time. 121*f8a1b7d9SAlexander Kabaev */ 122*f8a1b7d9SAlexander Kabaev template<typename _InputIterator> 123*f8a1b7d9SAlexander Kabaev inline bool __valid_range_aux(const _InputIterator & __first,const _InputIterator & __last,std::__false_type)124*f8a1b7d9SAlexander Kabaev __valid_range_aux(const _InputIterator& __first, 125*f8a1b7d9SAlexander Kabaev const _InputIterator& __last, std::__false_type) 126*f8a1b7d9SAlexander Kabaev { 127*f8a1b7d9SAlexander Kabaev typedef typename std::iterator_traits<_InputIterator>::iterator_category 128*f8a1b7d9SAlexander Kabaev _Category; 129*f8a1b7d9SAlexander Kabaev return __valid_range_aux2(__first, __last, _Category()); 130*f8a1b7d9SAlexander Kabaev } 131*f8a1b7d9SAlexander Kabaev 132*f8a1b7d9SAlexander Kabaev /** Don't know what these iterators are, or if they are even 133*f8a1b7d9SAlexander Kabaev * iterators (we may get an integral type for InputIterator), so 134*f8a1b7d9SAlexander Kabaev * see if they are integral and pass them on to the next phase 135*f8a1b7d9SAlexander Kabaev * otherwise. 136*f8a1b7d9SAlexander Kabaev */ 137*f8a1b7d9SAlexander Kabaev template<typename _InputIterator> 138*f8a1b7d9SAlexander Kabaev inline bool __valid_range(const _InputIterator & __first,const _InputIterator & __last)139*f8a1b7d9SAlexander Kabaev __valid_range(const _InputIterator& __first, const _InputIterator& __last) 140*f8a1b7d9SAlexander Kabaev { 141*f8a1b7d9SAlexander Kabaev typedef typename std::__is_integer<_InputIterator>::__type _Integral; 142*f8a1b7d9SAlexander Kabaev return __valid_range_aux(__first, __last, _Integral()); 143*f8a1b7d9SAlexander Kabaev } 144*f8a1b7d9SAlexander Kabaev 145*f8a1b7d9SAlexander Kabaev /** Safe iterators know how to check if they form a valid range. */ 146*f8a1b7d9SAlexander Kabaev template<typename _Iterator, typename _Sequence> 147*f8a1b7d9SAlexander Kabaev inline bool __valid_range(const _Safe_iterator<_Iterator,_Sequence> & __first,const _Safe_iterator<_Iterator,_Sequence> & __last)148*f8a1b7d9SAlexander Kabaev __valid_range(const _Safe_iterator<_Iterator, _Sequence>& __first, 149*f8a1b7d9SAlexander Kabaev const _Safe_iterator<_Iterator, _Sequence>& __last) 150*f8a1b7d9SAlexander Kabaev { return __first._M_valid_range(__last); } 151*f8a1b7d9SAlexander Kabaev 152*f8a1b7d9SAlexander Kabaev /* Checks that [first, last) is a valid range, and then returns 153*f8a1b7d9SAlexander Kabaev * __first. This routine is useful when we can't use a separate 154*f8a1b7d9SAlexander Kabaev * assertion statement because, e.g., we are in a constructor. 155*f8a1b7d9SAlexander Kabaev */ 156*f8a1b7d9SAlexander Kabaev template<typename _InputIterator> 157*f8a1b7d9SAlexander Kabaev inline _InputIterator __check_valid_range(const _InputIterator & __first,const _InputIterator & __last)158*f8a1b7d9SAlexander Kabaev __check_valid_range(const _InputIterator& __first, 159*f8a1b7d9SAlexander Kabaev const _InputIterator& __last 160*f8a1b7d9SAlexander Kabaev __attribute__((__unused__))) 161*f8a1b7d9SAlexander Kabaev { 162*f8a1b7d9SAlexander Kabaev _GLIBCXX_DEBUG_ASSERT(__valid_range(__first, __last)); 163*f8a1b7d9SAlexander Kabaev return __first; 164*f8a1b7d9SAlexander Kabaev } 165*f8a1b7d9SAlexander Kabaev 166*f8a1b7d9SAlexander Kabaev /** Checks that __s is non-NULL or __n == 0, and then returns __s. */ 167*f8a1b7d9SAlexander Kabaev template<typename _CharT, typename _Integer> 168*f8a1b7d9SAlexander Kabaev inline const _CharT* __check_string(const _CharT * __s,const _Integer & __n)169*f8a1b7d9SAlexander Kabaev __check_string(const _CharT* __s, 170*f8a1b7d9SAlexander Kabaev const _Integer& __n __attribute__((__unused__))) 171*f8a1b7d9SAlexander Kabaev { 172*f8a1b7d9SAlexander Kabaev #ifdef _GLIBCXX_DEBUG_PEDANTIC 173*f8a1b7d9SAlexander Kabaev _GLIBCXX_DEBUG_ASSERT(__s != 0 || __n == 0); 174*f8a1b7d9SAlexander Kabaev #endif 175*f8a1b7d9SAlexander Kabaev return __s; 176*f8a1b7d9SAlexander Kabaev } 177*f8a1b7d9SAlexander Kabaev 178*f8a1b7d9SAlexander Kabaev /** Checks that __s is non-NULL and then returns __s. */ 179*f8a1b7d9SAlexander Kabaev template<typename _CharT> 180*f8a1b7d9SAlexander Kabaev inline const _CharT* __check_string(const _CharT * __s)181*f8a1b7d9SAlexander Kabaev __check_string(const _CharT* __s) 182*f8a1b7d9SAlexander Kabaev { 183*f8a1b7d9SAlexander Kabaev #ifdef _GLIBCXX_DEBUG_PEDANTIC 184*f8a1b7d9SAlexander Kabaev _GLIBCXX_DEBUG_ASSERT(__s != 0); 185*f8a1b7d9SAlexander Kabaev #endif 186*f8a1b7d9SAlexander Kabaev return __s; 187*f8a1b7d9SAlexander Kabaev } 188*f8a1b7d9SAlexander Kabaev 189*f8a1b7d9SAlexander Kabaev // Can't check if an input iterator sequence is sorted, because we 190*f8a1b7d9SAlexander Kabaev // can't step through the sequence. 191*f8a1b7d9SAlexander Kabaev template<typename _InputIterator> 192*f8a1b7d9SAlexander Kabaev inline bool __check_sorted_aux(const _InputIterator &,const _InputIterator &,std::input_iterator_tag)193*f8a1b7d9SAlexander Kabaev __check_sorted_aux(const _InputIterator&, const _InputIterator&, 194*f8a1b7d9SAlexander Kabaev std::input_iterator_tag) 195*f8a1b7d9SAlexander Kabaev { return true; } 196*f8a1b7d9SAlexander Kabaev 197*f8a1b7d9SAlexander Kabaev // Can verify if a forward iterator sequence is in fact sorted using 198*f8a1b7d9SAlexander Kabaev // std::__is_sorted 199*f8a1b7d9SAlexander Kabaev template<typename _ForwardIterator> 200*f8a1b7d9SAlexander Kabaev inline bool __check_sorted_aux(_ForwardIterator __first,_ForwardIterator __last,std::forward_iterator_tag)201*f8a1b7d9SAlexander Kabaev __check_sorted_aux(_ForwardIterator __first, _ForwardIterator __last, 202*f8a1b7d9SAlexander Kabaev std::forward_iterator_tag) 203*f8a1b7d9SAlexander Kabaev { 204*f8a1b7d9SAlexander Kabaev if (__first == __last) 205*f8a1b7d9SAlexander Kabaev return true; 206*f8a1b7d9SAlexander Kabaev 207*f8a1b7d9SAlexander Kabaev _ForwardIterator __next = __first; 208*f8a1b7d9SAlexander Kabaev for (++__next; __next != __last; __first = __next, ++__next) { 209*f8a1b7d9SAlexander Kabaev if (*__next < *__first) 210*f8a1b7d9SAlexander Kabaev return false; 211*f8a1b7d9SAlexander Kabaev } 212*f8a1b7d9SAlexander Kabaev 213*f8a1b7d9SAlexander Kabaev return true; 214*f8a1b7d9SAlexander Kabaev } 215*f8a1b7d9SAlexander Kabaev 216*f8a1b7d9SAlexander Kabaev // Can't check if an input iterator sequence is sorted, because we can't step 217*f8a1b7d9SAlexander Kabaev // through the sequence. 218*f8a1b7d9SAlexander Kabaev template<typename _InputIterator, typename _Predicate> 219*f8a1b7d9SAlexander Kabaev inline bool __check_sorted_aux(const _InputIterator &,const _InputIterator &,_Predicate,std::input_iterator_tag)220*f8a1b7d9SAlexander Kabaev __check_sorted_aux(const _InputIterator&, const _InputIterator&, 221*f8a1b7d9SAlexander Kabaev _Predicate, std::input_iterator_tag) 222*f8a1b7d9SAlexander Kabaev { return true; } 223*f8a1b7d9SAlexander Kabaev 224*f8a1b7d9SAlexander Kabaev // Can verify if a forward iterator sequence is in fact sorted using 225*f8a1b7d9SAlexander Kabaev // std::__is_sorted 226*f8a1b7d9SAlexander Kabaev template<typename _ForwardIterator, typename _Predicate> 227*f8a1b7d9SAlexander Kabaev inline bool __check_sorted_aux(_ForwardIterator __first,_ForwardIterator __last,_Predicate __pred,std::forward_iterator_tag)228*f8a1b7d9SAlexander Kabaev __check_sorted_aux(_ForwardIterator __first, _ForwardIterator __last, 229*f8a1b7d9SAlexander Kabaev _Predicate __pred, std::forward_iterator_tag) 230*f8a1b7d9SAlexander Kabaev { 231*f8a1b7d9SAlexander Kabaev if (__first == __last) 232*f8a1b7d9SAlexander Kabaev return true; 233*f8a1b7d9SAlexander Kabaev 234*f8a1b7d9SAlexander Kabaev _ForwardIterator __next = __first; 235*f8a1b7d9SAlexander Kabaev for (++__next; __next != __last; __first = __next, ++__next) { 236*f8a1b7d9SAlexander Kabaev if (__pred(*__next, *__first)) 237*f8a1b7d9SAlexander Kabaev return false; 238*f8a1b7d9SAlexander Kabaev } 239*f8a1b7d9SAlexander Kabaev 240*f8a1b7d9SAlexander Kabaev return true; 241*f8a1b7d9SAlexander Kabaev } 242*f8a1b7d9SAlexander Kabaev 243*f8a1b7d9SAlexander Kabaev // Determine if a sequence is sorted. 244*f8a1b7d9SAlexander Kabaev template<typename _InputIterator> 245*f8a1b7d9SAlexander Kabaev inline bool __check_sorted(const _InputIterator & __first,const _InputIterator & __last)246*f8a1b7d9SAlexander Kabaev __check_sorted(const _InputIterator& __first, const _InputIterator& __last) 247*f8a1b7d9SAlexander Kabaev { 248*f8a1b7d9SAlexander Kabaev typedef typename std::iterator_traits<_InputIterator>::iterator_category 249*f8a1b7d9SAlexander Kabaev _Category; 250*f8a1b7d9SAlexander Kabaev return __check_sorted_aux(__first, __last, _Category()); 251*f8a1b7d9SAlexander Kabaev } 252*f8a1b7d9SAlexander Kabaev 253*f8a1b7d9SAlexander Kabaev template<typename _InputIterator, typename _Predicate> 254*f8a1b7d9SAlexander Kabaev inline bool __check_sorted(const _InputIterator & __first,const _InputIterator & __last,_Predicate __pred)255*f8a1b7d9SAlexander Kabaev __check_sorted(const _InputIterator& __first, const _InputIterator& __last, 256*f8a1b7d9SAlexander Kabaev _Predicate __pred) 257*f8a1b7d9SAlexander Kabaev { 258*f8a1b7d9SAlexander Kabaev typedef typename std::iterator_traits<_InputIterator>::iterator_category 259*f8a1b7d9SAlexander Kabaev _Category; 260*f8a1b7d9SAlexander Kabaev return __check_sorted_aux(__first, __last, __pred, 261*f8a1b7d9SAlexander Kabaev _Category()); 262*f8a1b7d9SAlexander Kabaev } 263*f8a1b7d9SAlexander Kabaev 264*f8a1b7d9SAlexander Kabaev // _GLIBCXX_RESOLVE_LIB_DEFECTS 265*f8a1b7d9SAlexander Kabaev // 270. Binary search requirements overly strict 266*f8a1b7d9SAlexander Kabaev // Determine if a sequence is partitioned w.r.t. this element. 267*f8a1b7d9SAlexander Kabaev template<typename _ForwardIterator, typename _Tp> 268*f8a1b7d9SAlexander Kabaev inline bool __check_partitioned(_ForwardIterator __first,_ForwardIterator __last,const _Tp & __value)269*f8a1b7d9SAlexander Kabaev __check_partitioned(_ForwardIterator __first, _ForwardIterator __last, 270*f8a1b7d9SAlexander Kabaev const _Tp& __value) 271*f8a1b7d9SAlexander Kabaev { 272*f8a1b7d9SAlexander Kabaev while (__first != __last && *__first < __value) 273*f8a1b7d9SAlexander Kabaev ++__first; 274*f8a1b7d9SAlexander Kabaev while (__first != __last && !(*__first < __value)) 275*f8a1b7d9SAlexander Kabaev ++__first; 276*f8a1b7d9SAlexander Kabaev return __first == __last; 277*f8a1b7d9SAlexander Kabaev } 278*f8a1b7d9SAlexander Kabaev 279*f8a1b7d9SAlexander Kabaev // Determine if a sequence is partitioned w.r.t. this element. 280*f8a1b7d9SAlexander Kabaev template<typename _ForwardIterator, typename _Tp, typename _Pred> 281*f8a1b7d9SAlexander Kabaev inline bool __check_partitioned(_ForwardIterator __first,_ForwardIterator __last,const _Tp & __value,_Pred __pred)282*f8a1b7d9SAlexander Kabaev __check_partitioned(_ForwardIterator __first, _ForwardIterator __last, 283*f8a1b7d9SAlexander Kabaev const _Tp& __value, _Pred __pred) 284*f8a1b7d9SAlexander Kabaev { 285*f8a1b7d9SAlexander Kabaev while (__first != __last && __pred(*__first, __value)) 286*f8a1b7d9SAlexander Kabaev ++__first; 287*f8a1b7d9SAlexander Kabaev while (__first != __last && !__pred(*__first, __value)) 288*f8a1b7d9SAlexander Kabaev ++__first; 289*f8a1b7d9SAlexander Kabaev return __first == __last; 290*f8a1b7d9SAlexander Kabaev } 291*f8a1b7d9SAlexander Kabaev } // namespace __gnu_debug 292*f8a1b7d9SAlexander Kabaev 293*f8a1b7d9SAlexander Kabaev #endif 294