File Coverage

/usr/include/c++/5/bits/regex_automaton.h
Criterion Covered Total %
statement 0 80 0.0
branch 0 42 0.0
condition n/a
subroutine n/a
pod n/a
total 0 122 0.0


line stmt bran cond sub pod time code
1             // class template regex -*- C++ -*-
2              
3             // Copyright (C) 2013-2015 Free Software Foundation, Inc.
4             //
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)
9             // any later version.
10              
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.
15              
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.
19              
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             // .
24              
25             /**
26             * @file bits/regex_automaton.h
27             * This is an internal header file, included by other library headers.
28             * Do not attempt to use it directly. @headername{regex}
29             */
30              
31             // This macro defines the maximal state number a NFA can have.
32             #ifndef _GLIBCXX_REGEX_STATE_LIMIT
33             #define _GLIBCXX_REGEX_STATE_LIMIT 100000
34             #endif
35              
36             namespace std _GLIBCXX_VISIBILITY(default)
37             {
38             namespace __detail
39             {
40             _GLIBCXX_BEGIN_NAMESPACE_VERSION
41              
42             /**
43             * @defgroup regex-detail Base and Implementation Classes
44             * @ingroup regex
45             * @{
46             */
47              
48             typedef long _StateIdT;
49             static const _StateIdT _S_invalid_state_id = -1;
50              
51             template
52             using _Matcher = std::function;
53              
54             /// Operation codes that define the type of transitions within the base NFA
55             /// that represents the regular expression.
56             enum _Opcode : int
57             {
58             _S_opcode_unknown,
59             _S_opcode_alternative,
60             _S_opcode_repeat,
61             _S_opcode_backref,
62             _S_opcode_line_begin_assertion,
63             _S_opcode_line_end_assertion,
64             _S_opcode_word_boundary,
65             _S_opcode_subexpr_lookahead,
66             _S_opcode_subexpr_begin,
67             _S_opcode_subexpr_end,
68             _S_opcode_dummy,
69             _S_opcode_match,
70             _S_opcode_accept,
71             };
72              
73             struct _State_base
74             {
75             _Opcode _M_opcode; // type of outgoing transition
76             _StateIdT _M_next; // outgoing transition
77             union // Since they are mutually exclusive.
78             {
79             size_t _M_subexpr; // for _S_opcode_subexpr_*
80             size_t _M_backref_index; // for _S_opcode_backref
81             struct
82             {
83             // for _S_opcode_alternative, _S_opcode_repeat and
84             // _S_opcode_subexpr_lookahead
85             _StateIdT _M_alt;
86             // for _S_opcode_word_boundary or _S_opcode_subexpr_lookahead or
87             // quantifiers (ungreedy if set true)
88             bool _M_neg;
89             };
90             };
91              
92 0           explicit _State_base(_Opcode __opcode)
93 0           : _M_opcode(__opcode), _M_next(_S_invalid_state_id)
94 0           { }
95              
96             protected:
97             ~_State_base() = default;
98              
99             public:
100             #ifdef _GLIBCXX_DEBUG
101             std::ostream&
102             _M_print(std::ostream& ostr) const;
103              
104             // Prints graphviz dot commands for state.
105             std::ostream&
106             _M_dot(std::ostream& __ostr, _StateIdT __id) const;
107             #endif
108             };
109              
110             template
111 0           struct _State : _State_base
112             {
113             typedef _Matcher _MatcherT;
114              
115             _MatcherT _M_matches; // for _S_opcode_match
116              
117 0           explicit _State(_Opcode __opcode) : _State_base(__opcode) { }
118             };
119              
120             struct _NFA_base
121             {
122             typedef size_t _SizeT;
123             typedef regex_constants::syntax_option_type _FlagT;
124              
125             explicit
126 0           _NFA_base(_FlagT __f)
127             : _M_flags(__f), _M_start_state(0), _M_subexpr_count(0),
128 0           _M_has_backref(false)
129 0           { }
130              
131             _NFA_base(_NFA_base&&) = default;
132              
133             protected:
134 0           ~_NFA_base() = default;
135              
136             public:
137             _FlagT
138             _M_options() const
139             { return _M_flags; }
140              
141             _StateIdT
142 0           _M_start() const
143 0           { return _M_start_state; }
144              
145             _SizeT
146 0           _M_sub_count() const
147 0           { return _M_subexpr_count; }
148              
149             std::vector _M_paren_stack;
150             _FlagT _M_flags;
151             _StateIdT _M_start_state;
152             _SizeT _M_subexpr_count;
153             bool _M_has_backref;
154             };
155              
156             template
157 0           struct _NFA
158             : _NFA_base, std::vector<_State<_TraitsT>>
159             {
160             typedef _State<_TraitsT> _StateT;
161             typedef _Matcher _MatcherT;
162              
163 0           _NFA(const typename _TraitsT::locale_type& __loc, _FlagT __flags)
164 0           : _NFA_base(__flags)
165 0           { _M_traits.imbue(__loc); }
166              
167             // for performance reasons _NFA objects should only be moved not copied
168             _NFA(const _NFA&) = delete;
169             _NFA(_NFA&&) = default;
170              
171             _StateIdT
172 0           _M_insert_accept()
173             {
174 0 0         auto __ret = _M_insert_state(_StateT(_S_opcode_accept));
175 0           return __ret;
176             }
177              
178             _StateIdT
179 0           _M_insert_alt(_StateIdT __next, _StateIdT __alt, bool __neg)
180             {
181 0           _StateT __tmp(_S_opcode_alternative);
182             // It labels every quantifier to make greedy comparison easier in BFS
183             // approach.
184 0           __tmp._M_next = __next;
185 0           __tmp._M_alt = __alt;
186 0 0         return _M_insert_state(std::move(__tmp));
    0          
187             }
188              
189             _StateIdT
190 0           _M_insert_repeat(_StateIdT __next, _StateIdT __alt, bool __neg)
191             {
192 0           _StateT __tmp(_S_opcode_repeat);
193             // It labels every quantifier to make greedy comparison easier in BFS
194             // approach.
195 0           __tmp._M_next = __next;
196 0           __tmp._M_alt = __alt;
197 0           __tmp._M_neg = __neg;
198 0 0         return _M_insert_state(std::move(__tmp));
    0          
199             }
200              
201             _StateIdT
202 0           _M_insert_matcher(_MatcherT __m)
203             {
204 0           _StateT __tmp(_S_opcode_match);
205 0 0         __tmp._M_matches = std::move(__m);
206 0 0         return _M_insert_state(std::move(__tmp));
    0          
207             }
208              
209             _StateIdT
210 0           _M_insert_subexpr_begin()
211             {
212 0           auto __id = this->_M_subexpr_count++;
213 0 0         this->_M_paren_stack.push_back(__id);
214 0           _StateT __tmp(_S_opcode_subexpr_begin);
215 0           __tmp._M_subexpr = __id;
216 0 0         return _M_insert_state(std::move(__tmp));
    0          
217             }
218              
219             _StateIdT
220 0           _M_insert_subexpr_end()
221             {
222 0           _StateT __tmp(_S_opcode_subexpr_end);
223 0           __tmp._M_subexpr = this->_M_paren_stack.back();
224 0           this->_M_paren_stack.pop_back();
225 0 0         return _M_insert_state(std::move(__tmp));
    0          
226             }
227              
228             _StateIdT
229             _M_insert_backref(size_t __index);
230              
231             _StateIdT
232 0           _M_insert_line_begin()
233 0 0         { return _M_insert_state(_StateT(_S_opcode_line_begin_assertion)); }
234              
235             _StateIdT
236 0           _M_insert_line_end()
237 0 0         { return _M_insert_state(_StateT(_S_opcode_line_end_assertion)); }
238              
239             _StateIdT
240 0           _M_insert_word_bound(bool __neg)
241             {
242 0           _StateT __tmp(_S_opcode_word_boundary);
243 0           __tmp._M_neg = __neg;
244 0 0         return _M_insert_state(std::move(__tmp));
    0          
245             }
246              
247             _StateIdT
248 0           _M_insert_lookahead(_StateIdT __alt, bool __neg)
249             {
250 0           _StateT __tmp(_S_opcode_subexpr_lookahead);
251 0           __tmp._M_alt = __alt;
252 0           __tmp._M_neg = __neg;
253 0 0         return _M_insert_state(std::move(__tmp));
    0          
254             }
255              
256             _StateIdT
257 0           _M_insert_dummy()
258 0 0         { return _M_insert_state(_StateT(_S_opcode_dummy)); }
259              
260             _StateIdT
261 0           _M_insert_state(_StateT __s)
262             {
263 0           this->push_back(std::move(__s));
264 0 0         if (this->size() > _GLIBCXX_REGEX_STATE_LIMIT)
265 0           __throw_regex_error(regex_constants::error_space);
266 0           return this->size()-1;
267             }
268              
269             // Eliminate dummy node in this NFA to make it compact.
270             void
271             _M_eliminate_dummy();
272              
273             #ifdef _GLIBCXX_DEBUG
274             std::ostream&
275             _M_dot(std::ostream& __ostr) const;
276             #endif
277             public:
278             _TraitsT _M_traits;
279             };
280              
281             /// Describes a sequence of one or more %_State, its current start
282             /// and end(s). This structure contains fragments of an NFA during
283             /// construction.
284             template
285             class _StateSeq
286             {
287             public:
288             typedef _NFA<_TraitsT> _RegexT;
289              
290             public:
291 0           _StateSeq(_RegexT& __nfa, _StateIdT __s)
292 0           : _M_nfa(__nfa), _M_start(__s), _M_end(__s)
293 0           { }
294              
295 0           _StateSeq(_RegexT& __nfa, _StateIdT __s, _StateIdT __end)
296 0           : _M_nfa(__nfa), _M_start(__s), _M_end(__end)
297 0           { }
298              
299             // Append a state on *this and change *this to the new sequence.
300             void
301 0           _M_append(_StateIdT __id)
302             {
303 0           _M_nfa[_M_end]._M_next = __id;
304 0           _M_end = __id;
305 0           }
306              
307             // Append a sequence on *this and change *this to the new sequence.
308             void
309 0           _M_append(const _StateSeq& __s)
310             {
311 0           _M_nfa[_M_end]._M_next = __s._M_start;
312 0           _M_end = __s._M_end;
313 0           }
314              
315             // Clones an entire sequence.
316             _StateSeq
317             _M_clone();
318              
319             public:
320             _RegexT& _M_nfa;
321             _StateIdT _M_start;
322             _StateIdT _M_end;
323             };
324              
325             //@} regex-detail
326             _GLIBCXX_END_NAMESPACE_VERSION
327             } // namespace __detail
328             } // namespace std
329              
330             #include