]> granicus.if.org Git - re2c/log
re2c
7 years agoFixed history comparison in case both latest subhistories are bottoms.
Ulya Trofimovich [Tue, 16 May 2017 21:11:55 +0000 (22:11 +0100)]
Fixed history comparison in case both latest subhistories are bottoms.

In this case, instead of comparing histories by the leftmost criterion
(as it should be, if only one of them is bottom), we should assume that
from standpoint of this tag histories are equal and move on to other
tags.

7 years agoAvoid exponential blowup in tagged epsilon-closure construction.
Ulya Trofimovich [Tue, 16 May 2017 17:07:00 +0000 (18:07 +0100)]
Avoid exponential blowup in tagged epsilon-closure construction.

The previous algorithm waited until the full epsilon-path is built,
then compared it to already existing paths. The new algorithm compares
and merges partial epsilon-paths as soon as they arrive at the same
NFA state, so the overall number of paths is bounded by the number of
NFA states at all times.

7 years agoDon't split tag history into individual sub-histories for tags.
Ulya Trofimovich [Sat, 29 Apr 2017 20:13:50 +0000 (21:13 +0100)]
Don't split tag history into individual sub-histories for tags.

This is necassary for correct comparison of orbit tag histories:
if orbit tag is nested in an outer capture, this outer capture is
under repetition and there is an epsilon-path through it, then
this epsilon-path may contain pieces of orbit history that belong
to different iterations of outer capture; these pieces will be
glued together and the boundary between them will be lost.

Example: ((""){0,3}){0,2}.

However, in a common history we can always find boundaries
(they are marked by tags that correspond to outer captures).

7 years agoSkeleton: constified tag variables to get rid of compiler warning.
Ulya Trofimovich [Sun, 16 Apr 2017 11:36:44 +0000 (12:36 +0100)]
Skeleton: constified tag variables to get rid of compiler warning.

7 years agoCommand normalization: update pointer to last command, it might change.
Ulya Trofimovich [Sun, 16 Apr 2017 10:10:25 +0000 (11:10 +0100)]
Command normalization: update pointer to last command, it might change.

If normalization of 'save' commands removes the last 'save' command,
then pointer to 'next' will point to the 'next' field of the removed
command and the chain of commands will break.

7 years agoAdded comment.
Ulya Trofimovich [Sun, 16 Apr 2017 10:10:03 +0000 (11:10 +0100)]
Added comment.

7 years agorelease.sh: added cppcheck and headers check to wish list.
Ulya Trofimovich [Sun, 16 Apr 2017 10:09:15 +0000 (11:09 +0100)]
release.sh: added cppcheck and headers check to wish list.

7 years agoForbid dependency cycles in tag commands.
Ulya Trofimovich [Fri, 14 Apr 2017 17:47:11 +0000 (18:47 +0100)]
Forbid dependency cycles in tag commands.

The previous attempt was to forbid 2-cycles; clearly it's not enough
and cycles of length greater than 2 also pose a problem.

We could allow cycles and deal with them by introducind a temporary
variable. However, this would create variables local to basic blocks,
which would complicate liveness analysis and dead code elimination.

7 years agoTag interference analysis: compare actual values, not formal RHS.
Ulya Trofimovich [Fri, 14 Apr 2017 10:45:23 +0000 (11:45 +0100)]
Tag interference analysis: compare actual values, not formal RHS.

Commands in the same basic block may have equal right hand sides,
e.g. 'x = y; z = y;'. Clearly in this case 'x' and 'z' do not interfere:
both versions are equal to 'y'. This optimization allows to deduplicate
cases like '("a" @x @y @z "b")*', where multiple tags have the same
values.

However, it is insufficient to find all commands in the current block
with RHS equal to current RHS. An example when this algorithm fails:
'x = y; w = z; z = y;', here 'x' and 'z' are assigned to the same RHS
'y', but if we merge them into one variable, 'w' will get the wrong
value.

The fix is as follows: first, ignore subsequent commands and consider
only commands that precede current command: if subsequent command that
sets LHS to the same value precedes any use of it, liveness propagation
through basic block would mark this LHS as dead and not interfering
anyway; otherwise (if use precedes setting to the same value), then it
indeed interferes with current LHS. This fixes the above example
'x = y; w = z; z = y;': when analysing 'x = y;' command we find that
'z' is alive (as before), but ignore subsequent 'z = y;' command (due
to the fix).

Second, when analysing preceding commands, calculate actual value for
each LHS (based on pessimistic assumption that on entry of basic block
all used versions are different). Formal RHS of some preceding command
may coincide with current formal RHS, but their values might differ:
'x = y; y = w; z = y;', here 'x = y;' and 'z = y;' have identical formal
RHS, but the value may be different. On the other hand, formal RHS
might be different, while their actual values are equal, as in
'x = y; w = y; z = w;'.

7 years agoSkeleton: don't forget to return error in case of tag mismatch.
Ulya Trofimovich [Thu, 13 Apr 2017 10:02:35 +0000 (11:02 +0100)]
Skeleton: don't forget to return error in case of tag mismatch.

7 years agoFinal tag versions in unreachable rules should be marked as unused.
Ulya Trofimovich [Wed, 12 Apr 2017 12:26:15 +0000 (13:26 +0100)]
Final tag versions in unreachable rules should be marked as unused.

