regex_executor.tcc 11.4 KB
Newer Older
1 2
// class template regex -*- C++ -*-

3
// Copyright (C) 2013-2014 Free Software Foundation, Inc.
4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36
//
// This file is part of the GNU ISO C++ Library.  This library is free
// software; you can redistribute it and/or modify it under the
// terms of the GNU General Public License as published by the
// Free Software Foundation; either version 3, or (at your option)
// any later version.

// This library is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
// GNU General Public License for more details.

// Under Section 7 of GPL version 3, you are granted additional
// permissions described in the GCC Runtime Library Exception, version
// 3.1, as published by the Free Software Foundation.

// You should have received a copy of the GNU General Public License and
// a copy of the GCC Runtime Library Exception along with this program;
// see the files COPYING3 and COPYING.RUNTIME respectively.  If not, see
// <http://www.gnu.org/licenses/>.

/**
 *  @file bits/regex_executor.tcc
 *  This is an internal header file, included by other library headers.
 *  Do not attempt to use it directly. @headername{regex}
 */

namespace std _GLIBCXX_VISIBILITY(default)
{
namespace __detail
{
_GLIBCXX_BEGIN_NAMESPACE_VERSION

37 38 39
  template<typename _BiIter, typename _Alloc, typename _TraitsT,
    bool __dfs_mode>
    bool _Executor<_BiIter, _Alloc, _TraitsT, __dfs_mode>::
40 41 42 43 44 45 46
    _M_search()
    {
      if (_M_flags & regex_constants::match_continuous)
	return _M_search_from_first();
      auto __cur = _M_begin;
      do
	{
47 48
	  _M_current = __cur;
	  if (_M_main<false>())
49 50 51 52 53 54 55
	    return true;
	}
      // Continue when __cur == _M_end
      while (__cur++ != _M_end);
      return false;
    }

56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98
  // This function operates in different modes, DFS mode or BFS mode, indicated
  // by template parameter __dfs_mode. See _M_main for details.
  //
  // ------------------------------------------------------------
  //
  // DFS mode:
  //
  // It applies a Depth-First-Search (aka backtracking) on given NFA and input
  // string.
  // At the very beginning the executor stands in the start state, then it tries
  // every possible state transition in current state recursively. Some state
  // transitions consume input string, say, a single-char-matcher or a
  // back-reference matcher; some don't, like assertion or other anchor nodes.
  // When the input is exhausted and/or the current state is an accepting state,
  // the whole executor returns true.
  //
  // TODO: This approach is exponentially slow for certain input.
  //       Try to compile the NFA to a DFA.
  //
  // Time complexity: o(match_length), O(2^(_M_nfa.size()))
  // Space complexity: \theta(match_results.size() + match_length)
  //
  // ------------------------------------------------------------
  //
  // BFS mode:
  //
  // Russ Cox's article (http://swtch.com/~rsc/regexp/regexp1.html)
  // explained this algorithm clearly.
  //
  // It first computes epsilon closure for every state that's still matching,
  // using the same DFS algorithm, but doesn't reenter states (set true in
  // _M_visited), nor follows _S_opcode_match.
  //
  // Then apply DFS using every _S_opcode_match (in _M_match_queue) as the start
  // state.
  //
  // It significantly reduces potential duplicate states, so has a better
  // upper bound; but it requires more overhead.
  //
  // Time complexity: o(match_length * match_results.size())
  //                  O(match_length * _M_nfa.size() * match_results.size())
  // Space complexity: o(_M_nfa.size() + match_results.size())
  //                   O(_M_nfa.size() * match_results.size())
99 100 101 102 103 104 105 106 107 108 109 110 111 112 113
  template<typename _BiIter, typename _Alloc, typename _TraitsT,
    bool __dfs_mode>
  template<bool __match_mode>
    bool _Executor<_BiIter, _Alloc, _TraitsT, __dfs_mode>::
    _M_main()
    {
      if (__dfs_mode)
	{
	  _M_has_sol = false;
	  _M_cur_results = _M_results;
	  _M_dfs<__match_mode>(_M_start_state);
	  return _M_has_sol;
	}
      else
	{
114
	  _M_match_queue->push_back(make_pair(_M_start_state, _M_results));
115 116 117 118 119 120 121 122
	  bool __ret = false;
	  while (1)
	    {
	      _M_has_sol = false;
	      if (_M_match_queue->empty())
		break;
	      _M_visited->assign(_M_visited->size(), false);
	      auto _M_old_queue = std::move(*_M_match_queue);
123
	      for (auto __task : _M_old_queue)
124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143
		{
		  _M_cur_results = __task.second;
		  _M_dfs<__match_mode>(__task.first);
		}
	      if (!__match_mode)
		__ret |= _M_has_sol;
	      if (_M_current == _M_end)
		break;
	      ++_M_current;
	    }
	  if (__match_mode)
	    __ret = _M_has_sol;
	  return __ret;
	}
    }

