aboutsummaryrefslogtreecommitdiff
path: root/lpcode.c
diff options
context:
space:
mode:
Diffstat (limited to 'lpcode.c')
-rw-r--r--lpcode.c43
1 files changed, 35 insertions, 8 deletions
diff --git a/lpcode.c b/lpcode.c
index 50e6764..2987cfc 100644
--- a/lpcode.c
+++ b/lpcode.c
@@ -196,8 +196,8 @@ int checkaux (TTree *tree, int pred) {
196 if (checkaux(sib2(tree), pred)) return 1; 196 if (checkaux(sib2(tree), pred)) return 1;
197 /* else return checkaux(sib1(tree), pred); */ 197 /* else return checkaux(sib1(tree), pred); */
198 tree = sib1(tree); goto tailcall; 198 tree = sib1(tree); goto tailcall;
199 case TLabChoice: /* labeled failure */ 199 case TLabChoice: case TRecov: /* labeled failure */
200 /* in a labeled ordered choice we do not know whether sib2 will be evaluated */ 200 /* we do not know whether sib2 will be evaluated */
201 tree = sib1(tree); goto tailcall; 201 tree = sib1(tree); goto tailcall;
202 case TCapture: case TGrammar: case TRule: 202 case TCapture: case TGrammar: case TRule:
203 /* return checkaux(sib1(tree), pred); */ 203 /* return checkaux(sib1(tree), pred); */
@@ -231,7 +231,7 @@ int fixedlenx (TTree *tree, int count, int len) {
231 return -1; /* may be a loop */ 231 return -1; /* may be a loop */
232 /* else return fixedlenx(sib2(tree), count); */ 232 /* else return fixedlenx(sib2(tree), count); */
233 tree = sib2(tree); goto tailcall; 233 tree = sib2(tree); goto tailcall;
234 case TSeq: { 234 case TSeq: case TRecov: { /* labeled failure */
235 len = fixedlenx(sib1(tree), count, len); 235 len = fixedlenx(sib1(tree), count, len);
236 if (len < 0) return -1; 236 if (len < 0) return -1;
237 /* else return fixedlenx(sib2(tree), count, len); */ 237 /* else return fixedlenx(sib2(tree), count, len); */
@@ -294,6 +294,12 @@ static int getfirst (TTree *tree, const Charset *follow, Charset *firstset) {
294 loopset(i, firstset->cs[i] |= csaux.cs[i]); 294 loopset(i, firstset->cs[i] |= csaux.cs[i]);
295 return e1 | e2; 295 return e1 | e2;
296 } 296 }
297 case TRecov: { /* labeled failure */
298 /* when p1 is not nullable, p2 has nothing to contribute;
299 and when p1 is nullable, then p2 will not match
300 return getfirst(sib1(tree), fullset, firstset); */
301 tree = sib1(tree); follow = fullset; goto tailcall;
302 }
297 case TSeq: { 303 case TSeq: {
298 if (!nullable(sib1(tree))) { 304 if (!nullable(sib1(tree))) {
299 /* when p1 is not nullable, p2 has nothing to contribute; 305 /* when p1 is not nullable, p2 has nothing to contribute;
@@ -372,7 +378,7 @@ static int headfail (TTree *tree) {
372 if (!nofail(sib2(tree))) return 0; 378 if (!nofail(sib2(tree))) return 0;
373 /* else return headfail(sib1(tree)); */ 379 /* else return headfail(sib1(tree)); */
374 tree = sib1(tree); goto tailcall; 380 tree = sib1(tree); goto tailcall;
375 case TChoice: case TLabChoice: /* labeled failure */ 381 case TChoice: case TLabChoice: case TRecov: /* labeled failure */
376 if (!headfail(sib1(tree))) return 0; 382 if (!headfail(sib1(tree))) return 0;
377 /* else return headfail(sib2(tree)); */ 383 /* else return headfail(sib2(tree)); */
378 tree = sib2(tree); goto tailcall; 384 tree = sib2(tree); goto tailcall;
@@ -392,7 +398,7 @@ static int needfollow (TTree *tree) {
392 case TChar: case TSet: case TAny: 398 case TChar: case TSet: case TAny:
393 case TFalse: case TTrue: case TAnd: case TNot: 399 case TFalse: case TTrue: case TAnd: case TNot:
394 case TRunTime: case TGrammar: case TCall: case TBehind: 400 case TRunTime: case TGrammar: case TCall: case TBehind:
395 case TThrow: case TLabChoice: /* (?)labeled failure */ 401 case TThrow: case TLabChoice: case TRecov: /* (?)labeled failure */
396 return 0; 402 return 0;
397 case TChoice: case TRep: 403 case TChoice: case TRep:
398 return 1; 404 return 1;
@@ -427,7 +433,7 @@ int sizei (const Instruction *i) {
427 return 2; 433 return 2;
428 case IThrow: /* labeled failure */ 434 case IThrow: /* labeled failure */
429 return 1; 435 return 1;
430 case ILabChoice: 436 case ILabChoice: case IRecov:
431 return (CHARSETINSTSIZE - 1) + 2; /* labeled failure */ 437 return (CHARSETINSTSIZE - 1) + 2; /* labeled failure */
432 default: return 1; 438 default: return 1;
433 } 439 }
@@ -492,7 +498,8 @@ static int addinstruction (CompileState *compst, Opcode op, int aux) {
492static int addoffsetinst (CompileState *compst, Opcode op) { 498static int addoffsetinst (CompileState *compst, Opcode op) {
493 int i = addinstruction(compst, op, 0); /* instruction */ 499 int i = addinstruction(compst, op, 0); /* instruction */
494 addinstruction(compst, (Opcode)0, 0); /* open space for offset */ 500 addinstruction(compst, (Opcode)0, 0); /* open space for offset */
495 assert(op == ITestSet || sizei(&getinstr(compst, i)) == 2); 501 assert(op == ITestSet || sizei(&getinstr(compst, i)) == 2 ||
502 op == IRecov || op == ILabChoice); /* labeled failure */
496 return i; 503 return i;
497} 504}
498 505
@@ -714,6 +721,21 @@ static void codelabchoice (CompileState *compst, TTree *p1, TTree *p2, int opt,
714 codegen(compst, p2, opt, NOINST, fl); 721 codegen(compst, p2, opt, NOINST, fl);
715 jumptohere(compst, pcommit); 722 jumptohere(compst, pcommit);
716} 723}
724
725static void coderecovery (CompileState *compst, TTree *p1, TTree *p2, int opt,
726 const Charset *fl, const byte *cs) {
727 int emptyp2 = (p2->tag == TTrue);
728 int pcommit;
729 int test = NOINST;
730 int precovery = addoffsetinst(compst, IRecov);
731 addcharset(compst, cs);
732 codegen(compst, p1, emptyp2, test, fullset);
733 pcommit = addoffsetinst(compst, ICommit);
734 jumptohere(compst, precovery);
735 jumptohere(compst, test);
736 codegen(compst, p2, opt, NOINST, fl);
737 jumptohere(compst, pcommit);
738}
717/* labeled failure end */ 739/* labeled failure end */
718 740
719 741
@@ -951,6 +973,10 @@ static void codegen (CompileState *compst, TTree *tree, int opt, int tt,
951 codelabchoice(compst, sib1(tree), sib2(tree), opt, fl, treelabelset(tree)); 973 codelabchoice(compst, sib1(tree), sib2(tree), opt, fl, treelabelset(tree));
952 break; 974 break;
953 } 975 }
976 case TRecov: { /* labeled failure */
977 coderecovery(compst, sib1(tree), sib2(tree), opt, fl, treelabelset(tree));
978 break;
979 }
954 default: assert(0); 980 default: assert(0);
955 } 981 }
956} 982}
@@ -972,7 +998,8 @@ static void peephole (CompileState *compst) {
972 redo: 998 redo:
973 switch (code[i].i.code) { 999 switch (code[i].i.code) {
974 case IChoice: case ICall: case ICommit: case IPartialCommit: 1000 case IChoice: case ICall: case ICommit: case IPartialCommit:
975 case IBackCommit: case ITestChar: case ITestSet: case ILabChoice: /* labeled failure */ 1001 case IBackCommit: case ITestChar: case ITestSet:
1002 case ILabChoice: case IRecov: /* labeled failure */
976 case ITestAny: { /* instructions with labels */ 1003 case ITestAny: { /* instructions with labels */
977 jumptothere(compst, i, finallabel(code, i)); /* optimize label */ 1004 jumptothere(compst, i, finallabel(code, i)); /* optimize label */
978 break; 1005 break;