7 years ago'--dump-adfa': fixed pretty-printing of tag commands.
Ulya Trofimovich [Wed, 12 Apr 2017 11:06:17 +0000 (12:06 +0100)]
'--dump-adfa': fixed pretty-printing of tag commands.

7 years agoPreserve order of commands when adding fallback tags.
Ulya Trofimovich [Wed, 12 Apr 2017 11:05:13 +0000 (12:05 +0100)]
Preserve order of commands when adding fallback tags.

7 years agoAdded debug for tag optimizations (CFG and interference table).
Ulya Trofimovich [Wed, 12 Apr 2017 10:26:18 +0000 (11:26 +0100)]
Added debug for tag optimizations (CFG and interference table).

7 years agoRecord full history for history-based tags on epsilon-paths.
Ulya Trofimovich [Tue, 11 Apr 2017 16:25:13 +0000 (17:25 +0100)]
Record full history for history-based tags on epsilon-paths.

7 years agoKeep tag histories for the whole time of determinization.
Ulya Trofimovich [Mon, 10 Apr 2017 21:35:15 +0000 (22:35 +0100)]
Keep tag histories for the whole time of determinization.

7 years agoTopsort: always initialize in-degree with zeroes.
Ulya Trofimovich [Sat, 8 Apr 2017 22:04:48 +0000 (23:04 +0100)]
Topsort: always initialize in-degree with zeroes.

Don't rely on the fact that topsort leaves all-zero in-degree:
it doesn't if cycles are present. One could manually zero the remaining
entries after topsort, but zero initialization is more reliable.
And it's quite efficient since we only need to initialize the entries
we use.

7 years agoDon't forget final and fallback commands when looking for history-based tags.
Ulya Trofimovich [Sat, 8 Apr 2017 18:58:50 +0000 (19:58 +0100)]
Don't forget final and fallback commands when looking for history-based tags.

7 years agoMerged a couple of slightly different 'topsort' versions into one.
Ulya Trofimovich [Sat, 8 Apr 2017 18:37:32 +0000 (19:37 +0100)]
Merged a couple of slightly different 'topsort' versions into one.

7 years ago'save' commands also need topological sorting: they may depend on history.
Ulya Trofimovich [Sat, 8 Apr 2017 10:35:14 +0000 (11:35 +0100)]
'save' commands also need topological sorting: they may depend on history.

7 years agoWhen mapping states, avoid double substitution in 'save' commands.
Ulya Trofimovich [Sat, 8 Apr 2017 08:46:41 +0000 (09:46 +0100)]
When mapping states, avoid double substitution in 'save' commands.

When mapping states, we need to create a list of 'copy' commands.
However, if there is a 'save' command with LHS equal to RHS of the 'copy'
command, then instead of the 'copy' command we just fix LHS of the 'save'
command, as explained in this note:

    /* note [save(X), copy(Y,X) optimization]
     *
     * save(X) command followed by a copy(Y,X) command can be optimized to
     * save(Y). This helps reduce the number commands and versions (new version
     * X is gone), but what is more important, it allows to put copy commands
     * in front of save commands. This order is necessary when it comes to
     * fallback commands.
     *
     * Note that in case of injective mapping there may be more than one copy
     * command matching the same save command: save(X), copy(Y,X), copy(Z,X).
     * In this case save command must be replicated for each copy command:
     * save(Y), save(Z).
     *
     * For each save(X) command there must be at least one copy(Y,X) command
     * (exactly one case of bijective mapping). This is because X version in
     * save(X) command must be a new version which cannot occur in the older
     * DFA state. Thus all save commands are transformed (maybe replicated) by
     * copy commands, and some copy commands are erased by save commands.
     *
     * This optimization is applied after checking priority violation, so it
     * cannot affect the check.
    */

However, the previous implementation would sometimes erroneously re-fix
the already fixed 'save' commands. This patch rewrites it in a way that
avoids re-fixing.

7 years agoFixed checking duplicates when creating tag commands.
Ulya Trofimovich [Fri, 7 Apr 2017 21:36:20 +0000 (22:36 +0100)]
Fixed checking duplicates when creating tag commands.

7 years agoDon't forget histories when comparing right hand sides of tag commands.
Ulya Trofimovich [Fri, 7 Apr 2017 17:49:58 +0000 (18:49 +0100)]
Don't forget histories when comparing right hand sides of tag commands.

7 years agoSkeleton: don't emit declarations for tags that are optimized out.
Ulya Trofimovich [Fri, 7 Apr 2017 16:19:09 +0000 (17:19 +0100)]
Skeleton: don't emit declarations for tags that are optimized out.

7 years agoDon't forget initial tag command when looking for history-based tags.
Ulya Trofimovich [Fri, 7 Apr 2017 16:17:56 +0000 (17:17 +0100)]
Don't forget initial tag command when looking for history-based tags.

7 years agoTag histories in commands must be addressed by absolute value.
Ulya Trofimovich [Fri, 7 Apr 2017 16:00:14 +0000 (17:00 +0100)]
Tag histories in commands must be addressed by absolute value.

7 years agoDon't sort and deduplicate 'save' commands for tags with history.
Ulya Trofimovich [Fri, 7 Apr 2017 15:44:44 +0000 (16:44 +0100)]
Don't sort and deduplicate 'save' commands for tags with history.

Otherwise chains like '2=(1,x); 1=(1,y)' will be rewritten to
'1=(1,y); 2=(1,x)', which is a totally different thing.

