451 lines
17 KiB
C++
451 lines
17 KiB
C++
/* dfa.h - declarations for GNU deterministic regexp compiler
|
||
Copyright (C) 1988 Free Software Foundation, Inc.
|
||
|
||
This program is free software; you can redistribute it and/or modify
|
||
it under the terms of the GNU General Public License as published by
|
||
the Free Software Foundation; either version 2, or (at your option)
|
||
any later version.
|
||
|
||
This program is distributed in the hope that it will be useful,
|
||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
GNU General Public License for more details.
|
||
|
||
You should have received a copy of the GNU General Public License
|
||
along with this program; if not, write to the Free Software
|
||
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */
|
||
|
||
/* Written June, 1988 by Mike Haertel */
|
||
|
||
#ifdef STDC_HEADERS
|
||
|
||
#include <stddef.h>
|
||
#include <stdlib.h>
|
||
|
||
#else /* !STDC_HEADERS */
|
||
|
||
#define const
|
||
#include <sys/types.h> /* For size_t. */
|
||
extern char *calloc(), *malloc(), *realloc();
|
||
extern void free();
|
||
|
||
#ifndef NULL
|
||
#define NULL 0
|
||
#endif
|
||
|
||
#endif /* ! STDC_HEADERS */
|
||
|
||
#include <ctype.h>
|
||
#ifndef isascii
|
||
#define ISALNUM(c) isalnum(c)
|
||
#define ISALPHA(c) isalpha(c)
|
||
#define ISUPPER(c) isupper(c)
|
||
#define ISLOWER(c) islower(c)
|
||
#else
|
||
#define ISALNUM(c) (isascii(c) && isalnum(c))
|
||
#define ISALPHA(c) (isascii(c) && isalpha(c))
|
||
#define ISUPPER(c) (isascii(c) && isupper(c))
|
||
#define ISLOWER(c) (isascii(c) && islower(c))
|
||
#endif
|
||
|
||
/* 1 means plain parentheses serve as grouping, and backslash
|
||
parentheses are needed for literal searching.
|
||
0 means backslash-parentheses are grouping, and plain parentheses
|
||
are for literal searching. */
|
||
#define RE_NO_BK_PARENS 1
|
||
|
||
/* 1 means plain | serves as the "or"-operator, and \| is a literal.
|
||
0 means \| serves as the "or"-operator, and | is a literal. */
|
||
#define RE_NO_BK_VBAR 2
|
||
|
||
/* 0 means plain + or ? serves as an operator, and \+, \? are literals.
|
||
1 means \+, \? are operators and plain +, ? are literals. */
|
||
#define RE_BK_PLUS_QM 4
|
||
|
||
/* 1 means | binds tighter than ^ or $.
|
||
0 means the contrary. */
|
||
#define RE_TIGHT_VBAR 8
|
||
|
||
/* 1 means treat \n as an _OR operator
|
||
0 means treat it as a normal character */
|
||
#define RE_NEWLINE_OR 16
|
||
|
||
/* 0 means that a special characters (such as *, ^, and $) always have
|
||
their special meaning regardless of the surrounding context.
|
||
1 means that special characters may act as normal characters in some
|
||
contexts. Specifically, this applies to:
|
||
^ - only special at the beginning, or after ( or |
|
||
$ - only special at the end, or before ) or |
|
||
*, +, ? - only special when not after the beginning, (, or | */
|
||
#define RE_CONTEXT_INDEP_OPS 32
|
||
|
||
/* Now define combinations of bits for the standard possibilities. */
|
||
#define RE_SYNTAX_AWK (RE_NO_BK_PARENS | RE_NO_BK_VBAR | RE_CONTEXT_INDEP_OPS)
|
||
#define RE_SYNTAX_EGREP (RE_SYNTAX_AWK | RE_NEWLINE_OR)
|
||
#define RE_SYNTAX_GREP (RE_BK_PLUS_QM | RE_NEWLINE_OR)
|
||
#define RE_SYNTAX_EMACS 0
|
||
|
||
/* Number of bits in an unsigned char. */
|
||
#define CHARBITS 8
|
||
|
||
/* First integer value that is greater than any character code. */
|
||
#define _NOTCHAR (1 << CHARBITS)
|
||
|
||
/* INTBITS need not be exact, just a lower bound. */
|
||
#define INTBITS (CHARBITS * sizeof (int))
|
||
|
||
/* Number of ints required to hold a bit for every character. */
|
||
#define _CHARSET_INTS ((_NOTCHAR + INTBITS - 1) / INTBITS)
|
||
|
||
/* Sets of unsigned characters are stored as bit vectors in arrays of ints. */
|
||
typedef int _charset[_CHARSET_INTS];
|
||
|
||
/* The regexp is parsed into an array of tokens in postfix form. Some tokens
|
||
are operators and others are terminal symbols. Most (but not all) of these
|
||
codes are returned by the lexical analyzer. */
|
||
#if __STDC__
|
||
|
||
typedef enum
|
||
{
|
||
_END = -1, /* _END is a terminal symbol that matches the
|
||
end of input; any value of _END or less in
|
||
the parse tree is such a symbol. Accepting
|
||
states of the DFA are those that would have
|
||
a transition on _END. */
|
||
|
||
/* Ordinary character values are terminal symbols that match themselves. */
|
||
|
||
_EMPTY = _NOTCHAR, /* _EMPTY is a terminal symbol that matches
|
||
the empty string. */
|
||
|
||
_BACKREF, /* _BACKREF is generated by \<digit>; it
|
||
it not completely handled. If the scanner
|
||
detects a transition on backref, it returns
|
||
a kind of "semi-success" indicating that
|
||
the match will have to be verified with
|
||
a backtracking matcher. */
|
||
|
||
_BEGLINE, /* _BEGLINE is a terminal symbol that matches
|
||
the empty string if it is at the beginning
|
||
of a line. */
|
||
|
||
_ALLBEGLINE, /* _ALLBEGLINE is a terminal symbol that
|
||
matches the empty string if it is at the
|
||
beginning of a line; _ALLBEGLINE applies
|
||
to the entire regexp and can only occur
|
||
as the first token thereof. _ALLBEGLINE
|
||
never appears in the parse tree; a _BEGLINE
|
||
is prepended with _CAT to the entire
|
||
regexp instead. */
|
||
|
||
_ENDLINE, /* _ENDLINE is a terminal symbol that matches
|
||
the empty string if it is at the end of
|
||
a line. */
|
||
|
||
_ALLENDLINE, /* _ALLENDLINE is to _ENDLINE as _ALLBEGLINE
|
||
is to _BEGLINE. */
|
||
|
||
_BEGWORD, /* _BEGWORD is a terminal symbol that matches
|
||
the empty string if it is at the beginning
|
||
of a word. */
|
||
|
||
_ENDWORD, /* _ENDWORD is a terminal symbol that matches
|
||
the empty string if it is at the end of
|
||
a word. */
|
||
|
||
_LIMWORD, /* _LIMWORD is a terminal symbol that matches
|
||
the empty string if it is at the beginning
|
||
or the end of a word. */
|
||
|
||
_NOTLIMWORD, /* _NOTLIMWORD is a terminal symbol that
|
||
matches the empty string if it is not at
|
||
the beginning or end of a word. */
|
||
|
||
_QMARK, /* _QMARK is an operator of one argument that
|
||
matches zero or one occurences of its
|
||
argument. */
|
||
|
||
_STAR, /* _STAR is an operator of one argument that
|
||
matches the Kleene closure (zero or more
|
||
occurrences) of its argument. */
|
||
|
||
_PLUS, /* _PLUS is an operator of one argument that
|
||
matches the positive closure (one or more
|
||
occurrences) of its argument. */
|
||
|
||
_CAT, /* _CAT is an operator of two arguments that
|
||
matches the concatenation of its
|
||
arguments. _CAT is never returned by the
|
||
lexical analyzer. */
|
||
|
||
_OR, /* _OR is an operator of two arguments that
|
||
matches either of its arguments. */
|
||
|
||
_LPAREN, /* _LPAREN never appears in the parse tree,
|
||
it is only a lexeme. */
|
||
|
||
_RPAREN, /* _RPAREN never appears in the parse tree. */
|
||
|
||
_SET /* _SET and (and any value greater) is a
|
||
terminal symbol that matches any of a
|
||
class of characters. */
|
||
} _token;
|
||
|
||
#else /* ! __STDC__ */
|
||
|
||
typedef short _token;
|
||
|
||
#define _END -1
|
||
#define _EMPTY _NOTCHAR
|
||
#define _BACKREF (_EMPTY + 1)
|
||
#define _BEGLINE (_EMPTY + 2)
|
||
#define _ALLBEGLINE (_EMPTY + 3)
|
||
#define _ENDLINE (_EMPTY + 4)
|
||
#define _ALLENDLINE (_EMPTY + 5)
|
||
#define _BEGWORD (_EMPTY + 6)
|
||
#define _ENDWORD (_EMPTY + 7)
|
||
#define _LIMWORD (_EMPTY + 8)
|
||
#define _NOTLIMWORD (_EMPTY + 9)
|
||
#define _QMARK (_EMPTY + 10)
|
||
#define _STAR (_EMPTY + 11)
|
||
#define _PLUS (_EMPTY + 12)
|
||
#define _CAT (_EMPTY + 13)
|
||
#define _OR (_EMPTY + 14)
|
||
#define _LPAREN (_EMPTY + 15)
|
||
#define _RPAREN (_EMPTY + 16)
|
||
#define _SET (_EMPTY + 17)
|
||
|
||
#endif /* ! __STDC__ */
|
||
|
||
/* Sets are stored in an array in the compiled regexp; the index of the
|
||
array corresponding to a given set token is given by _SET_INDEX(t). */
|
||
#define _SET_INDEX(t) ((t) - _SET)
|
||
|
||
/* Sometimes characters can only be matched depending on the surrounding
|
||
context. Such context decisions depend on what the previous character
|
||
was, and the value of the current (lookahead) character. Context
|
||
dependent constraints are encoded as 8 bit integers. Each bit that
|
||
is set indicates that the constraint succeeds in the corresponding
|
||
context.
|
||
|
||
bit 7 - previous and current are newlines
|
||
bit 6 - previous was newline, current isn't
|
||
bit 5 - previous wasn't newline, current is
|
||
bit 4 - neither previous nor current is a newline
|
||
bit 3 - previous and current are word-constituents
|
||
bit 2 - previous was word-constituent, current isn't
|
||
bit 1 - previous wasn't word-constituent, current is
|
||
bit 0 - neither previous nor current is word-constituent
|
||
|
||
Word-constituent characters are those that satisfy isalnum().
|
||
|
||
The macro _SUCCEEDS_IN_CONTEXT determines whether a a given constraint
|
||
succeeds in a particular context. Prevn is true if the previous character
|
||
was a newline, currn is true if the lookahead character is a newline.
|
||
Prevl and currl similarly depend upon whether the previous and current
|
||
characters are word-constituent letters. */
|
||
#define _MATCHES_NEWLINE_CONTEXT(constraint, prevn, currn) \
|
||
((constraint) & 1 << ((prevn) ? 2 : 0) + ((currn) ? 1 : 0) + 4)
|
||
#define _MATCHES_LETTER_CONTEXT(constraint, prevl, currl) \
|
||
((constraint) & 1 << ((prevl) ? 2 : 0) + ((currl) ? 1 : 0))
|
||
#define _SUCCEEDS_IN_CONTEXT(constraint, prevn, currn, prevl, currl) \
|
||
(_MATCHES_NEWLINE_CONTEXT(constraint, prevn, currn) \
|
||
&& _MATCHES_LETTER_CONTEXT(constraint, prevl, currl))
|
||
|
||
/* The following macros give information about what a constraint depends on. */
|
||
#define _PREV_NEWLINE_DEPENDENT(constraint) \
|
||
(((constraint) & 0xc0) >> 2 != ((constraint) & 0x30))
|
||
#define _PREV_LETTER_DEPENDENT(constraint) \
|
||
(((constraint) & 0x0c) >> 2 != ((constraint) & 0x03))
|
||
|
||
/* Tokens that match the empty string subject to some constraint actually
|
||
work by applying that constraint to determine what may follow them,
|
||
taking into account what has gone before. The following values are
|
||
the constraints corresponding to the special tokens previously defined. */
|
||
#define _NO_CONSTRAINT 0xff
|
||
#define _BEGLINE_CONSTRAINT 0xcf
|
||
#define _ENDLINE_CONSTRAINT 0xaf
|
||
#define _BEGWORD_CONSTRAINT 0xf2
|
||
#define _ENDWORD_CONSTRAINT 0xf4
|
||
#define _LIMWORD_CONSTRAINT 0xf6
|
||
#define _NOTLIMWORD_CONSTRAINT 0xf9
|
||
|
||
/* States of the recognizer correspond to sets of positions in the parse
|
||
tree, together with the constraints under which they may be matched.
|
||
So a position is encoded as an index into the parse tree together with
|
||
a constraint. */
|
||
typedef struct
|
||
{
|
||
unsigned index; /* Index into the parse array. */
|
||
unsigned constraint; /* Constraint for matching this position. */
|
||
} _position;
|
||
|
||
/* Sets of positions are stored as arrays. */
|
||
typedef struct
|
||
{
|
||
_position *elems; /* Elements of this position set. */
|
||
int nelem; /* Number of elements in this set. */
|
||
} _position_set;
|
||
|
||
/* A state of the regexp consists of a set of positions, some flags,
|
||
and the token value of the lowest-numbered position of the state that
|
||
contains an _END token. */
|
||
typedef struct
|
||
{
|
||
int hash; /* Hash of the positions of this state. */
|
||
_position_set elems; /* Positions this state could match. */
|
||
char newline; /* True if previous state matched newline. */
|
||
char letter; /* True if previous state matched a letter. */
|
||
char backref; /* True if this state matches a \<digit>. */
|
||
unsigned char constraint; /* Constraint for this state to accept. */
|
||
int first_end; /* Token value of the first _END in elems. */
|
||
} _dfa_state;
|
||
|
||
/* If an r.e. is at most MUST_MAX characters long, we look for a string which
|
||
must appear in it; whatever's found is dropped into the struct reg. */
|
||
|
||
#define MUST_MAX 50
|
||
|
||
/* A compiled regular expression. */
|
||
struct regexp
|
||
{
|
||
/* Stuff built by the scanner. */
|
||
_charset *charsets; /* Array of character sets for _SET tokens. */
|
||
int cindex; /* Index for adding new charsets. */
|
||
int calloc; /* Number of charsets currently allocated. */
|
||
|
||
/* Stuff built by the parser. */
|
||
_token *tokens; /* Postfix parse array. */
|
||
int tindex; /* Index for adding new tokens. */
|
||
int talloc; /* Number of tokens currently allocated. */
|
||
int depth; /* Depth required of an evaluation stack
|
||
used for depth-first traversal of the
|
||
parse tree. */
|
||
int nleaves; /* Number of leaves on the parse tree. */
|
||
int nregexps; /* Count of parallel regexps being built
|
||
with regparse(). */
|
||
|
||
/* Stuff owned by the state builder. */
|
||
_dfa_state *states; /* States of the regexp. */
|
||
int sindex; /* Index for adding new states. */
|
||
int salloc; /* Number of states currently allocated. */
|
||
|
||
/* Stuff built by the structure analyzer. */
|
||
_position_set *follows; /* Array of follow sets, indexed by position
|
||
index. The follow of a position is the set
|
||
of positions containing characters that
|
||
could conceivably follow a character
|
||
matching the given position in a string
|
||
matching the regexp. Allocated to the
|
||
maximum possible position index. */
|
||
int searchflag; /* True if we are supposed to build a searching
|
||
as opposed to an exact matcher. A searching
|
||
matcher finds the first and shortest string
|
||
matching a regexp anywhere in the buffer,
|
||
whereas an exact matcher finds the longest
|
||
string matching, but anchored to the
|
||
beginning of the buffer. */
|
||
|
||
/* Stuff owned by the executor. */
|
||
int tralloc; /* Number of transition tables that have
|
||
slots so far. */
|
||
int trcount; /* Number of transition tables that have
|
||
actually been built. */
|
||
int **trans; /* Transition tables for states that can
|
||
never accept. If the transitions for a
|
||
state have not yet been computed, or the
|
||
state could possibly accept, its entry in
|
||
this table is NULL. */
|
||
int **realtrans; /* Trans always points to realtrans + 1; this
|
||
is so trans[-1] can contain NULL. */
|
||
int **fails; /* Transition tables after failing to accept
|
||
on a state that potentially could do so. */
|
||
int *success; /* Table of acceptance conditions used in
|
||
regexecute and computed in build_state. */
|
||
int *newlines; /* Transitions on newlines. The entry for a
|
||
newline in any transition table is always
|
||
-1 so we can count lines without wasting
|
||
too many cycles. The transition for a
|
||
newline is stored separately and handled
|
||
as a special case. Newline is also used
|
||
as a sentinel at the end of the buffer. */
|
||
char must[MUST_MAX];
|
||
int mustn;
|
||
};
|
||
|
||
/* Some macros for user access to regexp internals. */
|
||
|
||
/* ACCEPTING returns true if s could possibly be an accepting state of r. */
|
||
#define ACCEPTING(s, r) ((r).states[s].constraint)
|
||
|
||
/* ACCEPTS_IN_CONTEXT returns true if the given state accepts in the
|
||
specified context. */
|
||
#define ACCEPTS_IN_CONTEXT(prevn, currn, prevl, currl, state, reg) \
|
||
_SUCCEEDS_IN_CONTEXT((reg).states[state].constraint, \
|
||
prevn, currn, prevl, currl)
|
||
|
||
/* FIRST_MATCHING_REGEXP returns the index number of the first of parallel
|
||
regexps that a given state could accept. Parallel regexps are numbered
|
||
starting at 1. */
|
||
#define FIRST_MATCHING_REGEXP(state, reg) (-(reg).states[state].first_end)
|
||
|
||
/* Entry points. */
|
||
|
||
#if __STDC__
|
||
|
||
/* Regsyntax() takes two arguments; the first sets the syntax bits described
|
||
earlier in this file, and the second sets the case-folding flag. */
|
||
extern void regsyntax(int, int);
|
||
|
||
/* Compile the given string of the given length into the given struct regexp.
|
||
Final argument is a flag specifying whether to build a searching or an
|
||
exact matcher. */
|
||
extern void regcompile(const char *, size_t, struct regexp *, int);
|
||
|
||
/* Execute the given struct regexp on the buffer of characters. The
|
||
first char * points to the beginning, and the second points to the
|
||
first character after the end of the buffer, which must be a writable
|
||
place so a sentinel end-of-buffer marker can be stored there. The
|
||
second-to-last argument is a flag telling whether to allow newlines to
|
||
be part of a string matching the regexp. The next-to-last argument,
|
||
if non-NULL, points to a place to increment every time we see a
|
||
newline. The final argument, if non-NULL, points to a flag that will
|
||
be set if further examination by a backtracking matcher is needed in
|
||
order to verify backreferencing; otherwise the flag will be cleared.
|
||
Returns NULL if no match is found, or a pointer to the first
|
||
character after the first & shortest matching string in the buffer. */
|
||
extern char *regexecute(struct regexp *, char *, char *, int, int *, int *);
|
||
|
||
/* Free the storage held by the components of a struct regexp. */
|
||
extern void regfree(struct regexp *);
|
||
|
||
/* Entry points for people who know what they're doing. */
|
||
|
||
/* Initialize the components of a struct regexp. */
|
||
extern void reginit(struct regexp *);
|
||
|
||
/* Incrementally parse a string of given length into a struct regexp. */
|
||
extern void regparse(const char *, size_t, struct regexp *);
|
||
|
||
/* Analyze a parsed regexp; second argument tells whether to build a searching
|
||
or an exact matcher. */
|
||
extern void reganalyze(struct regexp *, int);
|
||
|
||
/* Compute, for each possible character, the transitions out of a given
|
||
state, storing them in an array of integers. */
|
||
extern void regstate(int, struct regexp *, int []);
|
||
|
||
/* Error handling. */
|
||
|
||
/* Regerror() is called by the regexp routines whenever an error occurs. It
|
||
takes a single argument, a NUL-terminated string describing the error.
|
||
The default regerror() prints the error message to stderr and exits.
|
||
The user can provide a different regfree() if so desired. */
|
||
extern void regerror(const char *);
|
||
|
||
#else /* ! __STDC__ */
|
||
extern void regsyntax(), regcompile(), regfree(), reginit(), regparse();
|
||
extern void reganalyze(), regstate(), regerror();
|
||
extern char *regexecute();
|
||
#endif /* ! __STDC__ */
|