2014-10-11 15:26:29 +04:00
|
|
|
// Copyright 2007-2009 Russ Cox. All Rights Reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
2014-10-15 05:43:13 +04:00
|
|
|
#include "re1.5.h"
|
2014-10-11 15:26:29 +04:00
|
|
|
|
|
|
|
static int
|
|
|
|
recursiveloop(char *pc, const char *sp, Subject *input, const char **subp, int nsubp)
|
|
|
|
{
|
|
|
|
const char *old;
|
|
|
|
int off;
|
2017-10-02 21:20:47 +03:00
|
|
|
|
|
|
|
re1_5_stack_chk();
|
|
|
|
|
2014-10-11 15:26:29 +04:00
|
|
|
for(;;) {
|
|
|
|
if(inst_is_consumer(*pc)) {
|
|
|
|
// If we need to match a character, but there's none left, it's fail
|
|
|
|
if(sp >= input->end)
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
switch(*pc++) {
|
|
|
|
case Char:
|
|
|
|
if(*sp != *pc++)
|
|
|
|
return 0;
|
2019-11-28 14:47:21 +03:00
|
|
|
MP_FALLTHROUGH
|
2014-10-11 15:26:29 +04:00
|
|
|
case Any:
|
|
|
|
sp++;
|
|
|
|
continue;
|
2014-10-15 05:43:13 +04:00
|
|
|
case Class:
|
2014-10-16 14:56:13 +04:00
|
|
|
case ClassNot:
|
2014-10-15 05:43:13 +04:00
|
|
|
if (!_re1_5_classmatch(pc, sp))
|
|
|
|
return 0;
|
|
|
|
pc += *(unsigned char*)pc * 2 + 1;
|
|
|
|
sp++;
|
|
|
|
continue;
|
2015-02-21 13:33:20 +03:00
|
|
|
case NamedClass:
|
|
|
|
if (!_re1_5_namedclassmatch(pc, sp))
|
|
|
|
return 0;
|
|
|
|
pc++;
|
|
|
|
sp++;
|
|
|
|
continue;
|
2014-10-11 15:26:29 +04:00
|
|
|
case Match:
|
|
|
|
return 1;
|
|
|
|
case Jmp:
|
|
|
|
off = (signed char)*pc++;
|
|
|
|
pc = pc + off;
|
|
|
|
continue;
|
|
|
|
case Split:
|
|
|
|
off = (signed char)*pc++;
|
|
|
|
if(recursiveloop(pc, sp, input, subp, nsubp))
|
|
|
|
return 1;
|
|
|
|
pc = pc + off;
|
|
|
|
continue;
|
|
|
|
case RSplit:
|
|
|
|
off = (signed char)*pc++;
|
|
|
|
if(recursiveloop(pc + off, sp, input, subp, nsubp))
|
|
|
|
return 1;
|
|
|
|
continue;
|
|
|
|
case Save:
|
|
|
|
off = (unsigned char)*pc++;
|
|
|
|
if(off >= nsubp) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
old = subp[off];
|
|
|
|
subp[off] = sp;
|
|
|
|
if(recursiveloop(pc, sp, input, subp, nsubp))
|
|
|
|
return 1;
|
|
|
|
subp[off] = old;
|
|
|
|
return 0;
|
|
|
|
case Bol:
|
|
|
|
if(sp != input->begin)
|
|
|
|
return 0;
|
|
|
|
continue;
|
|
|
|
case Eol:
|
|
|
|
if(sp != input->end)
|
|
|
|
return 0;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
re1_5_fatal("recursiveloop");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
re1_5_recursiveloopprog(ByteProg *prog, Subject *input, const char **subp, int nsubp, int is_anchored)
|
|
|
|
{
|
|
|
|
return recursiveloop(HANDLE_ANCHORED(prog->insts, is_anchored), input->begin, input, subp, nsubp);
|
|
|
|
}
|