7 years ago'save' commands for tags with history may need fallback copies.
Ulya Trofimovich [Fri, 7 Apr 2017 15:19:00 +0000 (16:19 +0100)]
'save' commands for tags with history may need fallback copies.

For simple tags without history 'save' command is self-sufficient:
it does not depend on anything. However, for tags with history 'save'
commands depend on history, which may be overwritten on the way from
accepting state to fallback transition. We must take care and backup
such overwritten histories when leaving the accepting state.

If history is not overwritten, we don't need a copy, but we still
have to propagate its liveness forward on all fallthrough paths
outgoing from accepting state.

7 years agoAdded skeleton support for tags with history.
Ulya Trofimovich [Fri, 7 Apr 2017 12:47:48 +0000 (13:47 +0100)]
Added skeleton support for tags with history.

7 years agoFixed order of tag commands.
Ulya Trofimovich [Thu, 6 Apr 2017 17:01:31 +0000 (18:01 +0100)]
Fixed order of tag commands.

First, fallback 'copy' commands should go before 'save' commands.
Second, codegen should not reorder 'copy' and 'save' commands.
These two bugs shadowed each other and everything seemed to work.

7 years agoChanged generic API for tags.
Ulya Trofimovich [Thu, 6 Apr 2017 16:45:42 +0000 (17:45 +0100)]
Changed generic API for tags.

    - removed 'YYCOPYTAG', use simple assignment insted
      (it is hard to think of any other definition)
    - renamed 'YYBACKUPTAG' to 'YYTAGP'
    - added 'YYTAGN', 'YYTAGLISTP', 'YYTAGLISTN'
    - removed 're2c:tags:default' configuration

7 years agoAdded codegen primitives for tags with history.
Ulya Trofimovich [Thu, 6 Apr 2017 14:18:12 +0000 (15:18 +0100)]
Added codegen primitives for tags with history.

7 years agoFixed tag optimizations to respect tags with history.
Ulya Trofimovich [Wed, 5 Apr 2017 20:56:44 +0000 (21:56 +0100)]
Fixed tag optimizations to respect tags with history.

7 years agoAdded syntax for tags with history.
Ulya Trofimovich [Wed, 5 Apr 2017 19:45:42 +0000 (20:45 +0100)]
Added syntax for tags with history.

7 years agoStarted to add optional history for each tag.
Ulya Trofimovich [Wed, 5 Apr 2017 16:01:54 +0000 (17:01 +0100)]
Started to add optional history for each tag.

This patch adds tracking and saving tag history during determinization;
however, optimizations and codegen totally neglect histories.

7 years agoSeparated lookahead tags from tag versions in closure items.
Ulya Trofimovich [Wed, 5 Apr 2017 11:42:05 +0000 (12:42 +0100)]
Separated lookahead tags from tag versions in closure items.

This commit reverts commit 0b2ca9bf141259baf6cb0b067621c06db5f06b46:
"Pack lookahead tags together with tag versions in TDFA state."

Packing was possible because we were not interested in tag version
if closure item had this tag in lookahead set. However, if we are
to store the full history of each tag, we'll need this intermediate
version in spite of the fact that it is overwritten on the next
transition.

7 years agoFixed liveness analysis inside of basic block.
Ulya Trofimovich [Tue, 4 Apr 2017 12:41:41 +0000 (13:41 +0100)]
Fixed liveness analysis inside of basic block.

Liveness flows backwards through basic block, so it should be
propagated backwards starting with live-out set and applying
commands one by one in reverse order.

7 years agoMerge 'save' and 'copy' command lists into one common list.
Ulya Trofimovich [Tue, 4 Apr 2017 07:25:34 +0000 (08:25 +0100)]
Merge 'save' and 'copy' command lists into one common list.

Previously we used separate lists for them; however the structure
of 'save' and 'copy' commands is almost identical, so its easy to
use one C struct for both types. Also, this simplifies analysis:
both types of commands are handled in a uniform way.

However, now we have to do list transformations like sorting and
deleleting duplicates in chunks.

This is a preliminary step to allow intermixing 'copy' and 'save'
commands.

7 years agoExplicitly track order of closure items for leftmost disambiguation.
Ulya Trofimovich [Thu, 16 Mar 2017 22:39:57 +0000 (22:39 +0000)]
Explicitly track order of closure items for leftmost disambiguation.

Before this commit leftmost order was maintained implicitly by construction
of closure: NFA is traversed in left-first order, so the earliest item added
to closure is also the leftmost. Special care was taken to preserve this
order and never rearrange closure elements.

This is convenient if leftmost disambiguation strategy is the only one.
However, other strategies require explicit bookkeeping of disambiguation
information (POSIX needs tag history for orbit tags, maximize/minimize
needs most recent tag versions). Implicit handling of leftmost strategy
does not fall in line with other strategies.

By tracking leftmost order explicitly we achieve several goals:
    - uniform handling of disambiguation strategies
    - DFA states can be treated as good old sets again, which is more
      conventional and means less work for minimization

POSIX tests changed (but the '/test/posix_captures/.run/__run.sh' script
still passes all tests) because there was an error in the order items were
added to closure, which probably prevented minimization (if the new item
dominated the existing one, the old one was substituted with the new one,
while it should have been removed and the new one appended at the end).

Skeleton tests changed because the order of NFA states in DFA state does
not matter, so more states can be merged by determinization.

