32#ifndef _GLIBCXX_REGEX_STATE_LIMIT
33#define _GLIBCXX_REGEX_STATE_LIMIT 100000
36namespace std _GLIBCXX_VISIBILITY(default)
38_GLIBCXX_BEGIN_NAMESPACE_VERSION
48 typedef long _StateIdT;
49 _GLIBCXX17_INLINE
constexpr _StateIdT _S_invalid_state_id = -1;
51 template<
typename _CharT>
52 using _Matcher = std::function<bool (_CharT)>;
59 _S_opcode_alternative,
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,
83 size_t _M_backref_index;
94 __gnu_cxx::__aligned_membuf<_Matcher<char>> _M_matcher_storage;
98 explicit _State_base(
_Opcode __opcode) noexcept
99 : _M_opcode(__opcode), _M_next(_S_invalid_state_id)
104 _M_has_alt() const noexcept
106 return _M_opcode == _S_opcode_alternative
107 || _M_opcode == _S_opcode_repeat
108 || _M_opcode == _S_opcode_subexpr_lookahead;
121 template<
typename _Char_type>
122 struct _State : _State_base
124 typedef _Matcher<_Char_type> _MatcherT;
125 static_assert(
sizeof(_MatcherT) ==
sizeof(_Matcher<char>),
126 "std::function<bool(T)> has the same size as "
127 "std::function<bool(char)>");
128 static_assert(
alignof(_MatcherT) ==
alignof(_Matcher<char>),
129 "std::function<bool(T)> has the same alignment as "
130 "std::function<bool(char)>");
133 _State(
_Opcode __opcode) noexcept : _State_base(__opcode)
135 if (_M_opcode() == _S_opcode_match)
136 new (this->_M_matcher_storage._M_addr()) _MatcherT();
139 _State(
const _State& __rhs) : _State_base(__rhs)
141 if (__rhs._M_opcode() == _S_opcode_match)
142 new (this->_M_matcher_storage._M_addr())
143 _MatcherT(__rhs._M_get_matcher());
146 _State(_State&& __rhs) noexcept : _State_base(__rhs)
148 if (__rhs._M_opcode() == _S_opcode_match)
149 new (this->_M_matcher_storage._M_addr())
150 _MatcherT(
std::move(__rhs._M_get_matcher()));
154 operator=(
const _State&) =
delete;
158 if (_M_opcode() == _S_opcode_match)
159 _M_get_matcher().~_MatcherT();
165 _M_opcode() const noexcept
166 {
return _State_base::_M_opcode; }
169 _M_matches(_Char_type __char)
const
170 {
return _M_get_matcher()(__char); }
173 _M_get_matcher() noexcept
174 {
return *
static_cast<_MatcherT*
>(this->_M_matcher_storage._M_addr()); }
177 _M_get_matcher() const noexcept
179 return *
static_cast<const _MatcherT*
>(
180 this->_M_matcher_storage._M_addr());
189 _NFA_base(_FlagT __f) noexcept
190 : _M_flags(__f), _M_start_state(0), _M_subexpr_count(0),
191 _M_has_backref(
false)
194 _NFA_base(_NFA_base&&) =
default;
197 ~_NFA_base() =
default;
201 _M_options() const noexcept
205 _M_start() const noexcept
206 {
return _M_start_state; }
209 _M_sub_count() const noexcept
210 {
return _M_subexpr_count; }
212 _GLIBCXX_STD_C::vector<size_t> _M_paren_stack;
214 _StateIdT _M_start_state;
215 size_t _M_subexpr_count;
219 template<
typename _TraitsT>
221 : _NFA_base, _GLIBCXX_STD_C::vector<_State<typename _TraitsT::char_type>>
223 typedef typename _TraitsT::char_type _Char_type;
224 typedef _State<_Char_type> _StateT;
225 typedef _Matcher<_Char_type> _MatcherT;
227 _NFA(
const typename _TraitsT::locale_type& __loc, _FlagT __flags)
229 { _M_traits.imbue(__loc); }
232 _NFA(
const _NFA&) =
delete;
233 _NFA(_NFA&&) =
default;
238 auto __ret = _M_insert_state(_StateT(_S_opcode_accept));
243 _M_insert_alt(_StateIdT __next, _StateIdT __alt,
244 bool __neg __attribute__((__unused__)))
246 _StateT __tmp(_S_opcode_alternative);
249 __tmp._M_next = __next;
250 __tmp._M_alt = __alt;
251 return _M_insert_state(
std::move(__tmp));
255 _M_insert_repeat(_StateIdT __next, _StateIdT __alt,
bool __neg)
257 _StateT __tmp(_S_opcode_repeat);
260 __tmp._M_next = __next;
261 __tmp._M_alt = __alt;
262 __tmp._M_neg = __neg;
263 return _M_insert_state(
std::move(__tmp));
267 _M_insert_matcher(_MatcherT __m)
269 _StateT __tmp(_S_opcode_match);
271 return _M_insert_state(
std::move(__tmp));
275 _M_insert_subexpr_begin()
277 auto __id = this->_M_subexpr_count++;
278 this->_M_paren_stack.push_back(__id);
279 _StateT __tmp(_S_opcode_subexpr_begin);
280 __tmp._M_subexpr = __id;
281 return _M_insert_state(
std::move(__tmp));
285 _M_insert_subexpr_end()
287 _StateT __tmp(_S_opcode_subexpr_end);
288 __tmp._M_subexpr = this->_M_paren_stack.back();
289 this->_M_paren_stack.pop_back();
290 return _M_insert_state(
std::move(__tmp));
294 _M_insert_backref(
size_t __index);
297 _M_insert_line_begin()
298 {
return _M_insert_state(_StateT(_S_opcode_line_begin_assertion)); }
302 {
return _M_insert_state(_StateT(_S_opcode_line_end_assertion)); }
305 _M_insert_word_bound(
bool __neg)
307 _StateT __tmp(_S_opcode_word_boundary);
308 __tmp._M_neg = __neg;
309 return _M_insert_state(
std::move(__tmp));
313 _M_insert_lookahead(_StateIdT __alt,
bool __neg)
315 _StateT __tmp(_S_opcode_subexpr_lookahead);
316 __tmp._M_alt = __alt;
317 __tmp._M_neg = __neg;
318 return _M_insert_state(
std::move(__tmp));
323 {
return _M_insert_state(_StateT(_S_opcode_dummy)); }
326 _M_insert_state(_StateT __s)
329 if (this->
size() > _GLIBCXX_REGEX_STATE_LIMIT)
332 "Number of NFA states exceeds limit. Please use shorter regex "
333 "string, or use smaller brace expression, or make "
334 "_GLIBCXX_REGEX_STATE_LIMIT larger.");
335 return this->
size() - 1;
340 _M_eliminate_dummy();
353 template<
typename _TraitsT>
357 typedef _NFA<_TraitsT> _RegexT;
361 : _M_nfa(__nfa), _M_start(__s), _M_end(__s)
364 _StateSeq(_RegexT& __nfa, _StateIdT __s, _StateIdT __end)
365 : _M_nfa(__nfa), _M_start(__s), _M_end(__end)
370 _M_append(_StateIdT __id)
372 _M_nfa[_M_end]._M_next = __id;
380 _M_nfa[_M_end]._M_next = __s._M_start;
397_GLIBCXX_END_NAMESPACE_VERSION
constexpr std::remove_reference< _Tp >::type && move(_Tp &&__t) noexcept
Convert a value to an rvalue.
_Opcode
Operation codes that define the type of transitions within the base NFA that represents the regular e...
ISO C++ entities toplevel namespace is std.
constexpr error_type error_space(_S_error_space)
syntax_option_type
This is a bitmask type indicating how to interpret the regex.
Describes a sequence of one or more _State, its current start and end(s). This structure contains fra...
constexpr void push_back(const value_type &__x)
Add data to the end of the vector.
constexpr size_type size() const noexcept