| : loops> ( -- addr ) -3 loops +! loops @+ swap cells + 3@ ; |
: loops> ( -- addr ) -3 loops +! loops @+ swap cells + 3@ ; |
| : >loops ( addr -- ) loops @+ swap cells + 3! 3 loops +! ; |
: >loops ( addr -- ) loops @+ swap cells + 3! 3 loops +! ; |
| : BEGIN, ( -- ) ]] BEGIN [[ >loops ; |
: BEGIN, ( -- ) ]] BEGIN [[ >loops ; |
| : DONE, ( -- ) loops @ IF loops> ]] DONE [[ THEN ; |
: DONE, ( -- ) loops @ IF loops> ]] DONE [[ ELSE ." no done left!" cr THEN ; |
| |
|
| \ variables |
\ variables |
| |
|
| : var> ( -- addr ) -1 varstack +! |
: var> ( -- addr ) -1 varstack +! |
| varstack @+ swap cells + @ |
varstack @+ swap cells + @ |
| 1+ 2* cells vars + ; |
1+ 2* cells vars + ; |
| |
Variable greed-counts 9 cells allot \ no more than 9 nested greedy loops |
| |
: greed' ( -- addr ) greed-counts dup @ + ; |
| |
|
| \ start end |
\ start end |
| |
|
| 0 Value end$ |
0 Value end$ |
| |
0 Value last$ |
| 0 Value start$ |
0 Value start$ |
| : !end ( addr u -- addr ) over + to end$ dup to start$ ; |
: !end ( addr u -- addr ) over + to end$ dup to start$ ; |
| : end-rex? ( addr -- addr flag ) dup end$ u< ; |
: end-rex? ( addr -- addr flag ) dup end$ u< ; |
| : start-rex? ( addr -- addr flag ) dup start$ u> ; |
: start-rex? ( addr -- addr flag ) dup start$ u> ; |
| : ?end ( addr -- addr ) ]] dup end$ u> ?LEAVE [[ ; immediate |
: ?end ( addr -- addr ) ]] dup end$ u> ?LEAVE [[ ; immediate |
| : rest$ ( addr -- addr addr u ) dup end$ over - ; |
: rest$ ( addr -- addr addr u ) dup end$ over - ; |
| |
: >last ( addr -- flag ) dup to last$ end$ u<= ; |
| |
|
| \ start and end |
\ start and end |
| |
|
| |
|
| : (( ( addr u -- ) \ regexp-pattern |
: (( ( addr u -- ) \ regexp-pattern |
| \G start regexp block |
\G start regexp block |
| vars off varsmax off loops off |
vars off varsmax off loops off greed-counts off |
| ]] FORK AHEAD BUT JOIN !end [[ BEGIN, ; immediate |
]] FORK AHEAD BUT JOIN !end [[ BEGIN, ; immediate |
| : )) ( -- addr f ) \ regexp-pattern |
: )) ( -- flag ) \ regexp-pattern |
| \G end regexp block |
\G end regexp block |
| ]] ?end drop true ;S [[ |
]] >last ;S [[ |
| DONE, ]] drop false ;S THEN [[ ; immediate |
DONE, ]] drop false ;S THEN [[ ; immediate |
| |
|
| \ greedy loops |
\ greedy loops |
| |
|
| : {** ( addr -- addr addr ) \ regexp-pattern |
: {** ( addr -- addr addr ) \ regexp-pattern |
| \G greedy zero-or-more pattern |
\G greedy zero-or-more pattern |
| 0 ]] Literal >r BEGIN dup [[ BEGIN, ; immediate |
]] false >r BEGIN dup FORK BUT WHILE last$ r> 1+ >r REPEAT [[ |
| |
]] r> AHEAD BUT JOIN [[ |
| |
BEGIN, ; immediate |
| ' {** Alias {++ ( addr -- addr addr ) \ regexp-pattern |
' {** Alias {++ ( addr -- addr addr ) \ regexp-pattern |
| \G greedy one-or-more pattern |
\G greedy one-or-more pattern |
| immediate |
immediate |
| : n*} ( sys n -- ) \ regexp-pattern |
|
| \G At least @var{n} pattern |
|
| >r ]] r> 1+ >r end-rex? 0= UNTIL dup [[ DONE, ]] drop [[ |
|
| r@ IF r@ ]] r@ Literal u< IF r> 1+ drops dup LEAVE THEN [[ THEN |
|
| r@ ]] r> 1+ Literal U+DO FORK BUT [[ |
|
| ]] IF I' I - [[ r@ 1- ]] Literal + drops true UNLOOP ;S THEN LOOP [[ |
|
| r@ IF r@ ]] Literal drops [[ THEN |
|
| rdrop ]] dup LEAVE JOIN [[ ; immediate |
|
| : **} ( sys -- ) \ regexp-pattern |
: **} ( sys -- ) \ regexp-pattern |
| \G end of greedy zero-or-more pattern |
\G end of greedy zero-or-more pattern |
| 0 postpone n*} ; immediate |
]] >last ;S [[ DONE, ]] drop false ;S THEN [[ |
| |
]] 1+ false U+DO FORK BUT [[ |
| |
]] IF I' I - 1- drops UNLOOP true ;S THEN LOOP [[ |
| |
]] false ;S JOIN [[ ; immediate |
| : ++} ( sys -- ) \ regexp-pattern |
: ++} ( sys -- ) \ regexp-pattern |
| \G end of greedy zero-or-more pattern |
\G end of greedy zero-or-more pattern |
| 1 postpone n*} ; immediate |
]] >last ;S [[ DONE, ]] drop false ;S THEN [[ |
| |
]] false U+DO FORK BUT [[ |
| |
]] IF I' I - drops UNLOOP true ;S THEN LOOP [[ |
| |
]] drop false ;S JOIN [[ ; immediate |
| |
|
| \ non-greedy loops |
\ non-greedy loops |
| |
|
| : +} ( addr addr' -- addr' ) \ regexp-pattern |
: +} ( addr addr' -- addr' ) \ regexp-pattern |
| \G end of non-greedy one-or-more pattern |
\G end of non-greedy one-or-more pattern |
| ]] dup FORK BUT IF drop true ;S [[ |
]] dup FORK BUT IF drop true ;S [[ |
| DONE, ]] drop dup LEAVE THEN *} [[ ; immediate |
DONE, ]] drop false ;S [[ BEGIN, ]] THEN *} [[ ; immediate |
| |
|
| : // ( -- ) \ regexp-pattern |
: // ( -- ) \ regexp-pattern |
| \G search for string |
\G search for string |
| \ idea: try to match one alternative and then the rest of regexp. |
\ idea: try to match one alternative and then the rest of regexp. |
| \ if that fails, jump back to second alternative |
\ if that fails, jump back to second alternative |
| |
|
| : JOINs ( sys -- ) BEGIN dup WHILE ]] JOIN [[ REPEAT drop ; |
: THENs ( sys -- ) BEGIN dup WHILE ]] THEN [[ REPEAT drop ; |
| |
|
| : {{ ( addr -- addr addr ) \ regexp-pattern |
: {{ ( addr -- addr addr ) \ regexp-pattern |
| \G Start of alternatives |
\G Start of alternatives |
| 0 ]] dup BEGIN [[ vars @ ; immediate |
0 ]] dup FORK IF drop true ;S BUT JOIN [[ vars @ ; immediate |
| : || ( addr addr -- addr addr ) \ regexp-pattern |
: || ( addr addr -- addr addr ) \ regexp-pattern |
| \G separator between alternatives |
\G separator between alternatives |
| vars @ varsmax @ max varsmax ! |
vars @ varsmax @ max varsmax ! vars ! |
| ]] dup FORK IF 2drop true ;S THEN [[ >r >r >r vars ! |
]] AHEAD BUT THEN [[ |
| ]] DONE drop dup [[ r> r> r> ]] BEGIN [[ vars @ ; immediate |
]] dup FORK IF drop true ;S BUT JOIN [[ vars @ ; immediate |
| : }} ( addr addr -- addr ) \ regexp-pattern |
: }} ( addr addr -- addr ) \ regexp-pattern |
| \G end of alternatives |
\G end of alternatives |
| vars @ varsmax @ max vars ! |
vars @ varsmax @ max vars ! drop |
| ]] dup FORK IF 2drop true ;S THEN drop dup [[ >r >r >r drop |
]] AHEAD BUT THEN drop false ;S [[ THENs ; immediate |
| ]] DONE drop LEAVE ;S [[ r> r> r> JOINs ; immediate |
|
| |
|
| \ match variables |
\ match variables |
| |
|