7 years agoForbid 2-cycles in tag commands; fixed topsort to skip cycles.
Ulya Trofimovich [Wed, 8 Mar 2017 14:16:19 +0000 (14:16 +0000)]
Forbid 2-cycles in tag commands; fixed topsort to skip cycles.

7 years agoMoved a couple of files.
Ulya Trofimovich [Wed, 8 Mar 2017 08:48:30 +0000 (08:48 +0000)]
Moved a couple of files.

7 years agoMoved the main driver out of parser grammar file.
Ulya Trofimovich [Wed, 8 Mar 2017 08:32:16 +0000 (08:32 +0000)]
Moved the main driver out of parser grammar file.

7 years agoSplit options into constant and mutable; restricted access to mutable ones.
Ulya Trofimovich [Tue, 7 Mar 2017 21:29:17 +0000 (21:29 +0000)]
Split options into constant and mutable; restricted access to mutable ones.

Some options are immutable (target, output files; global switches like
conditions, reuse mode, storable states; support of flex syntax, etc.).
These options are passed as command-line arguments and never change.
It is safe to read them from any program point after parsing command-line
arguments.

Other options are configurable; they have block scope (may be specified
anywhere inside of the block and still affect the whole block).
Reading mutable options of yet unparsed block is not allowed because
they may affect the way RE2C parses current block (RE2C would be tempted
to base decisions on the latest option value, which may not be the final
one).

7 years agoForbid configuration of certain options.
Ulya Trofimovich [Tue, 7 Mar 2017 18:51:29 +0000 (18:51 +0000)]
Forbid configuration of certain options.

Configuration of these options either makes little sense (like changing
target on the fly) or is technically impossible to implement (the only
example is '-F, --flex-syntax' option which non-trivially affects the way
RE2C parses programs).

7 years ago'-Wcondition-order': check '-t, --type-header' option together with other conditions.
Ulya Trofimovich [Tue, 7 Mar 2017 18:43:05 +0000 (18:43 +0000)]
'-Wcondition-order': check '-t, --type-header' option together with other conditions.

7 years agoMoved enum out of struct and added prefix 'TARGET_' to all members.
Ulya Trofimovich [Tue, 7 Mar 2017 18:23:24 +0000 (18:23 +0000)]
Moved enum out of struct and added prefix 'TARGET_' to all members.

7 years agoDelay check if tags are allowed until the whole block has been parsed.
Ulya Trofimovich [Tue, 7 Mar 2017 18:16:56 +0000 (18:16 +0000)]
Delay check if tags are allowed until the whole block has been parsed.

7 years agoNow that we never re-parse anything we can always dicard old data.
Ulya Trofimovich [Tue, 7 Mar 2017 18:07:09 +0000 (18:07 +0000)]
Now that we never re-parse anything we can always dicard old data.

7 years agoTests for POSIX captures: avoid using 'flags:flex-syntax' configuration.
Ulya Trofimovich [Tue, 7 Mar 2017 17:57:48 +0000 (17:57 +0000)]
Tests for POSIX captures: avoid using 'flags:flex-syntax' configuration.

Use '--flex-syntax' command-line option instead.

The actual tests are not changed by this commit.

7 years agoAllow to set new encoding without explicitely resetting the current one.
Ulya Trofimovich [Tue, 7 Mar 2017 17:52:42 +0000 (17:52 +0000)]
Allow to set new encoding without explicitely resetting the current one.

7 years agoMore elegant handling of flex definitions.
Ulya Trofimovich [Mon, 6 Mar 2017 21:42:05 +0000 (21:42 +0000)]
More elegant handling of flex definitions.

7 years agoCheck precondition inside of the function rather than at every call site.
Ulya Trofimovich [Mon, 6 Mar 2017 17:31:22 +0000 (17:31 +0000)]
Check precondition inside of the function rather than at every call site.

Changes in tests are caused by the lack of newline after '/*!ignore:re2c*/'
directive. There's no particular reason why it should be emitted and RE2C
code is simpler without it.

7 years agoGathered all error checking related to '-r' option in one place.
Ulya Trofimovich [Mon, 6 Mar 2017 11:25:13 +0000 (11:25 +0000)]
Gathered all error checking related to '-r' option in one place.

7 years agoMoved remaining option-sensitive code away from AST.
Ulya Trofimovich [Mon, 6 Mar 2017 08:54:10 +0000 (08:54 +0000)]
Moved remaining option-sensitive code away from AST.

AST must be immutable and independent of options: it may be shared among
different blocks and conditions and reused with different options.

7 years agoMerged small header with part of AST into big header with the whole AST.
Ulya Trofimovich [Sun, 5 Mar 2017 11:51:30 +0000 (11:51 +0000)]
Merged small header with part of AST into big header with the whole AST.

7 years agoRenamed 'RegExp' to 'AST'.
Ulya Trofimovich [Sun, 5 Mar 2017 11:43:26 +0000 (11:43 +0000)]
Renamed 'RegExp' to 'AST'.

7 years agoDon't ever reparse '/*!rules:re2c ... */' block.
Ulya Trofimovich [Sun, 5 Mar 2017 10:43:54 +0000 (10:43 +0000)]
Don't ever reparse '/*!rules:re2c ... */' block.

AST is immutable and independent of options: encoding expansion and other
option-sensitive transformations are delayed until AST is converted to
intermediate representation. Thus AST can be shared among different blocks
and conditions and reused with any set of options without modifications.

Changes in tests are trivial: warnings from the rules block are no longer
emitted (we don't compile AST to DFA for rules block anymore, since it
will be discarded anyway); skeleton tests no longer emit '.input' and
'.keys' files for rules block.

