1 /*
2 *
3 * Copyright (c) 2002
4 * John Maddock
5 *
6 * Use, modification and distribution are subject to the
7 * Boost Software License, Version 1.0. (See accompanying file
8 * LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
9 *
10 */
11
12 /*
13 * LOCATION: see http://www.boost.org for most recent version.
14 * FILE perl_matcher_common.cpp
15 * VERSION see <boost/version.hpp>
16 * DESCRIPTION: Definitions of perl_matcher member functions that are
17 * specific to the non-recursive implementation.
18 */
19
20 #ifndef BOOST_REGEX_V4_PERL_MATCHER_NON_RECURSIVE_HPP
21 #define BOOST_REGEX_V4_PERL_MATCHER_NON_RECURSIVE_HPP
22
23 #include <new>
24
25 #ifdef BOOST_MSVC
26 #pragma warning(push)
27 #pragma warning(disable: 4103)
28 #endif
29 #ifdef BOOST_HAS_ABI_HEADERS
30 # include BOOST_ABI_PREFIX
31 #endif
32 #ifdef BOOST_MSVC
33 #pragma warning(pop)
34 #endif
35 #ifdef BOOST_MSVC
36 # pragma warning(push)
37 # pragma warning(disable: 4706)
38 #if BOOST_MSVC < 1910
39 #pragma warning(disable:4800)
40 #endif
41 #endif
42
43 namespace boost{
44 namespace BOOST_REGEX_DETAIL_NS{
45
46 template <class T>
inplace_destroy(T * p)47 inline void inplace_destroy(T* p)
48 {
49 (void)p; // warning suppression
50 p->~T();
51 }
52
53 struct saved_state
54 {
55 union{
56 unsigned int state_id;
57 // this padding ensures correct alignment on 64-bit platforms:
58 std::size_t padding1;
59 std::ptrdiff_t padding2;
60 void* padding3;
61 };
saved_stateboost::BOOST_REGEX_DETAIL_NS::saved_state62 saved_state(unsigned i) : state_id(i) {}
63 };
64
65 template <class BidiIterator>
66 struct saved_matched_paren : public saved_state
67 {
68 int index;
69 sub_match<BidiIterator> sub;
saved_matched_parenboost::BOOST_REGEX_DETAIL_NS::saved_matched_paren70 saved_matched_paren(int i, const sub_match<BidiIterator>& s) : saved_state(1), index(i), sub(s){}
71 };
72
73 template <class BidiIterator>
74 struct saved_position : public saved_state
75 {
76 const re_syntax_base* pstate;
77 BidiIterator position;
saved_positionboost::BOOST_REGEX_DETAIL_NS::saved_position78 saved_position(const re_syntax_base* ps, BidiIterator pos, int i) : saved_state(i), pstate(ps), position(pos){}
79 };
80
81 template <class BidiIterator>
82 struct saved_assertion : public saved_position<BidiIterator>
83 {
84 bool positive;
saved_assertionboost::BOOST_REGEX_DETAIL_NS::saved_assertion85 saved_assertion(bool p, const re_syntax_base* ps, BidiIterator pos)
86 : saved_position<BidiIterator>(ps, pos, saved_type_assertion), positive(p){}
87 };
88
89 template <class BidiIterator>
90 struct saved_repeater : public saved_state
91 {
92 repeater_count<BidiIterator> count;
saved_repeaterboost::BOOST_REGEX_DETAIL_NS::saved_repeater93 saved_repeater(int i, repeater_count<BidiIterator>** s, BidiIterator start, int current_recursion_id)
94 : saved_state(saved_state_repeater_count), count(i, s, start, current_recursion_id){}
95 };
96
97 struct saved_extra_block : public saved_state
98 {
99 saved_state *base, *end;
saved_extra_blockboost::BOOST_REGEX_DETAIL_NS::saved_extra_block100 saved_extra_block(saved_state* b, saved_state* e)
101 : saved_state(saved_state_extra_block), base(b), end(e) {}
102 };
103
104 struct save_state_init
105 {
106 saved_state** stack;
save_state_initboost::BOOST_REGEX_DETAIL_NS::save_state_init107 save_state_init(saved_state** base, saved_state** end)
108 : stack(base)
109 {
110 *base = static_cast<saved_state*>(get_mem_block());
111 *end = reinterpret_cast<saved_state*>(reinterpret_cast<char*>(*base)+BOOST_REGEX_BLOCKSIZE);
112 --(*end);
113 (void) new (*end)saved_state(0);
114 BOOST_ASSERT(*end > *base);
115 }
~save_state_initboost::BOOST_REGEX_DETAIL_NS::save_state_init116 ~save_state_init()
117 {
118 put_mem_block(*stack);
119 *stack = 0;
120 }
121 };
122
123 template <class BidiIterator>
124 struct saved_single_repeat : public saved_state
125 {
126 std::size_t count;
127 const re_repeat* rep;
128 BidiIterator last_position;
saved_single_repeatboost::BOOST_REGEX_DETAIL_NS::saved_single_repeat129 saved_single_repeat(std::size_t c, const re_repeat* r, BidiIterator lp, int arg_id)
130 : saved_state(arg_id), count(c), rep(r), last_position(lp){}
131 };
132
133 template <class Results>
134 struct saved_recursion : public saved_state
135 {
saved_recursionboost::BOOST_REGEX_DETAIL_NS::saved_recursion136 saved_recursion(int idx, const re_syntax_base* p, Results* pr, Results* pr2)
137 : saved_state(14), recursion_id(idx), preturn_address(p), internal_results(*pr), prior_results(*pr2) {}
138 int recursion_id;
139 const re_syntax_base* preturn_address;
140 Results internal_results, prior_results;
141 };
142
143 struct saved_change_case : public saved_state
144 {
145 bool icase;
saved_change_caseboost::BOOST_REGEX_DETAIL_NS::saved_change_case146 saved_change_case(bool c) : saved_state(18), icase(c) {}
147 };
148
149 struct incrementer
150 {
incrementerboost::BOOST_REGEX_DETAIL_NS::incrementer151 incrementer(unsigned* pu) : m_pu(pu) { ++*m_pu; }
~incrementerboost::BOOST_REGEX_DETAIL_NS::incrementer152 ~incrementer() { --*m_pu; }
operator >boost::BOOST_REGEX_DETAIL_NS::incrementer153 bool operator > (unsigned i) { return *m_pu > i; }
154 private:
155 unsigned* m_pu;
156 };
157
158 template <class BidiIterator, class Allocator, class traits>
match_all_states()159 bool perl_matcher<BidiIterator, Allocator, traits>::match_all_states()
160 {
161 static matcher_proc_type const s_match_vtable[34] =
162 {
163 (&perl_matcher<BidiIterator, Allocator, traits>::match_startmark),
164 &perl_matcher<BidiIterator, Allocator, traits>::match_endmark,
165 &perl_matcher<BidiIterator, Allocator, traits>::match_literal,
166 &perl_matcher<BidiIterator, Allocator, traits>::match_start_line,
167 &perl_matcher<BidiIterator, Allocator, traits>::match_end_line,
168 &perl_matcher<BidiIterator, Allocator, traits>::match_wild,
169 &perl_matcher<BidiIterator, Allocator, traits>::match_match,
170 &perl_matcher<BidiIterator, Allocator, traits>::match_word_boundary,
171 &perl_matcher<BidiIterator, Allocator, traits>::match_within_word,
172 &perl_matcher<BidiIterator, Allocator, traits>::match_word_start,
173 &perl_matcher<BidiIterator, Allocator, traits>::match_word_end,
174 &perl_matcher<BidiIterator, Allocator, traits>::match_buffer_start,
175 &perl_matcher<BidiIterator, Allocator, traits>::match_buffer_end,
176 &perl_matcher<BidiIterator, Allocator, traits>::match_backref,
177 &perl_matcher<BidiIterator, Allocator, traits>::match_long_set,
178 &perl_matcher<BidiIterator, Allocator, traits>::match_set,
179 &perl_matcher<BidiIterator, Allocator, traits>::match_jump,
180 &perl_matcher<BidiIterator, Allocator, traits>::match_alt,
181 &perl_matcher<BidiIterator, Allocator, traits>::match_rep,
182 &perl_matcher<BidiIterator, Allocator, traits>::match_combining,
183 &perl_matcher<BidiIterator, Allocator, traits>::match_soft_buffer_end,
184 &perl_matcher<BidiIterator, Allocator, traits>::match_restart_continue,
185 // Although this next line *should* be evaluated at compile time, in practice
186 // some compilers (VC++) emit run-time initialisation which breaks thread
187 // safety, so use a dispatch function instead:
188 //(::boost::is_random_access_iterator<BidiIterator>::value ? &perl_matcher<BidiIterator, Allocator, traits>::match_dot_repeat_fast : &perl_matcher<BidiIterator, Allocator, traits>::match_dot_repeat_slow),
189 &perl_matcher<BidiIterator, Allocator, traits>::match_dot_repeat_dispatch,
190 &perl_matcher<BidiIterator, Allocator, traits>::match_char_repeat,
191 &perl_matcher<BidiIterator, Allocator, traits>::match_set_repeat,
192 &perl_matcher<BidiIterator, Allocator, traits>::match_long_set_repeat,
193 &perl_matcher<BidiIterator, Allocator, traits>::match_backstep,
194 &perl_matcher<BidiIterator, Allocator, traits>::match_assert_backref,
195 &perl_matcher<BidiIterator, Allocator, traits>::match_toggle_case,
196 &perl_matcher<BidiIterator, Allocator, traits>::match_recursion,
197 &perl_matcher<BidiIterator, Allocator, traits>::match_fail,
198 &perl_matcher<BidiIterator, Allocator, traits>::match_accept,
199 &perl_matcher<BidiIterator, Allocator, traits>::match_commit,
200 &perl_matcher<BidiIterator, Allocator, traits>::match_then,
201 };
202 incrementer inc(&m_recursions);
203 if(inc > 80)
204 raise_error(traits_inst, regex_constants::error_complexity);
205 push_recursion_stopper();
206 do{
207 while(pstate)
208 {
209 matcher_proc_type proc = s_match_vtable[pstate->type];
210 ++state_count;
211 if(!(this->*proc)())
212 {
213 if(state_count > max_state_count)
214 raise_error(traits_inst, regex_constants::error_complexity);
215 if((m_match_flags & match_partial) && (position == last) && (position != search_base))
216 m_has_partial_match = true;
217 bool successful_unwind = unwind(false);
218 if((m_match_flags & match_partial) && (position == last) && (position != search_base))
219 m_has_partial_match = true;
220 if(false == successful_unwind)
221 return m_recursive_result;
222 }
223 }
224 }while(unwind(true));
225 return m_recursive_result;
226 }
227
228 template <class BidiIterator, class Allocator, class traits>
extend_stack()229 void perl_matcher<BidiIterator, Allocator, traits>::extend_stack()
230 {
231 if(used_block_count)
232 {
233 --used_block_count;
234 saved_state* stack_base;
235 saved_state* backup_state;
236 stack_base = static_cast<saved_state*>(get_mem_block());
237 backup_state = reinterpret_cast<saved_state*>(reinterpret_cast<char*>(stack_base)+BOOST_REGEX_BLOCKSIZE);
238 saved_extra_block* block = static_cast<saved_extra_block*>(backup_state);
239 --block;
240 (void) new (block) saved_extra_block(m_stack_base, m_backup_state);
241 m_stack_base = stack_base;
242 m_backup_state = block;
243 }
244 else
245 raise_error(traits_inst, regex_constants::error_stack);
246 }
247
248 template <class BidiIterator, class Allocator, class traits>
push_matched_paren(int index,const sub_match<BidiIterator> & sub)249 inline void perl_matcher<BidiIterator, Allocator, traits>::push_matched_paren(int index, const sub_match<BidiIterator>& sub)
250 {
251 //BOOST_ASSERT(index);
252 saved_matched_paren<BidiIterator>* pmp = static_cast<saved_matched_paren<BidiIterator>*>(m_backup_state);
253 --pmp;
254 if(pmp < m_stack_base)
255 {
256 extend_stack();
257 pmp = static_cast<saved_matched_paren<BidiIterator>*>(m_backup_state);
258 --pmp;
259 }
260 (void) new (pmp)saved_matched_paren<BidiIterator>(index, sub);
261 m_backup_state = pmp;
262 }
263
264 template <class BidiIterator, class Allocator, class traits>
push_case_change(bool c)265 inline void perl_matcher<BidiIterator, Allocator, traits>::push_case_change(bool c)
266 {
267 //BOOST_ASSERT(index);
268 saved_change_case* pmp = static_cast<saved_change_case*>(m_backup_state);
269 --pmp;
270 if(pmp < m_stack_base)
271 {
272 extend_stack();
273 pmp = static_cast<saved_change_case*>(m_backup_state);
274 --pmp;
275 }
276 (void) new (pmp)saved_change_case(c);
277 m_backup_state = pmp;
278 }
279
280 template <class BidiIterator, class Allocator, class traits>
push_recursion_stopper()281 inline void perl_matcher<BidiIterator, Allocator, traits>::push_recursion_stopper()
282 {
283 saved_state* pmp = m_backup_state;
284 --pmp;
285 if(pmp < m_stack_base)
286 {
287 extend_stack();
288 pmp = m_backup_state;
289 --pmp;
290 }
291 (void) new (pmp)saved_state(saved_type_recurse);
292 m_backup_state = pmp;
293 }
294
295 template <class BidiIterator, class Allocator, class traits>
push_assertion(const re_syntax_base * ps,bool positive)296 inline void perl_matcher<BidiIterator, Allocator, traits>::push_assertion(const re_syntax_base* ps, bool positive)
297 {
298 saved_assertion<BidiIterator>* pmp = static_cast<saved_assertion<BidiIterator>*>(m_backup_state);
299 --pmp;
300 if(pmp < m_stack_base)
301 {
302 extend_stack();
303 pmp = static_cast<saved_assertion<BidiIterator>*>(m_backup_state);
304 --pmp;
305 }
306 (void) new (pmp)saved_assertion<BidiIterator>(positive, ps, position);
307 m_backup_state = pmp;
308 }
309
310 template <class BidiIterator, class Allocator, class traits>
push_alt(const re_syntax_base * ps)311 inline void perl_matcher<BidiIterator, Allocator, traits>::push_alt(const re_syntax_base* ps)
312 {
313 saved_position<BidiIterator>* pmp = static_cast<saved_position<BidiIterator>*>(m_backup_state);
314 --pmp;
315 if(pmp < m_stack_base)
316 {
317 extend_stack();
318 pmp = static_cast<saved_position<BidiIterator>*>(m_backup_state);
319 --pmp;
320 }
321 (void) new (pmp)saved_position<BidiIterator>(ps, position, saved_state_alt);
322 m_backup_state = pmp;
323 }
324
325 template <class BidiIterator, class Allocator, class traits>
push_non_greedy_repeat(const re_syntax_base * ps)326 inline void perl_matcher<BidiIterator, Allocator, traits>::push_non_greedy_repeat(const re_syntax_base* ps)
327 {
328 saved_position<BidiIterator>* pmp = static_cast<saved_position<BidiIterator>*>(m_backup_state);
329 --pmp;
330 if(pmp < m_stack_base)
331 {
332 extend_stack();
333 pmp = static_cast<saved_position<BidiIterator>*>(m_backup_state);
334 --pmp;
335 }
336 (void) new (pmp)saved_position<BidiIterator>(ps, position, saved_state_non_greedy_long_repeat);
337 m_backup_state = pmp;
338 }
339
340 template <class BidiIterator, class Allocator, class traits>
push_repeater_count(int i,repeater_count<BidiIterator> ** s)341 inline void perl_matcher<BidiIterator, Allocator, traits>::push_repeater_count(int i, repeater_count<BidiIterator>** s)
342 {
343 saved_repeater<BidiIterator>* pmp = static_cast<saved_repeater<BidiIterator>*>(m_backup_state);
344 --pmp;
345 if(pmp < m_stack_base)
346 {
347 extend_stack();
348 pmp = static_cast<saved_repeater<BidiIterator>*>(m_backup_state);
349 --pmp;
350 }
351 (void) new (pmp)saved_repeater<BidiIterator>(i, s, position, this->recursion_stack.size() ? this->recursion_stack.back().idx : (INT_MIN + 3));
352 m_backup_state = pmp;
353 }
354
355 template <class BidiIterator, class Allocator, class traits>
push_single_repeat(std::size_t c,const re_repeat * r,BidiIterator last_position,int state_id)356 inline void perl_matcher<BidiIterator, Allocator, traits>::push_single_repeat(std::size_t c, const re_repeat* r, BidiIterator last_position, int state_id)
357 {
358 saved_single_repeat<BidiIterator>* pmp = static_cast<saved_single_repeat<BidiIterator>*>(m_backup_state);
359 --pmp;
360 if(pmp < m_stack_base)
361 {
362 extend_stack();
363 pmp = static_cast<saved_single_repeat<BidiIterator>*>(m_backup_state);
364 --pmp;
365 }
366 (void) new (pmp)saved_single_repeat<BidiIterator>(c, r, last_position, state_id);
367 m_backup_state = pmp;
368 }
369
370 template <class BidiIterator, class Allocator, class traits>
push_recursion(int idx,const re_syntax_base * p,results_type * presults,results_type * presults2)371 inline void perl_matcher<BidiIterator, Allocator, traits>::push_recursion(int idx, const re_syntax_base* p, results_type* presults, results_type* presults2)
372 {
373 saved_recursion<results_type>* pmp = static_cast<saved_recursion<results_type>*>(m_backup_state);
374 --pmp;
375 if(pmp < m_stack_base)
376 {
377 extend_stack();
378 pmp = static_cast<saved_recursion<results_type>*>(m_backup_state);
379 --pmp;
380 }
381 (void) new (pmp)saved_recursion<results_type>(idx, p, presults, presults2);
382 m_backup_state = pmp;
383 }
384
385 template <class BidiIterator, class Allocator, class traits>
match_toggle_case()386 bool perl_matcher<BidiIterator, Allocator, traits>::match_toggle_case()
387 {
388 // change our case sensitivity:
389 push_case_change(this->icase);
390 this->icase = static_cast<const re_case*>(pstate)->icase;
391 pstate = pstate->next.p;
392 return true;
393 }
394
395 template <class BidiIterator, class Allocator, class traits>
match_startmark()396 bool perl_matcher<BidiIterator, Allocator, traits>::match_startmark()
397 {
398 int index = static_cast<const re_brace*>(pstate)->index;
399 icase = static_cast<const re_brace*>(pstate)->icase;
400 switch(index)
401 {
402 case 0:
403 pstate = pstate->next.p;
404 break;
405 case -1:
406 case -2:
407 {
408 // forward lookahead assert:
409 const re_syntax_base* next_pstate = static_cast<const re_jump*>(pstate->next.p)->alt.p->next.p;
410 pstate = pstate->next.p->next.p;
411 push_assertion(next_pstate, index == -1);
412 break;
413 }
414 case -3:
415 {
416 // independent sub-expression, currently this is always recursive:
417 bool old_independent = m_independent;
418 m_independent = true;
419 const re_syntax_base* next_pstate = static_cast<const re_jump*>(pstate->next.p)->alt.p->next.p;
420 pstate = pstate->next.p->next.p;
421 bool r = false;
422 #if !defined(BOOST_NO_EXCEPTIONS)
423 try{
424 #endif
425 r = match_all_states();
426 if(!r && !m_independent)
427 {
428 // Must be unwinding from a COMMIT/SKIP/PRUNE and the independent
429 // sub failed, need to unwind everything else:
430 while(unwind(false));
431 return false;
432 }
433 #if !defined(BOOST_NO_EXCEPTIONS)
434 }
435 catch(...)
436 {
437 pstate = next_pstate;
438 // unwind all pushed states, apart from anything else this
439 // ensures that all the states are correctly destructed
440 // not just the memory freed.
441 while(unwind(true)) {}
442 throw;
443 }
444 #endif
445 pstate = next_pstate;
446 m_independent = old_independent;
447 #ifdef BOOST_REGEX_MATCH_EXTRA
448 if(r && (m_match_flags & match_extra))
449 {
450 //
451 // our captures have been stored in *m_presult
452 // we need to unpack them, and insert them
453 // back in the right order when we unwind the stack:
454 //
455 match_results<BidiIterator, Allocator> temp_match(*m_presult);
456 unsigned i;
457 for(i = 0; i < temp_match.size(); ++i)
458 (*m_presult)[i].get_captures().clear();
459 // match everything else:
460 #if !defined(BOOST_NO_EXCEPTIONS)
461 try{
462 #endif
463 r = match_all_states();
464 #if !defined(BOOST_NO_EXCEPTIONS)
465 }
466 catch(...)
467 {
468 pstate = next_pstate;
469 // unwind all pushed states, apart from anything else this
470 // ensures that all the states are correctly destructed
471 // not just the memory freed.
472 while(unwind(true)) {}
473 throw;
474 }
475 #endif
476 // now place the stored captures back:
477 for(i = 0; i < temp_match.size(); ++i)
478 {
479 typedef typename sub_match<BidiIterator>::capture_sequence_type seq;
480 seq& s1 = (*m_presult)[i].get_captures();
481 const seq& s2 = temp_match[i].captures();
482 s1.insert(
483 s1.end(),
484 s2.begin(),
485 s2.end());
486 }
487 }
488 #endif
489 return r;
490 }
491 case -4:
492 {
493 // conditional expression:
494 const re_alt* alt = static_cast<const re_alt*>(pstate->next.p);
495 BOOST_ASSERT(alt->type == syntax_element_alt);
496 pstate = alt->next.p;
497 if(pstate->type == syntax_element_assert_backref)
498 {
499 if(!match_assert_backref())
500 pstate = alt->alt.p;
501 break;
502 }
503 else
504 {
505 // zero width assertion, have to match this recursively:
506 BOOST_ASSERT(pstate->type == syntax_element_startmark);
507 bool negated = static_cast<const re_brace*>(pstate)->index == -2;
508 BidiIterator saved_position = position;
509 const re_syntax_base* next_pstate = static_cast<const re_jump*>(pstate->next.p)->alt.p->next.p;
510 pstate = pstate->next.p->next.p;
511 #if !defined(BOOST_NO_EXCEPTIONS)
512 try{
513 #endif
514 bool r = match_all_states();
515 position = saved_position;
516 if(negated)
517 r = !r;
518 if(r)
519 pstate = next_pstate;
520 else
521 pstate = alt->alt.p;
522 #if !defined(BOOST_NO_EXCEPTIONS)
523 }
524 catch(...)
525 {
526 pstate = next_pstate;
527 // unwind all pushed states, apart from anything else this
528 // ensures that all the states are correctly destructed
529 // not just the memory freed.
530 while(unwind(true)){}
531 throw;
532 }
533 #endif
534 break;
535 }
536 }
537 case -5:
538 {
539 push_matched_paren(0, (*m_presult)[0]);
540 m_presult->set_first(position, 0, true);
541 pstate = pstate->next.p;
542 break;
543 }
544 default:
545 {
546 BOOST_ASSERT(index > 0);
547 if((m_match_flags & match_nosubs) == 0)
548 {
549 push_matched_paren(index, (*m_presult)[index]);
550 m_presult->set_first(position, index);
551 }
552 pstate = pstate->next.p;
553 break;
554 }
555 }
556 return true;
557 }
558
559 template <class BidiIterator, class Allocator, class traits>
match_alt()560 bool perl_matcher<BidiIterator, Allocator, traits>::match_alt()
561 {
562 bool take_first, take_second;
563 const re_alt* jmp = static_cast<const re_alt*>(pstate);
564
565 // find out which of these two alternatives we need to take:
566 if(position == last)
567 {
568 take_first = jmp->can_be_null & mask_take;
569 take_second = jmp->can_be_null & mask_skip;
570 }
571 else
572 {
573 take_first = can_start(*position, jmp->_map, (unsigned char)mask_take);
574 take_second = can_start(*position, jmp->_map, (unsigned char)mask_skip);
575 }
576
577 if(take_first)
578 {
579 // we can take the first alternative,
580 // see if we need to push next alternative:
581 if(take_second)
582 {
583 push_alt(jmp->alt.p);
584 }
585 pstate = pstate->next.p;
586 return true;
587 }
588 if(take_second)
589 {
590 pstate = jmp->alt.p;
591 return true;
592 }
593 return false; // neither option is possible
594 }
595
596 template <class BidiIterator, class Allocator, class traits>
match_rep()597 bool perl_matcher<BidiIterator, Allocator, traits>::match_rep()
598 {
599 #ifdef BOOST_MSVC
600 #pragma warning(push)
601 #pragma warning(disable:4127 4244)
602 #endif
603 #ifdef BOOST_BORLANDC
604 #pragma option push -w-8008 -w-8066 -w-8004
605 #endif
606 const re_repeat* rep = static_cast<const re_repeat*>(pstate);
607
608 // find out which of these two alternatives we need to take:
609 bool take_first, take_second;
610 if(position == last)
611 {
612 take_first = rep->can_be_null & mask_take;
613 take_second = rep->can_be_null & mask_skip;
614 }
615 else
616 {
617 take_first = can_start(*position, rep->_map, (unsigned char)mask_take);
618 take_second = can_start(*position, rep->_map, (unsigned char)mask_skip);
619 }
620
621 if((m_backup_state->state_id != saved_state_repeater_count)
622 || (static_cast<saved_repeater<BidiIterator>*>(m_backup_state)->count.get_id() != rep->state_id)
623 || (next_count->get_id() != rep->state_id))
624 {
625 // we're moving to a different repeat from the last
626 // one, so set up a counter object:
627 push_repeater_count(rep->state_id, &next_count);
628 }
629 //
630 // If we've had at least one repeat already, and the last one
631 // matched the NULL string then set the repeat count to
632 // maximum:
633 //
634 next_count->check_null_repeat(position, rep->max);
635
636 if(next_count->get_count() < rep->min)
637 {
638 // we must take the repeat:
639 if(take_first)
640 {
641 // increase the counter:
642 ++(*next_count);
643 pstate = rep->next.p;
644 return true;
645 }
646 return false;
647 }
648
649 bool greedy = (rep->greedy) && (!(m_match_flags & regex_constants::match_any) || m_independent);
650 if(greedy)
651 {
652 // try and take the repeat if we can:
653 if((next_count->get_count() < rep->max) && take_first)
654 {
655 if(take_second)
656 {
657 // store position in case we fail:
658 push_alt(rep->alt.p);
659 }
660 // increase the counter:
661 ++(*next_count);
662 pstate = rep->next.p;
663 return true;
664 }
665 else if(take_second)
666 {
667 pstate = rep->alt.p;
668 return true;
669 }
670 return false; // can't take anything, fail...
671 }
672 else // non-greedy
673 {
674 // try and skip the repeat if we can:
675 if(take_second)
676 {
677 if((next_count->get_count() < rep->max) && take_first)
678 {
679 // store position in case we fail:
680 push_non_greedy_repeat(rep->next.p);
681 }
682 pstate = rep->alt.p;
683 return true;
684 }
685 if((next_count->get_count() < rep->max) && take_first)
686 {
687 // increase the counter:
688 ++(*next_count);
689 pstate = rep->next.p;
690 return true;
691 }
692 }
693 return false;
694 #ifdef BOOST_BORLANDC
695 #pragma option pop
696 #endif
697 #ifdef BOOST_MSVC
698 #pragma warning(pop)
699 #endif
700 }
701
702 template <class BidiIterator, class Allocator, class traits>
match_dot_repeat_slow()703 bool perl_matcher<BidiIterator, Allocator, traits>::match_dot_repeat_slow()
704 {
705 std::size_t count = 0;
706 const re_repeat* rep = static_cast<const re_repeat*>(pstate);
707 re_syntax_base* psingle = rep->next.p;
708 // match compulsary repeats first:
709 while(count < rep->min)
710 {
711 pstate = psingle;
712 if(!match_wild())
713 return false;
714 ++count;
715 }
716 bool greedy = (rep->greedy) && (!(m_match_flags & regex_constants::match_any) || m_independent);
717 if(greedy)
718 {
719 // repeat for as long as we can:
720 while(count < rep->max)
721 {
722 pstate = psingle;
723 if(!match_wild())
724 break;
725 ++count;
726 }
727 // remember where we got to if this is a leading repeat:
728 if((rep->leading) && (count < rep->max))
729 restart = position;
730 // push backtrack info if available:
731 if(count - rep->min)
732 push_single_repeat(count, rep, position, saved_state_greedy_single_repeat);
733 // jump to next state:
734 pstate = rep->alt.p;
735 return true;
736 }
737 else
738 {
739 // non-greedy, push state and return true if we can skip:
740 if(count < rep->max)
741 push_single_repeat(count, rep, position, saved_state_rep_slow_dot);
742 pstate = rep->alt.p;
743 return (position == last) ? (rep->can_be_null & mask_skip) : can_start(*position, rep->_map, mask_skip);
744 }
745 }
746
747 template <class BidiIterator, class Allocator, class traits>
match_dot_repeat_fast()748 bool perl_matcher<BidiIterator, Allocator, traits>::match_dot_repeat_fast()
749 {
750 if(m_match_flags & match_not_dot_null)
751 return match_dot_repeat_slow();
752 if((static_cast<const re_dot*>(pstate->next.p)->mask & match_any_mask) == 0)
753 return match_dot_repeat_slow();
754
755 const re_repeat* rep = static_cast<const re_repeat*>(pstate);
756 bool greedy = (rep->greedy) && (!(m_match_flags & regex_constants::match_any) || m_independent);
757 std::size_t count = static_cast<std::size_t>((std::min)(static_cast<std::size_t>(::boost::BOOST_REGEX_DETAIL_NS::distance(position, last)), greedy ? rep->max : rep->min));
758 if(rep->min > count)
759 {
760 position = last;
761 return false; // not enough text left to match
762 }
763 std::advance(position, count);
764
765 if(greedy)
766 {
767 if((rep->leading) && (count < rep->max))
768 restart = position;
769 // push backtrack info if available:
770 if(count - rep->min)
771 push_single_repeat(count, rep, position, saved_state_greedy_single_repeat);
772 // jump to next state:
773 pstate = rep->alt.p;
774 return true;
775 }
776 else
777 {
778 // non-greedy, push state and return true if we can skip:
779 if(count < rep->max)
780 push_single_repeat(count, rep, position, saved_state_rep_fast_dot);
781 pstate = rep->alt.p;
782 return (position == last) ? (rep->can_be_null & mask_skip) : can_start(*position, rep->_map, mask_skip);
783 }
784 }
785
786 template <class BidiIterator, class Allocator, class traits>
match_char_repeat()787 bool perl_matcher<BidiIterator, Allocator, traits>::match_char_repeat()
788 {
789 #ifdef BOOST_MSVC
790 #pragma warning(push)
791 #pragma warning(disable:4127)
792 #endif
793 #ifdef BOOST_BORLANDC
794 #pragma option push -w-8008 -w-8066 -w-8004
795 #endif
796 const re_repeat* rep = static_cast<const re_repeat*>(pstate);
797 BOOST_ASSERT(1 == static_cast<const re_literal*>(rep->next.p)->length);
798 const char_type what = *reinterpret_cast<const char_type*>(static_cast<const re_literal*>(rep->next.p) + 1);
799 std::size_t count = 0;
800 //
801 // start by working out how much we can skip:
802 //
803 bool greedy = (rep->greedy) && (!(m_match_flags & regex_constants::match_any) || m_independent);
804 std::size_t desired = greedy ? rep->max : rep->min;
805 if(::boost::is_random_access_iterator<BidiIterator>::value)
806 {
807 BidiIterator end = position;
808 // Move end forward by "desired", preferably without using distance or advance if we can
809 // as these can be slow for some iterator types.
810 std::size_t len = (desired == (std::numeric_limits<std::size_t>::max)()) ? 0u : ::boost::BOOST_REGEX_DETAIL_NS::distance(position, last);
811 if(desired >= len)
812 end = last;
813 else
814 std::advance(end, desired);
815 BidiIterator origin(position);
816 while((position != end) && (traits_inst.translate(*position, icase) == what))
817 {
818 ++position;
819 }
820 count = (unsigned)::boost::BOOST_REGEX_DETAIL_NS::distance(origin, position);
821 }
822 else
823 {
824 while((count < desired) && (position != last) && (traits_inst.translate(*position, icase) == what))
825 {
826 ++position;
827 ++count;
828 }
829 }
830
831 if(count < rep->min)
832 return false;
833
834 if(greedy)
835 {
836 if((rep->leading) && (count < rep->max))
837 restart = position;
838 // push backtrack info if available:
839 if(count - rep->min)
840 push_single_repeat(count, rep, position, saved_state_greedy_single_repeat);
841 // jump to next state:
842 pstate = rep->alt.p;
843 return true;
844 }
845 else
846 {
847 // non-greedy, push state and return true if we can skip:
848 if(count < rep->max)
849 push_single_repeat(count, rep, position, saved_state_rep_char);
850 pstate = rep->alt.p;
851 return (position == last) ? (rep->can_be_null & mask_skip) : can_start(*position, rep->_map, mask_skip);
852 }
853 #ifdef BOOST_BORLANDC
854 #pragma option pop
855 #endif
856 #ifdef BOOST_MSVC
857 #pragma warning(pop)
858 #endif
859 }
860
861 template <class BidiIterator, class Allocator, class traits>
match_set_repeat()862 bool perl_matcher<BidiIterator, Allocator, traits>::match_set_repeat()
863 {
864 #ifdef BOOST_MSVC
865 #pragma warning(push)
866 #pragma warning(disable:4127)
867 #endif
868 #ifdef BOOST_BORLANDC
869 #pragma option push -w-8008 -w-8066 -w-8004
870 #endif
871 const re_repeat* rep = static_cast<const re_repeat*>(pstate);
872 const unsigned char* map = static_cast<const re_set*>(rep->next.p)->_map;
873 std::size_t count = 0;
874 //
875 // start by working out how much we can skip:
876 //
877 bool greedy = (rep->greedy) && (!(m_match_flags & regex_constants::match_any) || m_independent);
878 std::size_t desired = greedy ? rep->max : rep->min;
879 if(::boost::is_random_access_iterator<BidiIterator>::value)
880 {
881 BidiIterator end = position;
882 // Move end forward by "desired", preferably without using distance or advance if we can
883 // as these can be slow for some iterator types.
884 std::size_t len = (desired == (std::numeric_limits<std::size_t>::max)()) ? 0u : ::boost::BOOST_REGEX_DETAIL_NS::distance(position, last);
885 if(desired >= len)
886 end = last;
887 else
888 std::advance(end, desired);
889 BidiIterator origin(position);
890 while((position != end) && map[static_cast<unsigned char>(traits_inst.translate(*position, icase))])
891 {
892 ++position;
893 }
894 count = (unsigned)::boost::BOOST_REGEX_DETAIL_NS::distance(origin, position);
895 }
896 else
897 {
898 while((count < desired) && (position != last) && map[static_cast<unsigned char>(traits_inst.translate(*position, icase))])
899 {
900 ++position;
901 ++count;
902 }
903 }
904
905 if(count < rep->min)
906 return false;
907
908 if(greedy)
909 {
910 if((rep->leading) && (count < rep->max))
911 restart = position;
912 // push backtrack info if available:
913 if(count - rep->min)
914 push_single_repeat(count, rep, position, saved_state_greedy_single_repeat);
915 // jump to next state:
916 pstate = rep->alt.p;
917 return true;
918 }
919 else
920 {
921 // non-greedy, push state and return true if we can skip:
922 if(count < rep->max)
923 push_single_repeat(count, rep, position, saved_state_rep_short_set);
924 pstate = rep->alt.p;
925 return (position == last) ? (rep->can_be_null & mask_skip) : can_start(*position, rep->_map, mask_skip);
926 }
927 #ifdef BOOST_BORLANDC
928 #pragma option pop
929 #endif
930 #ifdef BOOST_MSVC
931 #pragma warning(pop)
932 #endif
933 }
934
935 template <class BidiIterator, class Allocator, class traits>
match_long_set_repeat()936 bool perl_matcher<BidiIterator, Allocator, traits>::match_long_set_repeat()
937 {
938 #ifdef BOOST_MSVC
939 #pragma warning(push)
940 #pragma warning(disable:4127)
941 #endif
942 #ifdef BOOST_BORLANDC
943 #pragma option push -w-8008 -w-8066 -w-8004
944 #endif
945 typedef typename traits::char_class_type m_type;
946 const re_repeat* rep = static_cast<const re_repeat*>(pstate);
947 const re_set_long<m_type>* set = static_cast<const re_set_long<m_type>*>(pstate->next.p);
948 std::size_t count = 0;
949 //
950 // start by working out how much we can skip:
951 //
952 bool greedy = (rep->greedy) && (!(m_match_flags & regex_constants::match_any) || m_independent);
953 std::size_t desired = greedy ? rep->max : rep->min;
954 if(::boost::is_random_access_iterator<BidiIterator>::value)
955 {
956 BidiIterator end = position;
957 // Move end forward by "desired", preferably without using distance or advance if we can
958 // as these can be slow for some iterator types.
959 std::size_t len = (desired == (std::numeric_limits<std::size_t>::max)()) ? 0u : ::boost::BOOST_REGEX_DETAIL_NS::distance(position, last);
960 if(desired >= len)
961 end = last;
962 else
963 std::advance(end, desired);
964 BidiIterator origin(position);
965 while((position != end) && (position != re_is_set_member(position, last, set, re.get_data(), icase)))
966 {
967 ++position;
968 }
969 count = (unsigned)::boost::BOOST_REGEX_DETAIL_NS::distance(origin, position);
970 }
971 else
972 {
973 while((count < desired) && (position != last) && (position != re_is_set_member(position, last, set, re.get_data(), icase)))
974 {
975 ++position;
976 ++count;
977 }
978 }
979
980 if(count < rep->min)
981 return false;
982
983 if(greedy)
984 {
985 if((rep->leading) && (count < rep->max))
986 restart = position;
987 // push backtrack info if available:
988 if(count - rep->min)
989 push_single_repeat(count, rep, position, saved_state_greedy_single_repeat);
990 // jump to next state:
991 pstate = rep->alt.p;
992 return true;
993 }
994 else
995 {
996 // non-greedy, push state and return true if we can skip:
997 if(count < rep->max)
998 push_single_repeat(count, rep, position, saved_state_rep_long_set);
999 pstate = rep->alt.p;
1000 return (position == last) ? (rep->can_be_null & mask_skip) : can_start(*position, rep->_map, mask_skip);
1001 }
1002 #ifdef BOOST_BORLANDC
1003 #pragma option pop
1004 #endif
1005 #ifdef BOOST_MSVC
1006 #pragma warning(pop)
1007 #endif
1008 }
1009
1010 template <class BidiIterator, class Allocator, class traits>
match_recursion()1011 bool perl_matcher<BidiIterator, Allocator, traits>::match_recursion()
1012 {
1013 BOOST_ASSERT(pstate->type == syntax_element_recurse);
1014 //
1015 // See if we've seen this recursion before at this location, if we have then
1016 // we need to prevent infinite recursion:
1017 //
1018 for(typename std::vector<recursion_info<results_type> >::reverse_iterator i = recursion_stack.rbegin(); i != recursion_stack.rend(); ++i)
1019 {
1020 if(i->idx == static_cast<const re_brace*>(static_cast<const re_jump*>(pstate)->alt.p)->index)
1021 {
1022 if(i->location_of_start == position)
1023 return false;
1024 break;
1025 }
1026 }
1027 //
1028 // Backup call stack:
1029 //
1030 push_recursion_pop();
1031 //
1032 // Set new call stack:
1033 //
1034 if(recursion_stack.capacity() == 0)
1035 {
1036 recursion_stack.reserve(50);
1037 }
1038 recursion_stack.push_back(recursion_info<results_type>());
1039 recursion_stack.back().preturn_address = pstate->next.p;
1040 recursion_stack.back().results = *m_presult;
1041 pstate = static_cast<const re_jump*>(pstate)->alt.p;
1042 recursion_stack.back().idx = static_cast<const re_brace*>(pstate)->index;
1043 recursion_stack.back().location_of_start = position;
1044 //if(static_cast<const re_recurse*>(pstate)->state_id > 0)
1045 {
1046 push_repeater_count(-(2 + static_cast<const re_brace*>(pstate)->index), &next_count);
1047 }
1048
1049 return true;
1050 }
1051
1052 template <class BidiIterator, class Allocator, class traits>
match_endmark()1053 bool perl_matcher<BidiIterator, Allocator, traits>::match_endmark()
1054 {
1055 int index = static_cast<const re_brace*>(pstate)->index;
1056 icase = static_cast<const re_brace*>(pstate)->icase;
1057 if(index > 0)
1058 {
1059 if((m_match_flags & match_nosubs) == 0)
1060 {
1061 m_presult->set_second(position, index);
1062 }
1063 if(!recursion_stack.empty())
1064 {
1065 if(index == recursion_stack.back().idx)
1066 {
1067 pstate = recursion_stack.back().preturn_address;
1068 *m_presult = recursion_stack.back().results;
1069 push_recursion(recursion_stack.back().idx, recursion_stack.back().preturn_address, m_presult, &recursion_stack.back().results);
1070 recursion_stack.pop_back();
1071 push_repeater_count(-(2 + index), &next_count);
1072 }
1073 }
1074 }
1075 else if((index < 0) && (index != -4))
1076 {
1077 // matched forward lookahead:
1078 pstate = 0;
1079 return true;
1080 }
1081 pstate = pstate->next.p;
1082 return true;
1083 }
1084
1085 template <class BidiIterator, class Allocator, class traits>
match_match()1086 bool perl_matcher<BidiIterator, Allocator, traits>::match_match()
1087 {
1088 if(!recursion_stack.empty())
1089 {
1090 BOOST_ASSERT(0 == recursion_stack.back().idx);
1091 pstate = recursion_stack.back().preturn_address;
1092 push_recursion(recursion_stack.back().idx, recursion_stack.back().preturn_address, m_presult, &recursion_stack.back().results);
1093 *m_presult = recursion_stack.back().results;
1094 recursion_stack.pop_back();
1095 return true;
1096 }
1097 if((m_match_flags & match_not_null) && (position == (*m_presult)[0].first))
1098 return false;
1099 if((m_match_flags & match_all) && (position != last))
1100 return false;
1101 if((m_match_flags & regex_constants::match_not_initial_null) && (position == search_base))
1102 return false;
1103 m_presult->set_second(position);
1104 pstate = 0;
1105 m_has_found_match = true;
1106 if((m_match_flags & match_posix) == match_posix)
1107 {
1108 m_result.maybe_assign(*m_presult);
1109 if((m_match_flags & match_any) == 0)
1110 return false;
1111 }
1112 #ifdef BOOST_REGEX_MATCH_EXTRA
1113 if(match_extra & m_match_flags)
1114 {
1115 for(unsigned i = 0; i < m_presult->size(); ++i)
1116 if((*m_presult)[i].matched)
1117 ((*m_presult)[i]).get_captures().push_back((*m_presult)[i]);
1118 }
1119 #endif
1120 return true;
1121 }
1122
1123 template <class BidiIterator, class Allocator, class traits>
match_commit()1124 bool perl_matcher<BidiIterator, Allocator, traits>::match_commit()
1125 {
1126 // Ideally we would just junk all the states that are on the stack,
1127 // however we might not unwind correctly in that case, so for now,
1128 // just mark that we don't backtrack into whatever is left (or rather
1129 // we'll unwind it unconditionally without pausing to try other matches).
1130
1131 switch(static_cast<const re_commit*>(pstate)->action)
1132 {
1133 case commit_commit:
1134 restart = last;
1135 break;
1136 case commit_skip:
1137 if(base != position)
1138 {
1139 restart = position;
1140 // Have to decrement restart since it will get incremented again later:
1141 --restart;
1142 }
1143 break;
1144 case commit_prune:
1145 break;
1146 }
1147
1148 saved_state* pmp = m_backup_state;
1149 --pmp;
1150 if(pmp < m_stack_base)
1151 {
1152 extend_stack();
1153 pmp = m_backup_state;
1154 --pmp;
1155 }
1156 (void) new (pmp)saved_state(16);
1157 m_backup_state = pmp;
1158 pstate = pstate->next.p;
1159 return true;
1160 }
1161
1162 template <class BidiIterator, class Allocator, class traits>
match_then()1163 bool perl_matcher<BidiIterator, Allocator, traits>::match_then()
1164 {
1165 // Just leave a mark that we need to skip to next alternative:
1166 saved_state* pmp = m_backup_state;
1167 --pmp;
1168 if(pmp < m_stack_base)
1169 {
1170 extend_stack();
1171 pmp = m_backup_state;
1172 --pmp;
1173 }
1174 (void) new (pmp)saved_state(17);
1175 m_backup_state = pmp;
1176 pstate = pstate->next.p;
1177 return true;
1178 }
1179
1180 template <class BidiIterator, class Allocator, class traits>
skip_until_paren(int index,bool have_match)1181 bool perl_matcher<BidiIterator, Allocator, traits>::skip_until_paren(int index, bool have_match)
1182 {
1183 while(pstate)
1184 {
1185 if(pstate->type == syntax_element_endmark)
1186 {
1187 if(static_cast<const re_brace*>(pstate)->index == index)
1188 {
1189 if(have_match)
1190 return this->match_endmark();
1191 pstate = pstate->next.p;
1192 return true;
1193 }
1194 else
1195 {
1196 // Unenclosed closing ), occurs when (*ACCEPT) is inside some other
1197 // parenthesis which may or may not have other side effects associated with it.
1198 const re_syntax_base* sp = pstate;
1199 match_endmark();
1200 if(!pstate)
1201 {
1202 unwind(true);
1203 // unwind may leave pstate NULL if we've unwound a forward lookahead, in which
1204 // case just move to the next state and keep looking...
1205 if (!pstate)
1206 pstate = sp->next.p;
1207 }
1208 }
1209 continue;
1210 }
1211 else if(pstate->type == syntax_element_match)
1212 return true;
1213 else if(pstate->type == syntax_element_startmark)
1214 {
1215 int idx = static_cast<const re_brace*>(pstate)->index;
1216 pstate = pstate->next.p;
1217 skip_until_paren(idx, false);
1218 continue;
1219 }
1220 pstate = pstate->next.p;
1221 }
1222 return true;
1223 }
1224
1225 /****************************************************************************
1226
1227 Unwind and associated proceedures follow, these perform what normal stack
1228 unwinding does in the recursive implementation.
1229
1230 ****************************************************************************/
1231
1232 template <class BidiIterator, class Allocator, class traits>
unwind(bool have_match)1233 bool perl_matcher<BidiIterator, Allocator, traits>::unwind(bool have_match)
1234 {
1235 static unwind_proc_type const s_unwind_table[19] =
1236 {
1237 &perl_matcher<BidiIterator, Allocator, traits>::unwind_end,
1238 &perl_matcher<BidiIterator, Allocator, traits>::unwind_paren,
1239 &perl_matcher<BidiIterator, Allocator, traits>::unwind_recursion_stopper,
1240 &perl_matcher<BidiIterator, Allocator, traits>::unwind_assertion,
1241 &perl_matcher<BidiIterator, Allocator, traits>::unwind_alt,
1242 &perl_matcher<BidiIterator, Allocator, traits>::unwind_repeater_counter,
1243 &perl_matcher<BidiIterator, Allocator, traits>::unwind_extra_block,
1244 &perl_matcher<BidiIterator, Allocator, traits>::unwind_greedy_single_repeat,
1245 &perl_matcher<BidiIterator, Allocator, traits>::unwind_slow_dot_repeat,
1246 &perl_matcher<BidiIterator, Allocator, traits>::unwind_fast_dot_repeat,
1247 &perl_matcher<BidiIterator, Allocator, traits>::unwind_char_repeat,
1248 &perl_matcher<BidiIterator, Allocator, traits>::unwind_short_set_repeat,
1249 &perl_matcher<BidiIterator, Allocator, traits>::unwind_long_set_repeat,
1250 &perl_matcher<BidiIterator, Allocator, traits>::unwind_non_greedy_repeat,
1251 &perl_matcher<BidiIterator, Allocator, traits>::unwind_recursion,
1252 &perl_matcher<BidiIterator, Allocator, traits>::unwind_recursion_pop,
1253 &perl_matcher<BidiIterator, Allocator, traits>::unwind_commit,
1254 &perl_matcher<BidiIterator, Allocator, traits>::unwind_then,
1255 &perl_matcher<BidiIterator, Allocator, traits>::unwind_case,
1256 };
1257
1258 m_recursive_result = have_match;
1259 m_unwound_lookahead = false;
1260 m_unwound_alt = false;
1261 unwind_proc_type unwinder;
1262 bool cont;
1263 //
1264 // keep unwinding our stack until we have something to do:
1265 //
1266 do
1267 {
1268 unwinder = s_unwind_table[m_backup_state->state_id];
1269 cont = (this->*unwinder)(m_recursive_result);
1270 }while(cont);
1271 //
1272 // return true if we have more states to try:
1273 //
1274 return pstate ? true : false;
1275 }
1276
1277 template <class BidiIterator, class Allocator, class traits>
unwind_end(bool)1278 bool perl_matcher<BidiIterator, Allocator, traits>::unwind_end(bool)
1279 {
1280 pstate = 0; // nothing left to search
1281 return false; // end of stack nothing more to search
1282 }
1283
1284 template <class BidiIterator, class Allocator, class traits>
unwind_case(bool)1285 bool perl_matcher<BidiIterator, Allocator, traits>::unwind_case(bool)
1286 {
1287 saved_change_case* pmp = static_cast<saved_change_case*>(m_backup_state);
1288 icase = pmp->icase;
1289 boost::BOOST_REGEX_DETAIL_NS::inplace_destroy(pmp++);
1290 m_backup_state = pmp;
1291 return true;
1292 }
1293
1294 template <class BidiIterator, class Allocator, class traits>
unwind_paren(bool have_match)1295 bool perl_matcher<BidiIterator, Allocator, traits>::unwind_paren(bool have_match)
1296 {
1297 saved_matched_paren<BidiIterator>* pmp = static_cast<saved_matched_paren<BidiIterator>*>(m_backup_state);
1298 // restore previous values if no match was found:
1299 if(have_match == false)
1300 {
1301 m_presult->set_first(pmp->sub.first, pmp->index, pmp->index == 0);
1302 m_presult->set_second(pmp->sub.second, pmp->index, pmp->sub.matched, pmp->index == 0);
1303 }
1304 #ifdef BOOST_REGEX_MATCH_EXTRA
1305 //
1306 // we have a match, push the capture information onto the stack:
1307 //
1308 else if(pmp->sub.matched && (match_extra & m_match_flags))
1309 ((*m_presult)[pmp->index]).get_captures().push_back(pmp->sub);
1310 #endif
1311 // unwind stack:
1312 m_backup_state = pmp+1;
1313 boost::BOOST_REGEX_DETAIL_NS::inplace_destroy(pmp);
1314 return true; // keep looking
1315 }
1316
1317 template <class BidiIterator, class Allocator, class traits>
unwind_recursion_stopper(bool)1318 bool perl_matcher<BidiIterator, Allocator, traits>::unwind_recursion_stopper(bool)
1319 {
1320 boost::BOOST_REGEX_DETAIL_NS::inplace_destroy(m_backup_state++);
1321 pstate = 0; // nothing left to search
1322 return false; // end of stack nothing more to search
1323 }
1324
1325 template <class BidiIterator, class Allocator, class traits>
unwind_assertion(bool r)1326 bool perl_matcher<BidiIterator, Allocator, traits>::unwind_assertion(bool r)
1327 {
1328 saved_assertion<BidiIterator>* pmp = static_cast<saved_assertion<BidiIterator>*>(m_backup_state);
1329 pstate = pmp->pstate;
1330 position = pmp->position;
1331 bool result = (r == pmp->positive);
1332 m_recursive_result = pmp->positive ? r : !r;
1333 boost::BOOST_REGEX_DETAIL_NS::inplace_destroy(pmp++);
1334 m_backup_state = pmp;
1335 m_unwound_lookahead = true;
1336 return !result; // return false if the assertion was matched to stop search.
1337 }
1338
1339 template <class BidiIterator, class Allocator, class traits>
unwind_alt(bool r)1340 bool perl_matcher<BidiIterator, Allocator, traits>::unwind_alt(bool r)
1341 {
1342 saved_position<BidiIterator>* pmp = static_cast<saved_position<BidiIterator>*>(m_backup_state);
1343 if(!r)
1344 {
1345 pstate = pmp->pstate;
1346 position = pmp->position;
1347 }
1348 boost::BOOST_REGEX_DETAIL_NS::inplace_destroy(pmp++);
1349 m_backup_state = pmp;
1350 m_unwound_alt = !r;
1351 return r;
1352 }
1353
1354 template <class BidiIterator, class Allocator, class traits>
unwind_repeater_counter(bool)1355 bool perl_matcher<BidiIterator, Allocator, traits>::unwind_repeater_counter(bool)
1356 {
1357 saved_repeater<BidiIterator>* pmp = static_cast<saved_repeater<BidiIterator>*>(m_backup_state);
1358 boost::BOOST_REGEX_DETAIL_NS::inplace_destroy(pmp++);
1359 m_backup_state = pmp;
1360 return true; // keep looking
1361 }
1362
1363 template <class BidiIterator, class Allocator, class traits>
unwind_extra_block(bool)1364 bool perl_matcher<BidiIterator, Allocator, traits>::unwind_extra_block(bool)
1365 {
1366 saved_extra_block* pmp = static_cast<saved_extra_block*>(m_backup_state);
1367 void* condemmed = m_stack_base;
1368 m_stack_base = pmp->base;
1369 m_backup_state = pmp->end;
1370 boost::BOOST_REGEX_DETAIL_NS::inplace_destroy(pmp);
1371 put_mem_block(condemmed);
1372 return true; // keep looking
1373 }
1374
1375 template <class BidiIterator, class Allocator, class traits>
destroy_single_repeat()1376 inline void perl_matcher<BidiIterator, Allocator, traits>::destroy_single_repeat()
1377 {
1378 saved_single_repeat<BidiIterator>* p = static_cast<saved_single_repeat<BidiIterator>*>(m_backup_state);
1379 boost::BOOST_REGEX_DETAIL_NS::inplace_destroy(p++);
1380 m_backup_state = p;
1381 }
1382
1383 template <class BidiIterator, class Allocator, class traits>
unwind_greedy_single_repeat(bool r)1384 bool perl_matcher<BidiIterator, Allocator, traits>::unwind_greedy_single_repeat(bool r)
1385 {
1386 saved_single_repeat<BidiIterator>* pmp = static_cast<saved_single_repeat<BidiIterator>*>(m_backup_state);
1387
1388 // if we have a match, just discard this state:
1389 if(r)
1390 {
1391 destroy_single_repeat();
1392 return true;
1393 }
1394
1395 const re_repeat* rep = pmp->rep;
1396 std::size_t count = pmp->count;
1397 BOOST_ASSERT(rep->next.p != 0);
1398 BOOST_ASSERT(rep->alt.p != 0);
1399
1400 count -= rep->min;
1401
1402 if((m_match_flags & match_partial) && (position == last))
1403 m_has_partial_match = true;
1404
1405 BOOST_ASSERT(count);
1406 position = pmp->last_position;
1407
1408 // backtrack till we can skip out:
1409 do
1410 {
1411 --position;
1412 --count;
1413 ++state_count;
1414 }while(count && !can_start(*position, rep->_map, mask_skip));
1415
1416 // if we've hit base, destroy this state:
1417 if(count == 0)
1418 {
1419 destroy_single_repeat();
1420 if(!can_start(*position, rep->_map, mask_skip))
1421 return true;
1422 }
1423 else
1424 {
1425 pmp->count = count + rep->min;
1426 pmp->last_position = position;
1427 }
1428 pstate = rep->alt.p;
1429 return false;
1430 }
1431
1432 template <class BidiIterator, class Allocator, class traits>
unwind_slow_dot_repeat(bool r)1433 bool perl_matcher<BidiIterator, Allocator, traits>::unwind_slow_dot_repeat(bool r)
1434 {
1435 saved_single_repeat<BidiIterator>* pmp = static_cast<saved_single_repeat<BidiIterator>*>(m_backup_state);
1436
1437 // if we have a match, just discard this state:
1438 if(r)
1439 {
1440 destroy_single_repeat();
1441 return true;
1442 }
1443
1444 const re_repeat* rep = pmp->rep;
1445 std::size_t count = pmp->count;
1446 BOOST_ASSERT(rep->type == syntax_element_dot_rep);
1447 BOOST_ASSERT(rep->next.p != 0);
1448 BOOST_ASSERT(rep->alt.p != 0);
1449 BOOST_ASSERT(rep->next.p->type == syntax_element_wild);
1450
1451 BOOST_ASSERT(count < rep->max);
1452 pstate = rep->next.p;
1453 position = pmp->last_position;
1454
1455 if(position != last)
1456 {
1457 // wind forward until we can skip out of the repeat:
1458 do
1459 {
1460 if(!match_wild())
1461 {
1462 // failed repeat match, discard this state and look for another:
1463 destroy_single_repeat();
1464 return true;
1465 }
1466 ++count;
1467 ++state_count;
1468 pstate = rep->next.p;
1469 }while((count < rep->max) && (position != last) && !can_start(*position, rep->_map, mask_skip));
1470 }
1471 if(position == last)
1472 {
1473 // can't repeat any more, remove the pushed state:
1474 destroy_single_repeat();
1475 if((m_match_flags & match_partial) && (position == last) && (position != search_base))
1476 m_has_partial_match = true;
1477 if(0 == (rep->can_be_null & mask_skip))
1478 return true;
1479 }
1480 else if(count == rep->max)
1481 {
1482 // can't repeat any more, remove the pushed state:
1483 destroy_single_repeat();
1484 if(!can_start(*position, rep->_map, mask_skip))
1485 return true;
1486 }
1487 else
1488 {
1489 pmp->count = count;
1490 pmp->last_position = position;
1491 }
1492 pstate = rep->alt.p;
1493 return false;
1494 }
1495
1496 template <class BidiIterator, class Allocator, class traits>
unwind_fast_dot_repeat(bool r)1497 bool perl_matcher<BidiIterator, Allocator, traits>::unwind_fast_dot_repeat(bool r)
1498 {
1499 saved_single_repeat<BidiIterator>* pmp = static_cast<saved_single_repeat<BidiIterator>*>(m_backup_state);
1500
1501 // if we have a match, just discard this state:
1502 if(r)
1503 {
1504 destroy_single_repeat();
1505 return true;
1506 }
1507
1508 const re_repeat* rep = pmp->rep;
1509 std::size_t count = pmp->count;
1510
1511 BOOST_ASSERT(count < rep->max);
1512 position = pmp->last_position;
1513 if(position != last)
1514 {
1515
1516 // wind forward until we can skip out of the repeat:
1517 do
1518 {
1519 ++position;
1520 ++count;
1521 ++state_count;
1522 }while((count < rep->max) && (position != last) && !can_start(*position, rep->_map, mask_skip));
1523 }
1524
1525 // remember where we got to if this is a leading repeat:
1526 if((rep->leading) && (count < rep->max))
1527 restart = position;
1528 if(position == last)
1529 {
1530 // can't repeat any more, remove the pushed state:
1531 destroy_single_repeat();
1532 if((m_match_flags & match_partial) && (position == last) && (position != search_base))
1533 m_has_partial_match = true;
1534 if(0 == (rep->can_be_null & mask_skip))
1535 return true;
1536 }
1537 else if(count == rep->max)
1538 {
1539 // can't repeat any more, remove the pushed state:
1540 destroy_single_repeat();
1541 if(!can_start(*position, rep->_map, mask_skip))
1542 return true;
1543 }
1544 else
1545 {
1546 pmp->count = count;
1547 pmp->last_position = position;
1548 }
1549 pstate = rep->alt.p;
1550 return false;
1551 }
1552
1553 template <class BidiIterator, class Allocator, class traits>
unwind_char_repeat(bool r)1554 bool perl_matcher<BidiIterator, Allocator, traits>::unwind_char_repeat(bool r)
1555 {
1556 saved_single_repeat<BidiIterator>* pmp = static_cast<saved_single_repeat<BidiIterator>*>(m_backup_state);
1557
1558 // if we have a match, just discard this state:
1559 if(r)
1560 {
1561 destroy_single_repeat();
1562 return true;
1563 }
1564
1565 const re_repeat* rep = pmp->rep;
1566 std::size_t count = pmp->count;
1567 pstate = rep->next.p;
1568 const char_type what = *reinterpret_cast<const char_type*>(static_cast<const re_literal*>(pstate) + 1);
1569 position = pmp->last_position;
1570
1571 BOOST_ASSERT(rep->type == syntax_element_char_rep);
1572 BOOST_ASSERT(rep->next.p != 0);
1573 BOOST_ASSERT(rep->alt.p != 0);
1574 BOOST_ASSERT(rep->next.p->type == syntax_element_literal);
1575 BOOST_ASSERT(count < rep->max);
1576
1577 if(position != last)
1578 {
1579 // wind forward until we can skip out of the repeat:
1580 do
1581 {
1582 if(traits_inst.translate(*position, icase) != what)
1583 {
1584 // failed repeat match, discard this state and look for another:
1585 destroy_single_repeat();
1586 return true;
1587 }
1588 ++count;
1589 ++ position;
1590 ++state_count;
1591 pstate = rep->next.p;
1592 }while((count < rep->max) && (position != last) && !can_start(*position, rep->_map, mask_skip));
1593 }
1594 // remember where we got to if this is a leading repeat:
1595 if((rep->leading) && (count < rep->max))
1596 restart = position;
1597 if(position == last)
1598 {
1599 // can't repeat any more, remove the pushed state:
1600 destroy_single_repeat();
1601 if((m_match_flags & match_partial) && (position == last) && (position != search_base))
1602 m_has_partial_match = true;
1603 if(0 == (rep->can_be_null & mask_skip))
1604 return true;
1605 }
1606 else if(count == rep->max)
1607 {
1608 // can't repeat any more, remove the pushed state:
1609 destroy_single_repeat();
1610 if(!can_start(*position, rep->_map, mask_skip))
1611 return true;
1612 }
1613 else
1614 {
1615 pmp->count = count;
1616 pmp->last_position = position;
1617 }
1618 pstate = rep->alt.p;
1619 return false;
1620 }
1621
1622 template <class BidiIterator, class Allocator, class traits>
unwind_short_set_repeat(bool r)1623 bool perl_matcher<BidiIterator, Allocator, traits>::unwind_short_set_repeat(bool r)
1624 {
1625 saved_single_repeat<BidiIterator>* pmp = static_cast<saved_single_repeat<BidiIterator>*>(m_backup_state);
1626
1627 // if we have a match, just discard this state:
1628 if(r)
1629 {
1630 destroy_single_repeat();
1631 return true;
1632 }
1633
1634 const re_repeat* rep = pmp->rep;
1635 std::size_t count = pmp->count;
1636 pstate = rep->next.p;
1637 const unsigned char* map = static_cast<const re_set*>(rep->next.p)->_map;
1638 position = pmp->last_position;
1639
1640 BOOST_ASSERT(rep->type == syntax_element_short_set_rep);
1641 BOOST_ASSERT(rep->next.p != 0);
1642 BOOST_ASSERT(rep->alt.p != 0);
1643 BOOST_ASSERT(rep->next.p->type == syntax_element_set);
1644 BOOST_ASSERT(count < rep->max);
1645
1646 if(position != last)
1647 {
1648 // wind forward until we can skip out of the repeat:
1649 do
1650 {
1651 if(!map[static_cast<unsigned char>(traits_inst.translate(*position, icase))])
1652 {
1653 // failed repeat match, discard this state and look for another:
1654 destroy_single_repeat();
1655 return true;
1656 }
1657 ++count;
1658 ++ position;
1659 ++state_count;
1660 pstate = rep->next.p;
1661 }while((count < rep->max) && (position != last) && !can_start(*position, rep->_map, mask_skip));
1662 }
1663 // remember where we got to if this is a leading repeat:
1664 if((rep->leading) && (count < rep->max))
1665 restart = position;
1666 if(position == last)
1667 {
1668 // can't repeat any more, remove the pushed state:
1669 destroy_single_repeat();
1670 if((m_match_flags & match_partial) && (position == last) && (position != search_base))
1671 m_has_partial_match = true;
1672 if(0 == (rep->can_be_null & mask_skip))
1673 return true;
1674 }
1675 else if(count == rep->max)
1676 {
1677 // can't repeat any more, remove the pushed state:
1678 destroy_single_repeat();
1679 if(!can_start(*position, rep->_map, mask_skip))
1680 return true;
1681 }
1682 else
1683 {
1684 pmp->count = count;
1685 pmp->last_position = position;
1686 }
1687 pstate = rep->alt.p;
1688 return false;
1689 }
1690
1691 template <class BidiIterator, class Allocator, class traits>
unwind_long_set_repeat(bool r)1692 bool perl_matcher<BidiIterator, Allocator, traits>::unwind_long_set_repeat(bool r)
1693 {
1694 typedef typename traits::char_class_type m_type;
1695 saved_single_repeat<BidiIterator>* pmp = static_cast<saved_single_repeat<BidiIterator>*>(m_backup_state);
1696
1697 // if we have a match, just discard this state:
1698 if(r)
1699 {
1700 destroy_single_repeat();
1701 return true;
1702 }
1703
1704 const re_repeat* rep = pmp->rep;
1705 std::size_t count = pmp->count;
1706 pstate = rep->next.p;
1707 const re_set_long<m_type>* set = static_cast<const re_set_long<m_type>*>(pstate);
1708 position = pmp->last_position;
1709
1710 BOOST_ASSERT(rep->type == syntax_element_long_set_rep);
1711 BOOST_ASSERT(rep->next.p != 0);
1712 BOOST_ASSERT(rep->alt.p != 0);
1713 BOOST_ASSERT(rep->next.p->type == syntax_element_long_set);
1714 BOOST_ASSERT(count < rep->max);
1715
1716 if(position != last)
1717 {
1718 // wind forward until we can skip out of the repeat:
1719 do
1720 {
1721 if(position == re_is_set_member(position, last, set, re.get_data(), icase))
1722 {
1723 // failed repeat match, discard this state and look for another:
1724 destroy_single_repeat();
1725 return true;
1726 }
1727 ++position;
1728 ++count;
1729 ++state_count;
1730 pstate = rep->next.p;
1731 }while((count < rep->max) && (position != last) && !can_start(*position, rep->_map, mask_skip));
1732 }
1733 // remember where we got to if this is a leading repeat:
1734 if((rep->leading) && (count < rep->max))
1735 restart = position;
1736 if(position == last)
1737 {
1738 // can't repeat any more, remove the pushed state:
1739 destroy_single_repeat();
1740 if((m_match_flags & match_partial) && (position == last) && (position != search_base))
1741 m_has_partial_match = true;
1742 if(0 == (rep->can_be_null & mask_skip))
1743 return true;
1744 }
1745 else if(count == rep->max)
1746 {
1747 // can't repeat any more, remove the pushed state:
1748 destroy_single_repeat();
1749 if(!can_start(*position, rep->_map, mask_skip))
1750 return true;
1751 }
1752 else
1753 {
1754 pmp->count = count;
1755 pmp->last_position = position;
1756 }
1757 pstate = rep->alt.p;
1758 return false;
1759 }
1760
1761 template <class BidiIterator, class Allocator, class traits>
unwind_non_greedy_repeat(bool r)1762 bool perl_matcher<BidiIterator, Allocator, traits>::unwind_non_greedy_repeat(bool r)
1763 {
1764 saved_position<BidiIterator>* pmp = static_cast<saved_position<BidiIterator>*>(m_backup_state);
1765 if(!r)
1766 {
1767 position = pmp->position;
1768 pstate = pmp->pstate;
1769 ++(*next_count);
1770 }
1771 boost::BOOST_REGEX_DETAIL_NS::inplace_destroy(pmp++);
1772 m_backup_state = pmp;
1773 return r;
1774 }
1775
1776 template <class BidiIterator, class Allocator, class traits>
unwind_recursion(bool r)1777 bool perl_matcher<BidiIterator, Allocator, traits>::unwind_recursion(bool r)
1778 {
1779 // We are backtracking back inside a recursion, need to push the info
1780 // back onto the recursion stack, and do so unconditionally, otherwise
1781 // we can get mismatched pushes and pops...
1782 saved_recursion<results_type>* pmp = static_cast<saved_recursion<results_type>*>(m_backup_state);
1783 if (!r)
1784 {
1785 recursion_stack.push_back(recursion_info<results_type>());
1786 recursion_stack.back().idx = pmp->recursion_id;
1787 recursion_stack.back().preturn_address = pmp->preturn_address;
1788 recursion_stack.back().results = pmp->prior_results;
1789 recursion_stack.back().location_of_start = position;
1790 *m_presult = pmp->internal_results;
1791 }
1792 boost::BOOST_REGEX_DETAIL_NS::inplace_destroy(pmp++);
1793 m_backup_state = pmp;
1794 return true;
1795 }
1796
1797 template <class BidiIterator, class Allocator, class traits>
unwind_recursion_pop(bool r)1798 bool perl_matcher<BidiIterator, Allocator, traits>::unwind_recursion_pop(bool r)
1799 {
1800 // Backtracking out of a recursion, we must pop state off the recursion
1801 // stack unconditionally to ensure matched pushes and pops:
1802 saved_state* pmp = static_cast<saved_state*>(m_backup_state);
1803 if (!r && !recursion_stack.empty())
1804 {
1805 *m_presult = recursion_stack.back().results;
1806 position = recursion_stack.back().location_of_start;
1807 recursion_stack.pop_back();
1808 }
1809 boost::BOOST_REGEX_DETAIL_NS::inplace_destroy(pmp++);
1810 m_backup_state = pmp;
1811 return true;
1812 }
1813
1814 template <class BidiIterator, class Allocator, class traits>
push_recursion_pop()1815 void perl_matcher<BidiIterator, Allocator, traits>::push_recursion_pop()
1816 {
1817 saved_state* pmp = static_cast<saved_state*>(m_backup_state);
1818 --pmp;
1819 if(pmp < m_stack_base)
1820 {
1821 extend_stack();
1822 pmp = static_cast<saved_state*>(m_backup_state);
1823 --pmp;
1824 }
1825 (void) new (pmp)saved_state(15);
1826 m_backup_state = pmp;
1827 }
1828
1829 template <class BidiIterator, class Allocator, class traits>
unwind_commit(bool b)1830 bool perl_matcher<BidiIterator, Allocator, traits>::unwind_commit(bool b)
1831 {
1832 boost::BOOST_REGEX_DETAIL_NS::inplace_destroy(m_backup_state++);
1833 while(unwind(b) && !m_unwound_lookahead){}
1834 if(m_unwound_lookahead && pstate)
1835 {
1836 //
1837 // If we stop because we just unwound an assertion, put the
1838 // commit state back on the stack again:
1839 //
1840 m_unwound_lookahead = false;
1841 saved_state* pmp = m_backup_state;
1842 --pmp;
1843 if(pmp < m_stack_base)
1844 {
1845 extend_stack();
1846 pmp = m_backup_state;
1847 --pmp;
1848 }
1849 (void) new (pmp)saved_state(16);
1850 m_backup_state = pmp;
1851 }
1852 // This prevents us from stopping when we exit from an independent sub-expression:
1853 m_independent = false;
1854 return false;
1855 }
1856
1857 template <class BidiIterator, class Allocator, class traits>
unwind_then(bool b)1858 bool perl_matcher<BidiIterator, Allocator, traits>::unwind_then(bool b)
1859 {
1860 // Unwind everything till we hit an alternative:
1861 boost::BOOST_REGEX_DETAIL_NS::inplace_destroy(m_backup_state++);
1862 bool result = false;
1863 while((result = unwind(b)) && !m_unwound_alt){}
1864 // We're now pointing at the next alternative, need one more backtrack
1865 // since *all* the other alternatives must fail once we've reached a THEN clause:
1866 if(result && m_unwound_alt)
1867 unwind(b);
1868 return false;
1869 }
1870
1871 /*
1872 template <class BidiIterator, class Allocator, class traits>
1873 bool perl_matcher<BidiIterator, Allocator, traits>::unwind_parenthesis_pop(bool r)
1874 {
1875 saved_state* pmp = static_cast<saved_state*>(m_backup_state);
1876 if(!r)
1877 {
1878 --parenthesis_stack_position;
1879 }
1880 boost::BOOST_REGEX_DETAIL_NS::inplace_destroy(pmp++);
1881 m_backup_state = pmp;
1882 return true;
1883 }
1884
1885 template <class BidiIterator, class Allocator, class traits>
1886 void perl_matcher<BidiIterator, Allocator, traits>::push_parenthesis_pop()
1887 {
1888 saved_state* pmp = static_cast<saved_state*>(m_backup_state);
1889 --pmp;
1890 if(pmp < m_stack_base)
1891 {
1892 extend_stack();
1893 pmp = static_cast<saved_state*>(m_backup_state);
1894 --pmp;
1895 }
1896 (void) new (pmp)saved_state(16);
1897 m_backup_state = pmp;
1898 }
1899
1900 template <class BidiIterator, class Allocator, class traits>
1901 bool perl_matcher<BidiIterator, Allocator, traits>::unwind_parenthesis_push(bool r)
1902 {
1903 saved_position<BidiIterator>* pmp = static_cast<saved_position<BidiIterator>*>(m_backup_state);
1904 if(!r)
1905 {
1906 parenthesis_stack[parenthesis_stack_position++] = pmp->position;
1907 }
1908 boost::BOOST_REGEX_DETAIL_NS::inplace_destroy(pmp++);
1909 m_backup_state = pmp;
1910 return true;
1911 }
1912
1913 template <class BidiIterator, class Allocator, class traits>
1914 inline void perl_matcher<BidiIterator, Allocator, traits>::push_parenthesis_push(BidiIterator p)
1915 {
1916 saved_position<BidiIterator>* pmp = static_cast<saved_position<BidiIterator>*>(m_backup_state);
1917 --pmp;
1918 if(pmp < m_stack_base)
1919 {
1920 extend_stack();
1921 pmp = static_cast<saved_position<BidiIterator>*>(m_backup_state);
1922 --pmp;
1923 }
1924 (void) new (pmp)saved_position<BidiIterator>(0, p, 17);
1925 m_backup_state = pmp;
1926 }
1927 */
1928 } // namespace BOOST_REGEX_DETAIL_NS
1929 } // namespace boost
1930
1931 #ifdef BOOST_MSVC
1932 # pragma warning(pop)
1933 #endif
1934
1935 #ifdef BOOST_MSVC
1936 #pragma warning(push)
1937 #pragma warning(disable: 4103)
1938 #endif
1939 #ifdef BOOST_HAS_ABI_HEADERS
1940 # include BOOST_ABI_SUFFIX
1941 #endif
1942 #ifdef BOOST_MSVC
1943 #pragma warning(pop)
1944 #endif
1945
1946 #endif
1947
1948
1949