  // Return whether now match the given sub-NFA.
  template<typename _BiIter, typename _Alloc, typename _TraitsT,
    bool __dfs_mode>
    bool _Executor<_BiIter, _Alloc, _TraitsT, __dfs_mode>::
144
    _M_lookahead(_State<_TraitsT> __state)
145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162
    {
      _ResultsVec __what(_M_cur_results.size());
      auto __sub = std::unique_ptr<_Executor>(new _Executor(_M_current,
							    _M_end,
							    __what,
							    _M_re,
							    _M_flags));
      __sub->_M_start_state = __state._M_alt;
      if (__sub->_M_search_from_first())
	{
	  for (size_t __i = 0; __i < __what.size(); __i++)
	    if (__what[__i].matched)
	      _M_cur_results[__i] = __what[__i];
	  return true;
	}
      return false;
    }

163
  // TODO: Use a function vector to dispatch, instead of using switch-case.
164 165 166 167
  template<typename _BiIter, typename _Alloc, typename _TraitsT,
    bool __dfs_mode>
  template<bool __match_mode>
    void _Executor<_BiIter, _Alloc, _TraitsT, __dfs_mode>::
168 169
    _M_dfs(_StateIdT __i)
    {
170 171 172 173 174 175 176 177
      if (!__dfs_mode)
	{
	  if ((*_M_visited)[__i])
	    return;
	  (*_M_visited)[__i] = true;
	}

      const auto& __state = _M_nfa[__i];
178 179
      // Every change on _M_cur_results and _M_current will be rolled back after
      // finishing the recursion step.
180
      switch (__state._M_opcode)
181
	{
182 183 184 185
	// _M_alt branch is "match once more", while _M_next is "get me out
	// of this quantifier". Executing _M_next first or _M_alt first don't
	// mean the same thing, and we need to choose the correct order under
	// given greedy mode.
186
	case _S_opcode_alternative:
187
	  // Greedy.
188
	  if (!__state._M_neg)
189
	    {
190
	      // "Once more" is preferred in greedy mode.
191
	      _M_dfs<__match_mode>(__state._M_alt);
192
	      // If it's DFS executor and already accepted, we're done.
193 194 195
	      if (!__dfs_mode || !_M_has_sol)
		_M_dfs<__match_mode>(__state._M_next);
	    }
196
	  else // Non-greedy mode
197 198 199
	    {
	      if (__dfs_mode)
		{
200
		  // vice-versa.
201 202 203 204 205 206
		  _M_dfs<__match_mode>(__state._M_next);
		  if (!_M_has_sol)
		    _M_dfs<__match_mode>(__state._M_alt);
		}
	      else
		{
207 208 209
		  // DON'T attempt anything, because there's already another
		  // state with higher priority accepted. This state cannot be
		  // better by attempting its next node.
210 211 212
		  if (!_M_has_sol)
		    {
		      _M_dfs<__match_mode>(__state._M_next);
213 214 215
		      // DON'T attempt anything if it's already accepted. An
		      // accepted state *must* be better than a solution that
		      // matches a non-greedy quantifier one more time.
216 217 218 219 220
		      if (!_M_has_sol)
			_M_dfs<__match_mode>(__state._M_alt);
		    }
		}
	    }
221 222
	  break;
	case _S_opcode_subexpr_begin:
223 224 225
	  // If there's nothing changed since last visit, do NOT continue.
	  // This prevents the executor from get into infinite loop when using
	  // "()*" to match "".
Tim Shen committed
226
	  if (!_M_cur_results[__state._M_subexpr].matched
227
	      || _M_cur_results[__state._M_subexpr].first != _M_current)
228
	    {
229 230 231 232 233
	      auto& __res = _M_cur_results[__state._M_subexpr];
	      auto __back = __res.first;
	      __res.first = _M_current;
	      _M_dfs<__match_mode>(__state._M_next);
	      __res.first = __back;
234 235 236
	    }
	  break;
	case _S_opcode_subexpr_end:
237
	  if (_M_cur_results[__state._M_subexpr].second != _M_current
Tim Shen committed
238
	      || _M_cur_results[__state._M_subexpr].matched != true)
239
	    {
240 241 242 243 244 245
	      auto& __res = _M_cur_results[__state._M_subexpr];
	      auto __back = __res;
	      __res.second = _M_current;
	      __res.matched = true;
	      _M_dfs<__match_mode>(__state._M_next);
	      __res = __back;
246 247
	    }
	  else
248
	    _M_dfs<__match_mode>(__state._M_next);
249
	  break;
250
	case _S_opcode_line_begin_assertion:
251 252
	  if (_M_at_begin())
	    _M_dfs<__match_mode>(__state._M_next);
253 254
	  break;
	case _S_opcode_line_end_assertion:
255 256
	  if (_M_at_end())
	    _M_dfs<__match_mode>(__state._M_next);
257
	  break;
258 259
	case _S_opcode_word_boundary:
	  if (_M_word_boundary(__state) == !__state._M_neg)
260
	    _M_dfs<__match_mode>(__state._M_next);
261
	  break;
262 263
	// Here __state._M_alt offers a single start node for a sub-NFA.
	// We recursively invoke our algorithm to match the sub-NFA.
264
	case _S_opcode_subexpr_lookahead:
265 266
	  if (_M_lookahead(__state) == !__state._M_neg)
	    _M_dfs<__match_mode>(__state._M_next);
267
	  break;
268
	case _S_opcode_match:
269
	  if (__dfs_mode)
270
	    {
271 272 273 274 275 276
	      if (_M_current != _M_end && __state._M_matches(*_M_current))
		{
		  ++_M_current;
		  _M_dfs<__match_mode>(__state._M_next);
		  --_M_current;
		}
277
	    }
278 279
	  else
	    if (__state._M_matches(*_M_current))
280 281
	      _M_match_queue->push_back(make_pair(__state._M_next,
						  _M_cur_results));
282
	  break;
Tim Shen committed
283
	// First fetch the matched result from _M_cur_results as __submatch;
284
	// then compare it with
285 286
	// (_M_current, _M_current + (__submatch.second - __submatch.first)).
	// If matched, keep going; else just return and try another state.
287 288
	case _S_opcode_backref:
	  {
289
	    _GLIBCXX_DEBUG_ASSERT(__dfs_mode);
Tim Shen committed
290
	    auto& __submatch = _M_cur_results[__state._M_backref_index];
291 292
	    if (!__submatch.matched)
	      break;
293
	    auto __last = _M_current;
294
	    for (auto __tmp = __submatch.first;
295
		 __last != _M_end && __tmp != __submatch.second;
296 297
		 ++__tmp)
	      ++__last;
298
	    if (_M_re._M_traits.transform(__submatch.first,
Tim Shen committed
299
						__submatch.second)
300
		== _M_re._M_traits.transform(_M_current, __last))
301
	      {
302
		if (__last != _M_current)
303
		  {
304 305 306 307
		    auto __backup = _M_current;
		    _M_current = __last;
		    _M_dfs<__match_mode>(__state._M_next);
		    _M_current = __backup;
308 309
		  }
		else
310
		  _M_dfs<__match_mode>(__state._M_next);
311
	      }
312 313 314
	  }
	  break;
	case _S_opcode_accept:
315
	  if (__dfs_mode)
316
	    {
317 318 319 320 321 322 323 324 325 326
	      _GLIBCXX_DEBUG_ASSERT(!_M_has_sol);
	      if (__match_mode)
		_M_has_sol = _M_current == _M_end;
	      else
		_M_has_sol = true;
	      if (_M_current == _M_begin
		  && (_M_flags & regex_constants::match_not_null))
		_M_has_sol = false;
	      if (_M_has_sol)
		_M_results = _M_cur_results;
327
	    }
328
	  else
329
	    {
330 331 332 333 334 335 336 337 338
	      if (_M_current == _M_begin
		  && (_M_flags & regex_constants::match_not_null))
		break;
	      if (!__match_mode || _M_current == _M_end)
		if (!_M_has_sol)
		  {
		    _M_has_sol = true;
		    _M_results = _M_cur_results;
		  }
339
	    }
340 341 342
	  break;
	default:
	  _GLIBCXX_DEBUG_ASSERT(false);
343
	}
344 345
    }

346
  // Return whether now is at some word boundary.
347 348 349
  template<typename _BiIter, typename _Alloc, typename _TraitsT,
    bool __dfs_mode>
    bool _Executor<_BiIter, _Alloc, _TraitsT, __dfs_mode>::
350
    _M_word_boundary(_State<_TraitsT> __state) const
Tim Shen committed
351 352 353 354 355 356
    {
      // By definition.
      bool __ans = false;
      auto __pre = _M_current;
      --__pre;
      if (!(_M_at_begin() && _M_at_end()))
357 358 359 360 361 362 363 364 365 366 367
	{
	  if (_M_at_begin())
	    __ans = _M_is_word(*_M_current)
	      && !(_M_flags & regex_constants::match_not_bow);
	  else if (_M_at_end())
	    __ans = _M_is_word(*__pre)
	      && !(_M_flags & regex_constants::match_not_eow);
	  else
	    __ans = _M_is_word(*_M_current)
	      != _M_is_word(*__pre);
	}
Tim Shen committed
368 369 370
      return __ans;
    }

371 372 373
_GLIBCXX_END_NAMESPACE_VERSION
} // namespace __detail
} // namespace