7 years agoRestructured source files layout.
Ulya Trofimovich [Sat, 4 Mar 2017 23:27:52 +0000 (23:27 +0000)]
Restructured source files layout.

7 years agoDelay encoding expansion until AST is converted to intermediate representation.
Ulya Trofimovich [Sat, 4 Mar 2017 19:19:04 +0000 (19:19 +0000)]
Delay encoding expansion until AST is converted to intermediate representation.

7 years agoStore locations in AST so that error reporting does not depend on parser.
Ulya Trofimovich [Sat, 4 Mar 2017 17:00:59 +0000 (17:00 +0000)]
Store locations in AST so that error reporting does not depend on parser.

This allows us to delay things that should be delayed (like expanding
regexps with respect to current encoding), but which may throw an error
and need location info.

7 years agoAdded test for '--posix-captures': implicit grouping is forbidden.
Ulya Trofimovich [Sat, 4 Mar 2017 11:58:55 +0000 (11:58 +0000)]
Added test for '--posix-captures': implicit grouping is forbidden.

RE2C has named definitions, which allow to override operator precedence
without grouping parenthesis. However, in POSIX all parenthesis are
capturing: allowing some of them to be non-capturing would affect
disambiguation and might result in a different parse.

For now RE2C uses a crude criterion to determine if the given named
definition needs to be explicitely wrapped in parenthesis: if the topmost
constructor of the subexpression is either alternative, or catenation, or
another reference to another named definition, then wrapping is required.
This is an overapproximation, because it doesn't take the surrounding context
into account (e.g. catenation inside of an alternative doesn't need wrapping,
but catenation inside of an iteration does).

7 years agoAdded tests for POSIX captures.
Ulya Trofimovich [Fri, 3 Mar 2017 23:21:38 +0000 (23:21 +0000)]
Added tests for POSIX captures.

Tests are adopted from Glenn Fowler's testregex suite and its derivative
regex-posix-unittest by Christopher Kuklewicz.

Irrelevant tests have been dropped:

    - tests for specific POSIX features not supported by RE2C, like
      begin/end markers '$' and '^', backreferences, predefined character
      classes (most of them come from 'basic' subdirectory)

    - all 'leftassoc' tests: they are exact copies of 'rightassoc' tests,
      differ only in match results and are supposed to fail

    - all tests from 'categorize' that are supposed to fail

I also added a few tests of my own (in the 'other' subdirectory).

Tests are arranged as a bunch of '.dat' files (each file contains lines of
the form '<regexp> <string> <match>' or blank lines) and two scripts.

    - '__run.sh' runs all tests (generates temporary '.re' file with the
      given regexp, compiles it, runs against the given string and checks
      that the result coincides with the given result)

    - '__gen.sh' regenerates '.re' files for regular RE2C test suite (the
      generated files do not contain any code, just the bare regexp)

All regexps are pre-patched to exclude '\x00' (terminating NULL) character
from subregexps like '.' or '[^a]: otherwise RE2C-generated programs would
greedily consume all characters and fail. Note that if only '.' was present
in the original tests, we might have used '\n' as terminator, but '[^a]'
includes newline. Also '()' subregexp was substituted with '("")'.

7 years ago'--skeleton --posix-captures': use correct capture index when checking tags.
Ulya Trofimovich [Fri, 3 Mar 2017 23:09:19 +0000 (23:09 +0000)]
'--skeleton --posix-captures': use correct capture index when checking tags.

Instead of skipping fixed tags, skip orbit tags. For each orbit tag there is
a tag fixed on it, so if we check all fixed tags we'll surely check all orbit
tags. Anyway, with '--skeleton' all fixed tags correspond to orbit tags: other
fixed tags are forbidden with generic API (even zero-offset ones for now).

7 years agoRemove dead 'copy' commands.
Ulya Trofimovich [Fri, 3 Mar 2017 17:27:35 +0000 (17:27 +0000)]
Remove dead 'copy' commands.

For some reason dead code elimination pass simply ignored 'copy' commands
and removed only dead 'save' commands. Strange as it seems, this went
unnoticed until now (when one of the POSIX tests compiled into C++ code
that triggered GCC warning about use of uninitialized variable).

7 years agoFixed code generation for '--skeleton' with '--posix-captures'.
Ulya Trofimovich [Fri, 3 Mar 2017 17:06:31 +0000 (17:06 +0000)]
Fixed code generation for '--skeleton' with '--posix-captures'.

7 years agoAdded option '--posix-captures'.
Ulya Trofimovich [Fri, 3 Mar 2017 02:06:43 +0000 (02:06 +0000)]
Added option '--posix-captures'.

With this option re2c interprets all parenthetised subexpressions
as POSIX capturing groups and returns submatch according to POSIX
disambiguation rules (in variables 'yynmatch' and 'yypmatch', local
to each action).

The algorithm for disambiguation was invented by Christopher Kuklewicz.

7 years agoNow code generation can handle zero-offset fixed tags with generic API.
Ulya Trofimovich [Fri, 3 Mar 2017 01:47:03 +0000 (01:47 +0000)]
Now code generation can handle zero-offset fixed tags with generic API.

We'll need this for POSIX captures: in the prsence of orbit tag,
opening tag is needed only for disambiguation; after determinization
we can pretend that it is fixed on orbit tag with zero offset.
This will save some time and space on tag optimizations and maybe
reduce the number of tag variables.

