1 // Debugging support implementation -*- C++ -*-
3 // Copyright (C) 2015-2021 Free Software Foundation, Inc.
5 // This file is part of the GNU ISO C++ Library. This library is free
6 // software; you can redistribute it and/or modify it under the
7 // terms of the GNU General Public License as published by the
8 // Free Software Foundation; either version 3, or (at your option)
11 // This library is distributed in the hope that it will be useful,
12 // but WITHOUT ANY WARRANTY; without even the implied warranty of
13 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 // GNU General Public License for more details.
16 // Under Section 7 of GPL version 3, you are granted additional
17 // permissions described in the GCC Runtime Library Exception, version
18 // 3.1, as published by the Free Software Foundation.
20 // You should have received a copy of the GNU General Public License and
21 // a copy of the GCC Runtime Library Exception along with this program;
22 // see the files COPYING3 and COPYING.RUNTIME respectively. If not, see
23 // <http://www.gnu.org/licenses/>.
25 /** @file debug/stl_iterator.h
26 * This file is a GNU debug extension to the Standard C++ Library.
29 #ifndef _GLIBCXX_DEBUG_STL_ITERATOR_H
30 #define _GLIBCXX_DEBUG_STL_ITERATOR_H 1
32 #include <debug/helper_functions.h>
36 // Help Debug mode to see through reverse_iterator.
37 template<typename _Iterator
>
39 __valid_range(const std::reverse_iterator
<_Iterator
>& __first
,
40 const std::reverse_iterator
<_Iterator
>& __last
,
41 typename _Distance_traits
<_Iterator
>::__type
& __dist
)
42 { return __valid_range(__last
.base(), __first
.base(), __dist
); }
44 template<typename _Iterator
>
45 inline typename _Distance_traits
<_Iterator
>::__type
46 __get_distance(const std::reverse_iterator
<_Iterator
>& __first
,
47 const std::reverse_iterator
<_Iterator
>& __last
)
48 { return __get_distance(__last
.base(), __first
.base()); }
50 template<typename _Iterator
, typename _Size
>
52 __can_advance(const std::reverse_iterator
<_Iterator
>& __it
, _Size __n
)
53 { return __can_advance(__it
.base(), -__n
); }
55 template<typename _Iterator
, typename _Diff
>
57 __can_advance(const std::reverse_iterator
<_Iterator
>& __it
,
58 const std::pair
<_Diff
, _Distance_precision
>& __dist
,
60 { return __can_advance(__it
.base(), __dist
, -__way
); }
62 template<typename _Iterator
, typename _Sequence
>
63 inline std::reverse_iterator
<_Iterator
>
64 __base(const std::reverse_iterator
<_Safe_iterator
<
65 _Iterator
, _Sequence
, std::random_access_iterator_tag
> >& __it
)
66 { return std::reverse_iterator
<_Iterator
>(__it
.base().base()); }
68 #if __cplusplus < 201103L
69 template<typename _Iterator
>
70 struct _Unsafe_type
<std::reverse_iterator
<_Iterator
> >
72 typedef typename _Unsafe_type
<_Iterator
>::_Type _UnsafeType
;
73 typedef std::reverse_iterator
<_UnsafeType
> _Type
;
76 template<typename _Iterator
>
77 inline std::reverse_iterator
<typename _Unsafe_type
<_Iterator
>::_Type
>
78 __unsafe(const std::reverse_iterator
<_Iterator
>& __it
)
80 typedef typename _Unsafe_type
<_Iterator
>::_Type _UnsafeType
;
81 return std::reverse_iterator
<_UnsafeType
>(__unsafe(__it
.base()));
84 template<typename _Iterator
>
86 __unsafe(const std::reverse_iterator
<_Iterator
>& __it
)
87 -> decltype(std::__make_reverse_iterator(__unsafe(__it
.base())))
88 { return std::__make_reverse_iterator(__unsafe(__it
.base())); }
91 #if __cplusplus >= 201103L
92 // Help Debug mode to see through move_iterator.
93 template<typename _Iterator
>
95 __valid_range(const std::move_iterator
<_Iterator
>& __first
,
96 const std::move_iterator
<_Iterator
>& __last
,
97 typename _Distance_traits
<_Iterator
>::__type
& __dist
)
98 { return __valid_range(__first
.base(), __last
.base(), __dist
); }
100 template<typename _Iterator
>
101 inline typename _Distance_traits
<_Iterator
>::__type
102 __get_distance(const std::move_iterator
<_Iterator
>& __first
,
103 const std::move_iterator
<_Iterator
>& __last
)
104 { return __get_distance(__first
.base(), __last
.base()); }
106 template<typename _Iterator
, typename _Size
>
108 __can_advance(const std::move_iterator
<_Iterator
>& __it
, _Size __n
)
109 { return __can_advance(__it
.base(), __n
); }
111 template<typename _Iterator
, typename _Diff
>
113 __can_advance(const std::move_iterator
<_Iterator
>& __it
,
114 const std::pair
<_Diff
, _Distance_precision
>& __dist
,
116 { return __can_advance(__it
.base(), __dist
, __way
); }
118 template<typename _Iterator
>
120 __unsafe(const std::move_iterator
<_Iterator
>& __it
)
121 -> decltype(std::make_move_iterator(__unsafe(__it
.base())))
122 { return std::make_move_iterator(__unsafe(__it
.base())); }
124 template<typename _Iterator
>
126 __base(const std::move_iterator
<_Iterator
>& __it
)
127 -> decltype(std::make_move_iterator(__base(__it
.base())))
128 { return std::make_move_iterator(__base(__it
.base())); }