diff options
Diffstat (limited to 'lpcode.c')
-rw-r--r-- | lpcode.c | 43 |
1 files changed, 35 insertions, 8 deletions
@@ -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) { | |||
492 | static int addoffsetinst (CompileState *compst, Opcode op) { | 498 | static 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 | |||
725 | static 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; |