7 years agoPrepare default tags insertion for non-sequential tag enumeration.
Ulya Trofimovich [Fri, 3 Mar 2017 01:21:53 +0000 (01:21 +0000)]
Prepare default tags insertion for non-sequential tag enumeration.

This will be needed when we add POSIX captures: opening and closing
capture tags must have sequential numbers, but the tags may be located
far from each other in the regexp tree (other tags are possible on
the way from the opening tag to the closing tag).

7 years agoDelay expansion of counted repetition until NFA construction time.
Ulya Trofimovich [Fri, 3 Mar 2017 01:15:05 +0000 (01:15 +0000)]
Delay expansion of counted repetition until NFA construction time.

This is necessary to avoid creating default tags for tags that
are duplicated by expansion of counted repetition. Default tags
should only be created if zero repetitions are acceptable; this case
is still expanded early.

7 years agoKeep the full history of tag occurences found by epsilon-closure.
Ulya Trofimovich [Fri, 3 Mar 2017 00:56:49 +0000 (00:56 +0000)]
Keep the full history of tag occurences found by epsilon-closure.

Currently this makes no difference because tag duplication in NFA
is forbidden. Anyway, we are only interested in the last occurence
of each tag on the epsilon-path to the given NFA state.

This will make difference for POSIX disambiguation strategy: we'll
have to allow tag duplication in NFA (because of counted repetition
and iteration expansion).

7 years agoMoved splitting charset and nullable rule analysis from AST to regexp.
Ulya Trofimovich [Sat, 25 Feb 2017 19:04:14 +0000 (19:04 +0000)]
Moved splitting charset and nullable rule analysis from AST to regexp.

7 years agoAdded yet another intermediate representation (after AST, before NFA).
Ulya Trofimovich [Sat, 25 Feb 2017 17:00:49 +0000 (17:00 +0000)]
Added yet another intermediate representation (after AST, before NFA).

Rationale: AST must be immutable, because it is shared among different
regexps, rules, conditions and blocks. NFA is mutable, but inconvenient
for some transformations (adding fixed and default tags, rewriting parts
of regexp, etc.). Before this commit all these transformations were done
in one pass (during convertion of AST to NFA) which made the code rather
complex. In order to add POSIX captures we'll need to add even more
complexity (tracking captures, creating tags for them, maintaining tag
order, etc.).

7 years ago'--dump-nfa': correctly print symbol ranges on NFA transitions.
Ulya Trofimovich [Fri, 24 Feb 2017 17:15:14 +0000 (17:15 +0000)]
'--dump-nfa': correctly print symbol ranges on NFA transitions.

7 years agoKeep fixed and variable tags together in one array.
Ulya Trofimovich [Fri, 24 Feb 2017 16:59:33 +0000 (16:59 +0000)]
Keep fixed and variable tags together in one array.

One must take fixed tags into account during determinization
if disambiguation policy is based on tag values.

Also, NFA construction is much easier when tags have common enumeration:
it allows to perdorm tag creation, tag fixing and adding default tags
in three distinct passes (for now they are intermixed and the code is
very complex).

7 years agoStop preserving order of tag versions when mapping TDFA states.
Ulya Trofimovich [Tue, 14 Feb 2017 14:51:47 +0000 (14:51 +0000)]
Stop preserving order of tag versions when mapping TDFA states.

We no longer need it; now that disambiguation policy is leftmost rather
than tag-centric, relative order of tag versions is irrelevant.
Tag versions have no impact on conflict resolution; any bijective
mapping is ok.

7 years agoMoved tests to subdirectory.
Ulya Trofimovich [Tue, 14 Feb 2017 09:53:24 +0000 (09:53 +0000)]
Moved tests to subdirectory.

7 years agoChanged disambiguation policy: leftmost rather than tag-centric.
Ulya Trofimovich [Mon, 13 Feb 2017 21:59:06 +0000 (21:59 +0000)]
Changed disambiguation policy: leftmost rather than tag-centric.

Instead of tracking priority for each tag version and resolving conflicting
configurations encountered by the epsilon-closure based on version priorities
(trying to minimize or maximize each tag), we now simply pick the leftmost
epsilon-path through the NFA.

NFA construction has to accommodate this policy: for example, greedy iteration
should be compiled so that always choosing the left alternative correpongs to
re-iteration, while choosing the right alternative corresponds to leaving the
iteration. The necessary adjustments have been done by earlier commit
03c65121da3f8bcb412576568cfff6cbca039470.

Now we can drop priority comparison when mapping TDFA states and stick to
simple bijection. This may add cycles in reordering commands. (Has not been
done yet.)

The rationale of switching to leftmost policy is this: with tag maximization/
minimization policy tags could influence the regexp: for example, regexp
    "a"* @p "a"*
has different meaning depending on whether tag 'p' is minimized (first iteration
is non-greedy) or maximized (first iteration is greedy). Moreover, adding
another tag with the opposite policy and higher priority would change the
meaning (suppose 'q' is minimized and earlier tags have higher priority):
    "a"* @q @p "a"*
This all makes regexps somewhat fragile, since adding just one tag may change
the whole regexp (and in the presence of other tags, it may change parse
results).

But even worse, with tag-centric disambiguation simple (untagged) subexpressions
are inferior to tagged ones. This contradicts intuition and even POSIX standard,
where  a*(a*)  is the same as  (a*)(a*) . We don't keep to POSIX (and we can't
because of last-longest instead of first-longest semantics of iteration as shown
by the  (<|<a|<ab|<aba|abab|baba|b>|>)*  example), but even this somewhat
tag-centric standard respects non-captured parts of the regexp. Note that
Laurikari in his TRE library has to do special preprocessing of the regexp in
order to assign correct minimization/maximisation direction to each tag
depending on the context of the tag. His original tag-centric policy (open tag
is minimized and has higher priority than closing tag which is maximized,
nested tags have lower priority) is insufficient to accommodate POSIX.

7 years agoDon't trace tags in epcilon-cycles when building tagged epsilon-closure.
Ulya Trofimovich [Mon, 13 Feb 2017 21:00:49 +0000 (21:00 +0000)]
Don't trace tags in epcilon-cycles when building tagged epsilon-closure.

7 years agoIn greedy regexps first alternative must correspond to consuming path.
Ulya Trofimovich [Mon, 13 Feb 2017 15:01:12 +0000 (15:01 +0000)]
In greedy regexps first alternative must correspond to consuming path.

By convention first alternative has higher priority.
So, for example, the following must be true:
    r*     = rr* | <empty>
    r{n,m} = r{n} (r{m - n} | r{m - n - 1} | ... | r{1} | <empty>)
    r{n,}  = r{n} (rr* | <empty>)

For now we don't care about priorities: this is a preparatory step
before transition to greedy leftmost semantics for tags.

7 years agoTags that occur earlier in regular expression must have greater priority.
Ulya Trofimovich [Mon, 13 Feb 2017 12:52:32 +0000 (12:52 +0000)]
Tags that occur earlier in regular expression must have greater priority.

7 years agoAdded debug option '--dump-nfa'.
Ulya Trofimovich [Mon, 13 Feb 2017 12:37:40 +0000 (12:37 +0000)]
Added debug option '--dump-nfa'.

7 years agore2c: cleanup a few clang ++ warnings
Sergei Trofimovich [Sun, 12 Feb 2017 11:16:30 +0000 (11:16 +0000)]
re2c: cleanup a few clang ++ warnings

Found those as:

    $ make CXXFLAGS="-Weverything -Werror -Wno-gnu-statement-expression -Wno-nested-anon-types"

- cleaned struct<->class declaration mismatch
- 'static' around local function
- 'RE2C_GXX_ATTRIBUTE ((noreturn))' around functions known to fail

Signed-off-by: Sergei Trofimovich <slyfox@gentoo.org>
7 years agoPack lookahead tags together with tag versions in TDFA state.
Ulya Trofimovich [Thu, 9 Feb 2017 16:01:51 +0000 (16:01 +0000)]
Pack lookahead tags together with tag versions in TDFA state.

This packing is possible because we are never interested in version
of the tag if the given item has this tag in lookahead set.

7 years agoUse 'normal form' of version matrix to test if states are mappable.
Ulya Trofimovich [Thu, 9 Feb 2017 13:10:31 +0000 (13:10 +0000)]
Use 'normal form' of version matrix to test if states are mappable.

Normal form is calculated for every new state before searching for
a mappable counterpart. All versions are renumbered to occupy consequent
numbers starting from one; version order is preserved.

7 years agoAdded tests that fail if mapping of TDFA states ignores version order.
Ulya Trofimovich [Thu, 9 Feb 2017 13:01:26 +0000 (13:01 +0000)]
Added tests that fail if mapping of TDFA states ignores version order.

These tests result in incorrect automaton unless mapping preserves
relative order of versions in TDFA states (respects version priority).

7 years agoStop supporting non-bijective mappings of TDFA states.
Ulya Trofimovich [Thu, 9 Feb 2017 12:58:43 +0000 (12:58 +0000)]
Stop supporting non-bijective mappings of TDFA states.

They are not very helpful, yet add much complexity to the mapping step.
This commit removes option '--non-bijective-mappings'.

7 years agoAssert that operands of 'copy' command are different.
Ulya Trofimovich [Wed, 8 Feb 2017 17:03:57 +0000 (17:03 +0000)]
Assert that operands of 'copy' command are different.

7 years ago'--dump-dfa-raw': better formatting of item's tags.
Ulya Trofimovich [Wed, 1 Feb 2017 13:18:13 +0000 (13:18 +0000)]
'--dump-dfa-raw': better formatting of item's tags.

7 years ago'--dump-dfa-raw': show proper shadowed items with updated tag versions.
Ulya Trofimovich [Wed, 1 Feb 2017 09:47:01 +0000 (09:47 +0000)]
'--dump-dfa-raw': show proper shadowed items with updated tag versions.

7 years agoAllocate final tag versions on the go instead of preallocating versions [1 .. N].
Ulya Trofimovich [Tue, 31 Jan 2017 18:53:57 +0000 (18:53 +0000)]
Allocate final tag versions on the go instead of preallocating versions [1 .. N].

One cannot rely the fact that final versions are [1 .. N] anyway (they are renamed
during tag optimization), so this is not convenient as it seems.

Now versions start from 1, which looks better (more logical) on the pictures.

7 years ago'--dump-dfa-raw': show shadowed NFA sub-states.
Ulya Trofimovich [Tue, 31 Jan 2017 09:50:58 +0000 (09:50 +0000)]
'--dump-dfa-raw': show shadowed NFA sub-states.

7 years agoFixed yet another case of disordered tags update and cursor increment.
Ulya Trofimovich [Wed, 25 Jan 2017 17:50:45 +0000 (17:50 +0000)]
Fixed yet another case of disordered tags update and cursor increment.

/* note [tag hoisting, skip hoisting and tunneling]
 *
 * Tag hoisting is simple: if all transitions have the same commands,
 * they can be hoisted out of conditional branches.
 *
 * Skip hoisting is only relevant with '--eager-skip' option.
 * Normally this option is off and skip is lazy: it happens after
 * transition to the next state, if this state is consuming.
 * However, with '--eager-skip' skip happens before transition to the next
 * state. Different transitions may disagree: some of them go to consuming
 * states, others don't. If they agree, skip can be hoisted (just like tags).
 *
 * '--eager-skip' makes tag hoisting more complicated, because now we have
 * to care about the type of automaton: lookahead TDFAs must skip after
 * writing tags, while non-lookahead TDFAs must skip before writing tags.
 * Therefore skip hoising cannot be done without tag hoisting in lookahead
 * TDFAs, and vice versa with non-lookahead TDFAs.
 * (Note that '--eager-skip' is implied by '--no-lookahead').
 *
 * Tunneling splits base states in two parts: head and body. Body has all
 * the conditional branches (transitions on symbols), while head has just
 * one unconditional jump to body.
 *
 * Normally tag hoisting should go before tunneling: hoisting may add new
 * candidates to be merged by tunneling. However, with '--eager-skip' tag
 * hoisting is interwined with skip hoisting, and the latter needs to know
 * which states are consuming. This is not possible if tunneling is still
 * to be done, because it may turn consuming states into non-consuming ones.
 * Another option is to disallow splitting states with non-hoisted skip
 * in the presence of '--eager-skip' (this way skip hoisting wouldn't need
 * to know tunneling results), but it's much worse for tunneling.
 */

Found by slyfox's fuzzer. :)

7 years agoDon't loose 'YYBACKUPTAG' command in case of equal mapped versions.
Ulya Trofimovich [Mon, 23 Jan 2017 22:28:16 +0000 (22:28 +0000)]
Don't loose 'YYBACKUPTAG' command in case of equal mapped versions.

Found by slyfox's fuzzer. :)

7 years agocorrect some minor manpage typos
jcfp [Mon, 23 Jan 2017 19:29:07 +0000 (20:29 +0100)]
correct some minor manpage typos

these showed up in lintian during a maintenance update of the debian re2c package

7 years agoUpdate tags after reading character in TDFAs without lookahead.
Ulya Trofimovich [Mon, 23 Jan 2017 21:02:24 +0000 (21:02 +0000)]
Update tags after reading character in TDFAs without lookahead.

Fixed forgotten special case when tags and cursor increment
are hoisted out of conditional branches.

Found by slyfox's fuzzer. :)

7 years agoFixed memory leak.
Ulya Trofimovich [Mon, 23 Jan 2017 18:09:05 +0000 (18:09 +0000)]
Fixed memory leak.

Found by valgrind:

 1,177 bytes in 1 blocks are definitely lost in loss record 1 of 1
    at 0x4C2AE40: operator new[](unsigned long) (in /usr/lib64/valgrind/vgpreload_memcheck-amd64-linux.so)
    by 0x42153F: re2c::cfg_t::compact(re2c::cfg_t const&, int*) (compact.cc:12)
    by 0x422088: re2c::optimize_tags(re2c::dfa_t&) (optimize.cc:18)
    by 0x42DE3A: re2c::compile(re2c::spec_t const&, re2c::Output&) (compile.cc:82)
    by 0x446742: re2c::parse(re2c::Scanner&, re2c::Output&) (parser.ypp:484)
    by 0x402DC0: main (main.cc:38)

7 years agoIterate CFG nodes in postorder to speed up liveness analysis.
Ulya Trofimovich [Mon, 23 Jan 2017 18:01:49 +0000 (18:01 +0000)]
Iterate CFG nodes in postorder to speed up liveness analysis.

Deep-first search postorder is the best possible order for solving
control-flow equations for liveness analysis.

7 years agoUse the same tag version for all transitions from the given state.
Ulya Trofimovich [Mon, 23 Jan 2017 17:42:37 +0000 (17:42 +0000)]
Use the same tag version for all transitions from the given state.

This reduces the overall number of versions greatly, which is crucial
for the space and time efficiency of tag liveness analysis.

Of course, bottom and normal transitions still have different versions.

Differents tags on the same transition also have different versions.
It is not incorrect to give them the same version: all tags updated by
one and the same transition are either set to bottom, or to current
input position (which is the same for all tags on the given transition).
But this would create many states with a very special "shape" that do
not map to other states. In fact, this would cause exponential blowup
in the number of states (even in simple cases).

7 years agoCompact tag versions before doing liveness analysis.
Ulya Trofimovich [Mon, 23 Jan 2017 16:38:04 +0000 (16:38 +0000)]
Compact tag versions before doing liveness analysis.

After TDFA construction tag versions occupy non-contiguous range
of natural numbers: there may be "holes" between versions. This wastes
space (and time) spent on liveness analysis and subsequent optimization
passes. This commit adds a simple pass that renumbers versions so that
they occupy contiguios range.

7 years agoUpdated parser for bison-3.0.4.
Ulya Trofimovich [Mon, 23 Jan 2017 16:19:24 +0000 (16:19 +0000)]
Updated parser for bison-3.0.4.