Mercurial > ~dholland > hg > ag > index.cgi
view doc/manual/sf.tex @ 10:5b21f127e957
silence random gcc warning
author | David A. Holland |
---|---|
date | Mon, 30 May 2022 23:58:14 -0400 |
parents | 13d2b8934445 |
children |
line wrap: on
line source
\chapter{Syntax Files} \index{Syntax file}\index{File} Input files to AnaGram are called \agterm{syntax files}. A syntax file comprises a grammar and associated C or C++ code. The grammar consists of a number of productions along with supportng information such as configuration sections and definitions of character sets. The associated code consists of reduction procedures (see \S 8.2.13) and embedded C or C++ code (\S 8.2.17). This chapter explains the rules for writing syntax files acceptable to AnaGram. The rules for interfacing your parser to the balance of your program are given in Chapter 9. \section{Lexical Conventions} \index{Lexical conventions} \subsection{Statements} \index{Statements} For purposes of this manual, AnaGram statements are considered to be productions, definition statements, configuration sections, and blocks of embedded C or C++ code, all discussed individually below. Each statement must begin on a new line. It is a good idea to separate statements visually in your file by using blank lines freely. There are generally no restrictions on the \index{Statements}\index{Order of statements}order of statements in a syntax file. Good programming practice, however, suggests that definitions and configuration sections should precede the grammar itself. \subsection{Spaces and Tabs} \index{Spaces}\index{Tabs} AnaGram allows spaces and tabs to be used freely to improve the readability of grammars. Spaces and tabs are ignored, except when embedded in a token name, in a character set definition, or in a keyword. Within a token name, any sequence of spaces and tabs counts as a single space. \subsection{Continuation Lines} \index{Continuation lines} AnaGram statements normally end with a newline character or the end of file. If AnaGram encounters the end of a line and the statement it is reading appears to be complete, it will not look for a continuation. To continue a statement to another line, just make sure that what you have on the first line is clearly incomplete. For example, \begin{indentingcode}{0.4in} prep phrase -> preposition, "the", noun \end{indentingcode} looks complete to AnaGram, whereas \begin{indentingcode}{0.4in} prep phrase -> preposition, "the", noun, \end{indentingcode} looks incomplete because of the dangling comma at the end. \subsection{Comments} \index{Comments} AnaGram accepts comments in accordance with the rules of C and C++, that is, normal C comments bracketed with \agcode{/*} and \agcode{*/}, as well as comments which begin with \agcode{//} and continue to the end of line. AnaGram also observes these conventions when skipping over embedded C code. Since the ANSI standard for C insists that normal C comments do not nest, AnaGram, by default, disallows nested comments. You may, however, set a configuration parameter, \index{Nest comments}\index{Configuration switches}\index{Comments} \agparam{nest comments}, to allow nested comments. See Appendix A. In any case, AnaGram will use the same convention for embedded C as it uses for AnaGram proper. You can change the convention in the middle of the file if necessary. AnaGram treats each comment delimited with \agcode{/*} and \agcode{*/} as though it were a single space. You can even put such comments in the middle of token names if you should want to. A comment that begins with \agcode{//} is treated as though the end of line occurred at the \agcode{//}. \subsection{Blank Lines and Form Feeds} \index{Blank lines} Because blank lines and form feeds are visual separators, AnaGram will not skip either looking for a continuation line. Therefore blank lines and form feeds can occur only between AnaGram statements, not in the middle of a statement. It is a good idea to separate groups of productions with a blank line or two, lest an accidental dangling comma make AnaGram think the beginning of the next production is a continuation of the present one. \section{Elements of Grammars} \subsection{Names} \index{Name}\index{Token} You may use names to represent tokens, character sets, keywords and \index{Virtual productions}\index{Production}virtual productions. Names follow the same general rules as for any programming language, with the notable exception that they may have embedded white space. Names are made up of letters, digits, or underscores. They may not begin with a digit. Any sequence of embedded spaces, tabs or comments counts as a single space. AnaGram distinguishes between upper and lower case\index{Case sensitivity}, so that \agcode{Word} and \agcode{word} are different names. There is no particular limit to the length of a name. There are no reserved words as such, although \agcode{grammar}, \agcode{eof}, and \agcode{error} will be treated as reserved words unless you take special action by setting appropriate configuration parameters. The names AnaGram uses for \index{Configuration parameters}configuration parameters follow the same rules as for other names, except that \index{Case sensitivity}case is ignored. \subsection{Reserved Words} \index{Reserved words}\index{Words} % XXX shouldn't that be \index{Grammar token}? AnaGram treats tokens with the names \index{Grammar}\agcode{grammar}, \index{Eof token}\index{Token}\agcode{eof}, and \index{Error token}\index{Token}\agcode{error} in a special manner unless certain measures are taken. Since you can override AnaGram's use of these names, they are not reserved words in the true sense. If your grammar has a token named \agcode{grammar}, AnaGram will take that token to be the grammar token for your grammar unless you set the \index{Token}\index{Grammar token}\index{Configuration parameters} \agparam{grammar token} configuration parameter or mark some other token as the grammar token using ``\index{ \_dol}\$''.% See below ???. If your grammar has a token named \agcode{error} and you take no further steps, AnaGram will assume you wish to use error token resynchronization in case of \index{Syntax error}\index{Errors}syntax error. See Chapter 9. If you wish to use some other token as an error token you may select it using the \index{Configuration parameters}\index{Token}\index{Error token} \agparam{error token} configuration parameter. If you wish to use \agcode{error} as a token name, but do not want error token resynchronization, you may set the \agparam{error token} configuration parameter to any name that is not used in your grammar. You may then use \agcode{error} as a token name without causing AnaGram to include error token resynchronization in your parser. \index{Resynchronization} If you select automatic resynchronization or error token resynchronization (see Chapter 9), AnaGram will look for a token called \agcode{eof} to use as an end of file indicator. You may either name your end of file token \agcode{eof} or you may set the \agparam{eof token} configuration parameter with the name of your end of file token. \subsection{Variable Names} \index{Name}\index{C variable names} With AnaGram you can associate C/C++ variable names with the \index{Semantic value}\index{Token}\index{Value}semantic values of tokens for use in your \index{Reduction procedure}reduction procedures. Each name follows the corresponding token in the grammar rule on the right of the production, separated from the token by a colon. AnaGram allows variable names made up of letters, digits, and underscores. They may not begin with a digit. Embedded spaces, tabs or comments, are not allowed, of course. AnaGram imposes no restriction on length, but uses your variable names just as you have written them in the code it generates to call reduction procedures. Remember that your compiler may have a limit on the length of variable names. Also, AnaGram itself uses C variable names beginning with \agcode{ag{\us}}. It is therefore wise to avoid using names of this form. \subsection{Terminal Tokens} \index{Terminal token}\index{Token} A \agterm{terminal token} is a token which does not appear on the left side of a production. It represents, therefore, a basic unit of input to your parser. You have several options with respect to terminal tokens. If the input to your parser consists of ASCII characters, you may define terminal tokens explicitly as ASCII characters or as sets of ASCII characters. If you have an input procedure which produces numeric codes, you may define the terminal tokens directly in terms of these numeric codes. On the other hand, you may leave the terminal tokens completely undefined. In this case, you must provide an input procedure which can determine the appropriate \index{Token}\index{Token number}\index{Number}token numbers. It is an all or none situation. If you provide any explicit definitions, you must provide them for all terminal tokens. Input procedures and token input are discussed in Chapter 9. Examples of non-character input may be found in the Macro Preprocessor example in the \agfile{examples/mpp} directory on your AnaGram distribution disk.% Further examples are given in Chapter ???. % XXX change ``on ...distribution disk'' to ``in ...distribution''. \subsection{Character Representations} \index{Character representations} In specifying admissible input characters you may use \index{Character constants}character constants following the normal C conventions. Remember that a character constant may specify only a single character. Although some C compilers will allow constructs such as \agcode{'mv'}, AnaGram doesn't allow this. AnaGram recognizes the same escape sequences as C, including octal and hex sequences, even though this is, strictly speaking, unnecessary. The escape sequences AnaGram recognizes are: % % It would be nice to be able to just write this and tell latex to set % it in three columns. but no... that would be too easy. % % %\begin{tabular}{ll} %\agcode{{\bs}a}&alert (bell) character\\ %\agcode{{\bs}b}&backspace\\ %\agcode{{\bs}f}&formfeed\\ %\agcode{{\bs}n}&newline\\ %\agcode{{\bs}r}&carriage return\\ %\agcode{{\bs}t}&horizontal tab\\ %\agcode{{\bs}v}&vertical tab\\ %\agcode{{\bs\bs}}&backslash\\ %\agcode{{\bs}?}&question mark\\ %\agcode{{\bs}'}&single quote\\ %\agcode{{\bs}"}&double quote\\ %\agcode{{\bs}ooo}&octal number\\ %\agcode{{\bs}xhh}&hexadecimal number\\ %\end{tabular} \begin{indenting}{0.4in} \begin{tabular}{llllll} \agcode{{\bs}a}&alert (bell) character& \agcode{{\bs}t}&horizontal tab& \agcode{{\bs}'}&single quote\\ \agcode{{\bs}b}&backspace& \agcode{{\bs}v}&vertical tab& \agcode{{\bs}"}&double quote\\ \agcode{{\bs}f}&formfeed& \agcode{{\bs\bs}}&backslash& \agcode{{\bs}\textit{ooo}}&octal number\\ \agcode{{\bs}n}&newline& \agcode{{\bs}?}&question mark& \agcode{{\bs}x\textit{hh}}&hexadecimal number\\ \agcode{{\bs}r}&carriage return\\ \end{tabular} \end{indenting} \bigskip The octal escape sequence allows up to three octal digits, in accordance with ANSI specifications for C. The hexadecimal numbers may contain an arbitrary number of digits; however AnaGram will truncate the result to sixteen bits. A backslash followed by any character other than those listed above will cause a syntax error. You may also represent characters by writing the numeric code explicitly, in decimal, octal, or hexadecimal representations. AnaGram follows the C conventions for integer constants: a leading \agcode{0} means the number is octal, a leading \agcode{0x} or \agcode{0X} means it is hexadecimal. The hex digits \agcode{a-f} may be either upper or lower case\index{Case sensitivity}. Numbers may be preceded by an optional minus sign. If your parser uses a pre-existing \index{Lexical scanner}lexical scanner and you wish to use the code numbers it generates to identify tokens, you may simply treat those code numbers as character numbers. You may use the numbers directly in your productions, or you may use definition statements to name them. You may also use an \agparam{enum} statement within a configuration section to attach names to the code numbers. % XXX shouldn't this use of enum be indexed? AnaGram also allows a special notation for control characters. You may represent a control character by using the ``\^{}'' character preceding any printing ascii character. Thus you can write \agcode{\^{}z} or \agcode{\^{}Z} to represent the DOS end-of-file character. Notice that quotation marks are not necessary. Examples of character representations: \begin{indenting}{0.4in} \begin{tabular}{cccc} \agcode{'K'}&\agcode{-1}&\agcode{0}&\agcode{'{\bs}t'}\\ \agcode{\^{}J}&\agcode{'{\bs}xff'}&\agcode{077}&\agcode{0XF3}\\ \end{tabular} \end{indenting} \subsection{Character Ranges} \index{Character range}\index{Range} It is convenient to be able to specify ranges of characters when writing a grammar. AnaGram supports several ways of representing ranges of characters. The first is an extension of the notation for character constants: \agcode{'a-z'} is the set of lower case characters. You can even use escape sequences such as \agcode{'{\bs}n-{\bs}r'} if you like. The order of characters used to specify the range is immaterial: \agcode{'z-a'} is the same as \agcode{'a-z'}. AnaGram will, however, issue a warning just in case the unusual order results from a clerical error. The second way to specify a range is by using two arbitrary character representations, as described above, separated by two dots. For example, \agcode{\^{}C..\^{}Z}, \agcode{3..26}, \agcode{3..032}, \agcode{3..0x1a}, and \agcode{\^{}C..0x1a}, all represent the same range of characters. Similarly, \agcode{'A-F'}, \agcode{'A'..'F'}, \agcode{0101..0106}, \agcode{0x41..0x46}, \agcode{65..70}, and \agcode{65..'F'} all represent the same range of characters. \subsection{Character Sets} \index{Character sets} If you provide explicit definitions for terminal tokens, the basic input unit for your parser will be considered a character set, even if your input procedure provides numeric codes that are not actually characters. As a terminal token, a character set will be matched by any input character that is a member of the set. Character sets may be named in definition statements, but they may also appear on the right sides of productions without being named. A character set may consist of one or more characters. You can specify a character set that consists of a single character by using any of the character representation methods described above. You can specify a set consisting of a range of characters by using any of the representations of character ranges described above. \index{Character sets} To specify more complicated sets, you can write \index{Expressions}\index{Set expressions}expressions using conventional set theoretic operations. In AnaGram input, these operations are specified as follows: \index{Union}\index{Difference}\index{Intersection}\index{Complement} \begin{indenting}{0.4in} \begin{tabular}{cl} \agcode{A + B}&(union)\\ \agcode{A - B}&(difference)\\ \agcode{A \& B}&(intersection)\\ \agcode{\~{}A}&(complement)\\ \end{tabular} \end{indenting} where \agcode{A} and \agcode{B} are arbitrary sets. Union and difference have the same precedence. Intersection has higher precedence and complement has the highest precedence. Thus in the expression \begin{indentingcode}{0.4in} A + \~{}B\&C \end{indentingcode} the complement operation is performed first, then the intersection, and finally the union. Watch out! In an AnaGram syntax file \agcode{65 + 97} represents the character set which consists of lower case \agcode{a} and upper case \agcode{A}. It does not represent 162, the sum of 65 and 97. Parentheses may be used to force the order of evaluation: \begin{indentingcode}{0.4in} \~{}(A \& (B+C)) \end{indentingcode} In this example the union of \agcode{B} and \agcode{C} is calculated, then the intersection of this set with \agcode{A} is calculated, and finally the complement is evaluated. The computation of the \index{Complement}complement of a \index{Character sets}set requires a definition of the \index{Universe}universe of set elements. AnaGram will define the universe to be the set of unsigned 8-bit characters, unless one or more characters outside that range have been specified. In that case, the universe will consist of all characters on the interval defined by the lesser of zero and the lowest character code used and the greater of 255 and the highest character code used. The complement of a character set is everything in this universe except the characters in the set. Characters which make up part of the character universe, but are not legitimate input according to your grammar, are lumped together into a special token which will cause an error if it occurs in your input. When your parser reads an input character, it uses that character to index a conversion table in order to determine the appropriate \index{Token number}\index{Token}\index{Number}token number. If the \index{Range}\index{Test range}\index{Configuration switches} \agparam{test range} configuration switch is on, its default setting, your parser will include code to verify that the character is in bounds before it indexes the conversion table. If you are satisfied that checking bounds is unnecessary, you may turn the \agparam{test range} switch off and get a slightly higher level of performance from your parser. For efficient processing, it is well to keep the number of tokens to a minimum. Therefore if you have a choice between defining a construct as a token, with a production, or a set, with a definition, the set is to be preferred. Some useful character sets are: \begin{indenting}{0.4in} \begin{tabular}{ll} \agcode{'a-z' + 'A-Z'}&Alphabetic characters\\ \agcode{'a-f' + 'A-F'}&Hex digits\\ \agcode{'0-9'}&Decimal digits\\ \agcode{0..127}&ASCII character set\\ \agcode{32..126}&Printing ASCII characters\\ \agcode{\~{}'{\bs}n'}&Anything but newline\\ \agcode{\^{}Z}&Windows/DOS end of file indicator\\ \agcode{-1}&Stream I/O end of file indicator\\ \agcode{0}&String terminator\\ \agcode{32..126 - 'a-z' - 'A-Z' - '0-9'}&Punctuation\\ \end{tabular} \end{indenting} \bigskip % XXX ``punctuation'' is wrong; it should subtract off space too Note that \agcode{'a-z'} is a range of characters but \agcode{32..126 - 'a-z'} is a set difference. When AnaGram encounters a character set in a grammar rule, it assigns a token number to the character set. If it has previously seen the same character set it will assign the same token number; however, it assigns the same token number only if the set expressions are obviously the same. Thus, AnaGram will assign the same token number every time it sees \agcode{A + B}, but will assign a different token number if it sees \agcode{B + A}. Only when AnaGram has finished scanning the entire syntax file can it actually evaluate the character sets. If it finds that several different tokens all refer to the same character set, it will create a single token that represents the true character set and create \index{Shell productions}\index{Production}``shell productions'' for the others. \index{Character sets}If the character sets you use in your grammar overlap, they do not properly represent \index{Terminal token}\index{Token}terminal tokens. To deal with this situation, AnaGram identifies all overlaps among character sets and extends your grammar by adding a number of extra productions. For instance, suppose your grammar uses the following character sets as though they were terminal tokens: \begin{indentingcode}{0.4in} 'a-z' + 'A-Z' '0-9' '0-7' 'a-f' + 'A-F' \end{indentingcode} AnaGram will then modify your grammar by adding the following productions: \begin{indentingcode}{0.4in} 'a-z' + 'A-Z' -> 'a-f' + 'A-F' | 'g-z' + 'G-Z' '0-9' -> '0-7' + '8-9' \end{indentingcode} Although the tokens \agcode{'a-z' + 'A-Z'} and \agcode{'0-9'} are technically now \index{Nonterminal token}\index{Token}nonterminal tokens, for purposes of determining the \index{Token}\index{Data type}data type of their \index{Semantic value}\index{token}\index{Value}semantic values, AnaGram continues to regard them as terminal tokens. This \index{Partition}\index{Universe}\index{Character universe} ``partitioning'' of the character universe is described in Chapter 6. \subsection{Keyword Strings} \index{Keywords} In your grammar, AnaGram recognizes character strings within double quotes (e.g., \agcode{"IF"}) as keywords. The strings follow the same syntactic rules as strings in C. The same escape sequences are honored. AnaGram does not, however, allow for the concatenation of adjacent strings. Note that AnaGram strings are used only for the definition of keywords in your grammar, not for messages to be displayed or printed. Keyword strings may not include null characters and must be at least one character long. You may have any number of keywords. Each is treated as a single terminal token. A keyword may be given a name by means of a definition statement. Keywords may appear in virtual productions. AnaGram's keyword recognition works in the following way. First, for each state in your parser, AnaGram prepares a list of all the keywords that are admissible in that state. Your parser will recognize a keyword \emph{only} if it is in an appropriate state; otherwise it will appear to be an anonymous sequence of characters. Your parser, in any state, checks for keywords it expects before it checks for acceptable characters. That is, \emph{keywords take precedence} over simple characters. It does not look for keywords that would not be acceptable input. The parser will do whatever lookahead is necessary in order to pick up the entire keyword. Thus if the character \agcode{I} and the keyword \agcode{IF} are both legitimate input at some point, \agcode{IF} will be recognized, if present, in preference to \agcode{I}. If several admissible keywords match the input, such as \agcode{IF} and \agcode{IFF}, the parser will select the longest match, \agcode{IFF} in this example. AnaGram does not incorporate keywords into its character sets. Keywords stand apart and should not appear in definitions of character sets. In particular, they are not considered as belonging to the complement of a character set. Thus for the production \begin{indentingcode}{0.4in} next char -> \~{}('{\bs}n' + \^{}Z) \end{indentingcode} a keyword would not be considered legitimate input. Note also that a keyword consisting of a single character does not belong to the character universe. Because of this fact, AnaGram's treatment of \agcode{'X'} and \agcode{"X"} is very different. If this seems confusing at first, try using only keywords which are at least two characters long until you have some experience with them. AnaGram's keyword recognition logic normally does not make any assumptions about what precedes or follows a keyword. Thus if \agcode{int} is a keyword, your parser will be capable of plucking it out of a string of characters such as \agcode{disintegrate} if, according to your grammar, it could follow \agcode{dis}. The \agparam{sticky} declaration and the \agparam{distinguish keywords} statement, described below, can prevent such unwanted recognition of keywords. A keyword following a \agparam{sticky} token will not be recognized if the first character of the keyword can be shifted in as part of the \agparam{sticky} token. The \agparam{distinguish keywords} statement prevents recognition of a keyword if it is followed immediately by a character of the sort that makes up the keyword. \subsection{Type Specifications For Tokens} \index{Token}\index{Token type}\index{Type declarations} When you write productions or token declarations (see below), AnaGram allows you to specify the data type\index{Token}\index{Data type} of the \index{Semantic value}\index{Token}\index{Value}semantic value of a token by using a C or C++ data type specification. The restrictions are that AnaGram does not allow specification of array or function types, nor explicit structure types. Types that are defined with typedef statements, structure definitions, or class definitions, including template classes, in your embedded C or C++ are acceptable. Thus the following specifications, for example, are acceptable: \begin{indentingcode}{0.4in} void int char * unsigned long *near static float *far my{\us}type double * struct descriptor struct widget * vector <double> * \end{indentingcode} On the other hand, the following specifications are \emph{not} valid: \begin{indentingcode}{0.4in} int[20] int *(int, unsigned char) \bra int x,y; float z; \ket struct \bra int k; float z; \ket \end{indentingcode} Note that AnaGram itself does nothing with the type specifications. It simply passes them on to your compiler as appropriate. \subsection{Productions} \index{Production} Productions are the basic units of a grammar. A production consists of a left side and a right side. \index{Left side}The left side of a production consists of one or more token names, joined by commas, optionally preceded by a type specification enclosed in parentheses. \index{Right side}The right side begins with an arrow and may either begin on the same line as the left side or on a new line. For example: \begin{indentingcode}{0.4in} program -> statement list, eof expression -> expression, plus, term (int) variable name, function name -> name:n = look{\us}up(n); \end{indentingcode} The part of the right side of a production following the arrow is called a \index{Grammar rule}\index{Rule}\agterm{grammar rule}, discussed below. A production need not have a right side at all. In this case, it is simply called a \index{Declaration}\index{Token}\agterm{token declaration}. AnaGram assigns \index{Token number}\index{Token}\index{Number}token numbers to the token names on the left side, and, if there is a type specification, records the data type for each of the tokens declared. Declarations of this sort are most useful when using input from a \index{Lexical scanner}lexical scanner. See Chapter 9 for a discussion of techniques for interfacing a lexical scanner to your parser. If you do not intend to use a lexical scanner you will have no need for token declarations. If you do not explicitly specify the type for the \index{Semantic value}\index{Token}\index{Value}semantic value of a token, it will be determined by the configuration parameter \index{Default token type}\index{Configuration parameters}\index{Token} \agparam{default token type} if it is a \index{Nonterminal token}\index{Token}nonterminal token or by the \index{Configuration parameters}configuration parameter \index{Input token type}\index{Default input type}\agparam{default input type} if it is a \index{Token}terminal token. \agparam{Default token type} defaults to \agcode{void}. \agparam{Default input type} defaults to \agcode{int}. If a production has more than one token on the left side, as in the third example above, it is called a \index{Semantically determined production}\index{Production} \agterm{semantically determined production}. Semantically determined productions are a useful tool for exerting semantic control over syntactic analysis. A semantically determined production should have a reduction procedure which determines on a case by case basis which of the tokens on the left side should be taken as the reduction token. If there is no reduction procedure, or if the reduction procedure does not make a choice, the reduction token will be the first syntactically correct token on the left side of the production. In the example above, \agcode{variable name} will be the reduction token unless \agcode{look{\us}up} changes it to \agcode{function name}. Semantically determined productions are discussed more fully in Chapter 9. If several productions have the same left side, it does not need to be repeated. Subsequent right hand sides must each start on a new line. For example: \begin{indentingcode}{0.4in} integer -> digit -> integer, digit name -> letter -> name, letter -> name, digit \end{indentingcode} On the other hand, you do not have to group productions with the same left side. You could write the above productions as follows, although it would certainly not be good programming practice: \begin{indentingcode}{0.4in} name -> name, digit integer -> integer, digit name -> name, letter integer -> digit name -> letter \end{indentingcode} Nevertheless, there are a few occasions involving complex cross recursions and semantically determined productions where it is not possible to group productions neatly. The right side of a production can be empty. Such a production is called a \index{Null productions}\index{Production}\agterm{null production}. Null productions are useful to denote an optional element in a grammar, or a list that may be empty. For example: \begin{indentingcode}{0.4in} optional widget -> -> widget optional qualifiers -> -> optional qualifiers, qualifier \end{indentingcode} A second way to write multiple productions with the same left side uses the \index{Vertical bar}\index{|}vertical bar character, ``$|$'', to separate the grammar rules. The productions given above for \agcode{name}, \agcode{optional widget}, and \agcode{optional qualifiers} can also be written: \begin{indentingcode}{0.4in} name -> letter | name, letter | name, digit optional widget -> | widget optional qualifiers -> | optional qualifiers, qualifier \end{indentingcode} Note that a null production cannot \emph{follow} a vertical bar. A token that has a null production is called a \index{Zero length token}\index{Token}\agterm{zero length token}, since it can be represented by an empty sequence of input characters, that is to say, by nothing at all. Furthermore, even if a token doesn't have any null productions, if it has at least one rule consisting entirely of zero length tokens it is also a zero length token. In the Token Table window, AnaGram notes which tokens are zero length, because they can be a source of conflicts. \subsection{Grammar Token} Every grammar must have a single token which produces the entire grammar. This token is variously called the \index{Token}\index{Grammar token}\agterm{grammar token}, the \index{Goal token}\agterm{goal token} or the \index{Start token}\agterm{start token}. AnaGram provides several methods you may use to specify which token in your grammar is the grammar token. You may simply use the name \agcode{grammar} for the grammar token. If you wish to use some other more descriptive name for your grammar token, you may mark it with a following dollar sign when it appears on the left side of a production. Alternatively, you may set the \index{Grammar token}\index{Configuration parameters}\agparam{grammar token} configuration parameter to specify the grammar token. Here are examples of the methods: \begin{indentingcode}{0.4in} grammar -> [statement | newline]/... program \$ -> [statement | newline]/... {}[ grammar token = program ] program -> [statement | newline]/... \end{indentingcode} If you should use more than one of these techniques, AnaGram resolves the issue in the following manner: A marked token or a configuration parameter setting always takes precedence over simply naming a token \agcode{grammar}. If you mark more than one token or set the configuration parameter more than once, the last setting or mark wins. \subsection{Grammar Rules} \index{Rule}\index{Grammar rule} The part of a production to the right of the arrow is more often called a \agterm{grammar rule}, or simply \agterm{rule}. A grammar rule is a sequence of \index{Rule elements}\agterm{rule elements}, joined by commas, as in the examples of productions given above. Rule elements are token names, character set expressions, virtual productions, or immediate actions (see below). Each rule element may be optionally followed by a parameter assignment. The entire rule may be followed by an optional reduction procedure. A \index{Parameter assignment}parameter assignment is a colon followed by a C variable name. Here are some examples of rule elements with parameter assignments: \begin{indentingcode}{0.4in} '0-9':d integer:n expression:x declaration:declaration{\us}descriptor \end{indentingcode} The parameters you assign to tokens in your grammar rule become the formal parameters for your \index{Reduction procedure}reduction procedure. The data type\index{Data type}\index{Reduction procedure arguments} of the parameter is determined by the data type for the semantic value of the token to which it is assigned. If your grammar rule has parameter assignments, but does not have a reduction procedure, AnaGram will give you a warning in case the lack of a reduction procedure is an oversight. If you don't need a reduction procedure you may safely ignore the warning. On the other hand, AnaGram has no way to determine whether you have failed to make necessary parameter assignments. You won't find out until you compile your parser, when your compiler will give you error messages for undefined symbols. AnaGram assigns a unique rule number to each rule in your grammar. Rules are numbered sequentially as they are encountered in the syntax file. AnaGram constructs rule zero itself. Rule zero normally has a single element, the grammar token, unless you have a \agparam{disregard} statement in your grammar. In this case there will be two elements. \subsection{Reduction Procedures} \index{Reduction procedure} % XXX somewhere in here it ought to say something like % ``in the parsing literature reduction procedures are often known as % \agterm{semantic actions}.'' % Note that R. says there's some subtle difference between the usual % concept of semantic action and AG's concept of reduction procedure. % I don't know what this difference is and I hope she can recall it. % % D. thinks this note ought to be at the end; R. wants it at the top. A \agterm{reduction procedure} is a piece of C code which optionally follows a production. The code is executed when your parser identifies the production in its input. There are two forms for reduction procedures, a short form and a long form. The short form consists of a single C expression. The long form consists of an arbitrary block of C code. When AnaGram builds a parser, it inspects the grammar rule to which the procedure is attached and identifies the parameters for the procedure. It uses these parameters as the formal parameters for the procedure. If the \index{Macros}\index{Allow macros}\index{Configuration switches} \agparam{allow macros} configuration switch has not been turned off, AnaGram codes the reduction procedure as a macro definition whenever possible. Otherwise AnaGram codes it as a function definition. AnaGram builds the name for a reduction procedure by appending its internal procedure number to the string \agcode{ag{\us}rp{\us}}. Thus reduction procedures are numbered in the order in which they are encountered in the syntax file. Both long and short form reduction procedures are preceded by an equal sign which follows the production. The short form consists of a C or C++ expression terminated by a semicolon. When the grammar rule is reduced, the expression will be evaluated and its value will become the value of the reduction token. The expression and the terminating semicolon must be entirely on a single line. Note that, if you really need to make the expression longer than will fit on one line, you can embed a newline in a comment. Some examples of short form reduction procedures are: % XXX is there anything we can do about the ugly underscores? \begin{indentingcode}{0.4in} =0; =1; =10*n + d-'0'; = special{\us}processor(first{\us}parameter, second{\us}parameter); =word{\us}count++; =widget(constant{\us}1*parameter{\us}1 + constant{\us}2*parameter{\us}2 /* {} */ + constant{\us}3*parameter{\us}3); \end{indentingcode} A long form reduction procedure consists of an arbitrary block of C or C++ code, enclosed in braces (\bra \ket). AnaGram will code the reduction procedure as a function. To return a value for the reduction token, simply use the \agcode{return} statement. There are effectively no restrictions on the content or length of a reduction procedure. Of course, if there are unbalanced braces, unterminated comments or unterminated string literals, AnaGram will not be able to determine where the reduction procedure ends. AnaGram treats \index{Comments}nested comments within a reduction procedure according to the value of the \index{Nest comments}\index{Configuration switches}\agparam{nest comments} configuration switch at the point where it encounters the reduction procedure. From a practical point of view it is not usually good practice to have a reduction procedure that is more than a few lines long since a long procedure will hamper your overall view of your grammar. Long reduction procedures should be written as separate named functions, and should either be included in the embedded C portion of your syntax file or should be included in a wholly separate module. Here is an example of a long form reduction procedure: \begin{indentingcode}{0.4in} =\bra if (flag) \bra total += x; return identify(x); \ket else \bra total = 0; flag = 1; return init{\us}table(x); \ket \ket \end{indentingcode} If a rule does not have a reduction procedure, the semantic value of the reduction token will be set to the \index{Semantic value}\index{Token}\index{Value}semantic value of the first token in the rule, unless the rule is a \index{Null productions}null production. In the latter case, the value of the reduction token will be set to zero. % XXX and what if zero isn't a valid value for the type? a compiler % error will occur. % XXX add something like % % Variables appearing in reduction procedures which do not have a % parameter assignment in the corresponding grammar rule can be % declared globally or (file)-statically in your embedded C, or % alternatively could be added to the parser control block using % the \agparam{extend pcb} statement (q.v. | See Section ....). % (Reword this.) % % Should also discuss the sequencing of reduction procedure calls % so that people understand what happens if you use such variables. % % also ``A reduction procedure can be used to terminate parsing for % semantic reasons''. % \subsection{Immediate Actions} \index{Immediate action}\index{Action} An immediate action is a rule element that consists of executable C or C++ code embedded within a grammar rule to be executed when it is encountered. An immediate action is denoted by the use of an exclamation point, \index{!}``!''. The content of an immediate action may be written following the rules for either long form or short form reduction procedures. As with any other rule element, it must be separated from preceding and following rule elements by commas. In the grammar for a simple desk calculator, one might write \begin{indentingcode}{0.4in} transaction -> !printf('\#');, expression:x = printf("\%d{\bs}n", x); \end{indentingcode} % XXX s/apparent/visible/ Notice that the only apparent difference between an immediate action and a reduction procedure is that the immediate action is preceded by ``!'' instead of ``=''. The immediate action must be followed by a comma to separate it from the following rule element. Immediate actions may also be used in definitions: \begin{indentingcode}{0.4in} prompt = !printf('\#'); \end{indentingcode} AnaGram implements an immediate action by creating a special token for it. AnaGram then creates a single null production for the token. Finally, the immediate action is implemented as the reduction procedure for the null production. For example, you could implement \agcode{prompt} by writing a null production with a reduction procedure: \begin{indentingcode}{0.4in} prompt -> = printf('\#'); \end{indentingcode} This production would be equivalent to the definition above. There are two ways, however, in which immediate actions differ from the equivalent null production. Immediate actions may access any parameter assignments which precede them in the rule in which they occur. On the other hand, there is no way to assign a data type to the semantic value, if any, returned by the immediate action. Therefore, the type is determined by your setting of the \index{Default token type}\index{Configuration parameters} \agparam{default token type} configuration parameter. \subsection{Virtual Productions} \index{Virtual productions}\index{Production} Virtual productions are a convenient short form notation for common grammatical constructs involving choice and repetition. The notation represents an extension of notation commonly used in programming manuals. A virtual production may be written in a grammar rule at any place where you could write a token name, even within another virtual production. Note that use of virtual productions is never \emph{required}, since the equivalent productions can always be written out explicitly instead. When AnaGram encounters a virtual production, it replaces the virtual production with a new token and writes appropriate productions for the new token. When you look at your syntax tables using AnaGram windows, you will see the productions that AnaGram generates. AnaGram keeps a record of virtual productions, so that generally if you use the same virtual production a second time, you get the same set of tokens and productions that were generated the first time it was used. This is not the case if the virtual productions contain reduction procedures or immediate actions, since AnaGram is not equipped to determine whether two pieces of C code are equivalent. Thus, a virtual production that contains a reduction procedure will be unique and will not be reused. One disadvantage of virtual productions is that there is no way to specify the data type of the \index{Semantic value}\index{Virtual production}semantic value of a virtual production. Therefore, if you have a reduction procedure within a virtual production, its return value must be consistent with the type defined by the \index{Default token type}\index{Configuration parameters}\agparam{default token type} configuration parameter. The simplest virtual production is the \index{Token}\index{Optional token}\agterm{optional token}. If \agcode{x} is an arbitrary token name or set expression, you can indicate an optional \agcode{x} by writing \index{?}\agcode{x?}. You may also indicate a repetition of \agcode{x} by using the ellipsis with either \agcode{x} or \agcode{x?}. \index{...}\index{Ellipsis}Thus \agcode{x...} represents one or more instances of \agcode{x} and \index{?...}\agcode{x?...} represents zero or more instances of \agcode{x}. For example: \begin{indentingcode}{0.4in} '+'? \end{indentingcode} can be used to represent an optional plus sign, that is, a choice between a plus sign and nothing at all. Similarly, \begin{indentingcode}{0.4in} '{\bs}n'?... \end{indentingcode} represents an optional sequence of newline characters. \index{Brackets}\index{Braces}\index{\_opb\_clb}\index{[]} The next category of virtual productions uses brackets or braces to indicate a choice among a number of enclosed grammar rules separated by vertical bars. A single rule may also be enclosed. Note that \emph{rules}, with following reduction procedures, are allowed, not simply tokens. Braces are used to indicate that one option must be chosen. Brackets are used to indicate the choice is optional, i.e. may be omitted altogether. The ellipsis following a set of options within brackets or braces indicates the option may be repeated an indefinite number of times. You can use braces to indicate a simple choice among a number of options. A Cobol grammar offers the following choice of equivalent keywords: \begin{indentingcode}{0.4in} \bra "RECORD", "IS"? | "RECORDS", "ARE"? \ket \end{indentingcode} \index{\_opb\_clb...}\index{ []...} You may use the ellipsis with braces to indicate an arbitrary positive number of repetitions of the choice: \begin{indentingcode}{0.4in} {\bra}type specifier | storage class specifier{\ket}... \end{indentingcode} This expression requires at least one type specifier or storage class specifier, but will accept any number. \index{[]} To make a choice optional, use brackets instead of braces. An example, again drawn from a Cobol grammar, is: \begin{indentingcode}{0.4in} {}["LIMIT", "IS"? | "LIMITS", "ARE"?] \end{indentingcode} \index{[]...} Ellipses may be used with brackets to indicate an arbitrary number of choices that may be omitted altogether: \begin{indentingcode}{0.4in} {}[argument, [',', argument]...] \end{indentingcode} This expression describes an optional argument list with arguments separated by commas. If you use a null production within braces, it must be the first option: \begin{indentingcode}{0.4in} \bra | '+' | '-' \ket \end{indentingcode} Normally, you would do this only if you wanted to attach a reduction procedure to the null production. Note that if you include a null production within braces, and add an ellipsis after the closing brace for repetition, your grammar will be ambiguous. Just exactly how many times does the null production occur? Use brackets instead, and omit the null production. Null productions are not allowed with brackets, since they would be intrinsically ambiguous. The options within braces or brackets may be grammar rules of any length or complexity and may themselves contain virtual productions of arbitrary complexity. Nevertheless, in practice, clarity suffers as soon as the options get very complex. Virtual productions are most important and useful when used in simple situations. In those situations they will enhance the clarity of your grammar. Here is an example that is moderately complex, even though each rule consists of a single token: \begin{indentingcode}{0.4in} \bra{\bra}"on" | "true"\ket = 1; | {\bra}"off" | "false"\ket = 0; | integer\ket \end{indentingcode} This example can be used to allow as input either an integer or, for special cases, keywords. You could write this option out in the following way: \begin{indentingcode}{0.4in} p1 -> p2 = 1; -> p3 = 0; -> integer p2 -> "on" -> "true" p3 -> "off" -> "false" \end{indentingcode} The final category of virtual production provides a notation for \index{Alternating sequence}\agterm{alternating sequences}. An alternating sequence is a set of choices which may be repeated arbitrarily subject to the side condition that no choice may follow itself, in other words, that the choices must alternate. Alternating sequences are written with either brackets or braces depending on whether the sequence is optional or not, followed by \index{/...}``\agcode{/...}''. Note that the choices themselves may allow sequences. For example: \begin{indentingcode}{0.4in} program -> [statement | newline...]/..., eof \end{indentingcode} represents a sequence of statements separated by one or more newlines. Any two statements must be separated by one or more newline characters, and newlines may also appear at the beginning and the end of the program. Null productions are not allowed within alternating sequences, since they are intrinsically ambiguous in all cases. \subsection{Definition Statements} \index{Definitions}\index{Definition statement}\index{Statement} A definition statement is simply a shorthand way of naming a character set, a \index{Virtual productions}\index{Production}virtual production, a keyword string, or an immediate action. It can also be used for providing an alternate name for a token. Definitions have the form: \begin{indentingcode}{0.4in} name = \codemeta{character set} name = \codemeta{virtual production} name = \codemeta{keyword} name = \codemeta{immediate action} name = \codemeta{token name} \end{indentingcode} The name may be any name acceptable to AnaGram. The name can then be used anywhere you might have used the expression on the right side. \index{!}For example: \begin{indentingcode}{0.4in} upper case letter = 'A-Z' lower case letter = 'a-z' letter = upper case letter + lower case letter statement list = statement?... while keyword = "WHILE" prompt = !printf("Please enter name:"); \end{indentingcode} It is important to recognize that a definition statement that names a set does not define a token. A token is defined only when the set is used in a grammar rule, and then only if the set is used directly, not in combination with some other set. Furthermore, if you use a character set directly in a grammar rule, and in some other rule you use a name that refers to the same set of characters, you will get two different tokens. For example, if you have defined \agcode{upper case letter} as in the above example and use both \agcode{upper case letter} and \agcode{'A-Z'} in grammar rules, AnaGram will assign different \index{Token number}\index{Token}\index{Number}token numbers to accommodate any differences in attributes you may assign to the tokens. Renaming tokens is a convenient way to connect two independently written portions of a grammar. % See the C grammar in the EXAMPLES directory of your distribution % disk for an example. \subsection{Embedded C} \index{Embedded C} You may encapsulate C or C++ code in your syntax file by enclosing it in braces (\bra \ket). Such pieces of code are copied to the parser file untouched, in the order they are found in the syntax file. There may be any number of such pieces of embedded C. The only restriction is that they must not start on the same line as some other AnaGram statement, and following AnaGram statements must also start on fresh lines. Normally, the blocks of embedded C in your syntax file are copied to the parser file \emph{following} a set of definitions and declarations AnaGram needs for the code it generates. However, if the \emph{first} statement in your \index{Syntax file}syntax file is a block of embedded C, it will \emph{precede} AnaGram's definitions and declarations. This block of embedded C is called the \index{Prologue}\index{C prologue}``C prologue''. There are two main reasons for this special treatment. First, you may want to have a title and \index{Copyright notice}copyright notice in your parser. If you include them in an initial block of embedded C they will be right at the beginning of both your syntax file and your parser file. Second, if some of your tokens have data type\index{Token}\index{Data type}s other than those predefined in C or C++, you may include the definitions here, so they will be available to the code AnaGram generates. AnaGram scans embedded C only insofar as is necessary to find the closing right brace. Therefore any braces used within embedded C must balance properly. AnaGram skips braces enclosed in character constants and string literals, as well as braces enclosed in comments. It also recognizes C++ style comments that begin with \agcode{//}. \index{Comments}Treatment of nested versus non-nested comments is controlled by the \index{Nest comments}\index{Configuration switches}\agparam{nest comments} configuration parameter. AnaGram will use the status of this parameter in effect at the beginning of the section of embedded C. AnaGram, of course, can be confused by unterminated strings, unbalanced brackets, and unterminated comments. The most likely outcome, in such a situation, is that AnaGram will encounter the end of file looking for the end of the embedded C. Should this happen, AnaGram will identify the beginning of the piece of embedded C which caused the problem. The code you include as embedded C, of course, has to coexist with the code AnaGram generates. In order to keep the potential for conflicts to a minimum, all variables and functions which AnaGram defines begin either with the name of your parser or with the letters \agcode{ag{\us}}. You should avoid variable names which begin with these letters. Reduction procedures are copied to the \index{Parser file}\index{File}parser file in the order in which they are defined \emph{following} all of the embedded C. Thus your reduction procedures may freely use variables and macros defined anywhere in your embedded C. \subsection{Configuration Sections} \index{Configuration section} A configuration section is a special section of your syntax file enclosed in brackets. Within a configuration section you may set the values of configuration parameters or switches, or you may use one or more of several available attribute statements to specify special treatment for certain tokens. There can be as many or as few configuration sections in your syntax file as you wish. Each configuration section must begin on a new line. Any AnaGram statement which follows a configuration section must also begin on a new line. Within a configuration section, each parameter setting and each attribute statement must begin on a new line. The rules for using comments and continuation lines are the same as for the rest of AnaGram. Configuration parameters control the way AnaGram interprets your syntax file and the way it builds your parser. A full discussion of the use of configuration parameters, including a complete discussion of each parameter and its default value, is given in Appendix A. \index{Attribute statements}\index{Statement} Attribute statements comprise the \index{Precedence declarations}precedence declarations \agparam{left}, \agparam{right}, and \agparam{nonassoc}; the \agparam{sticky} declaration; the \agparam{distinguish keywords} statement; the \agparam{hidden} declaration; the \agparam{disregard} and \agparam{lexeme} statements; the \agparam{enum} statement; the \index{Reserve keywords}\agparam{reserve keywords} declaration; and the \index{Rename macro}\agparam{rename macro} statement. The precedence declarations and the \index{Sticky declaration}\index{Declaration}\agparam{sticky} declaration may be used to resolve conflicts in your grammar. The \agparam{distinguish keywords} statement may be used to control keyword recognition. The \index{Hidden declaration}\index{Declaration}\agparam{hidden} declaration causes certain token names not to be used when your parser produces \index{Syntax error}\index{Errors}\index{Error messages}syntax error messages. You may use the \agparam{disregard} and \agparam{lexeme} statements to cause your parser to skip automatically over certain tokens in its input. The \agparam{enum} statement is almost identical to the enum statement in C. It can be used to assign names to input codes in grammars which are taking input from a \index{Lexical scanner}lexical scanner or another parser. The \index{Reserve keywords}\agparam{reserve keywords} declaration allows you to specify certain keywords as reserved words. The \index{Rename macro}\agparam{rename macro} statement allows you to override the names AnaGram uses for various macro definitions it creates in the code it generates. Attribute statements are discussed below. Except for \agparam{disregard} and \agparam{rename macro} statements, attribute statements accept lists of operands enclosed in braces (\bra \ket) and separated by commas. A dangling comma following the last item in a list will be ignored. \subsection{Setting Configuration Parameters} \index{Configuration parameters}\index{Parameters} Each configuration parameter has a name that follows the AnaGram conventions for symbol names, except that AnaGram ignores case\index{Case sensitivity} when looking up configuration parameter names. There are a number of varieties of configuration parameters. The simplest, \index{Configuration switches}\index{Switches}configuration switches, simply turn some feature of AnaGram on or off. These parameters need simply be stated to turn the feature on, or negated with the tilde (\agcode{\~{}}) to turn the feature off: \begin{indentingcode}{0.4in} nest comments \end{indentingcode} causes AnaGram to allow nested comments, and \begin{indentingcode}{0.4in} \~{}nest comments \end{indentingcode} causes AnaGram to disallow nested comments. You may also set or reset configuration switches with explicit on or off values: \begin{indentingcode}{0.4in} nest comments = on nest comments = off \end{indentingcode} The remaining configuration parameters are assigned values using a simple assignment statement. Depending on the parameter, the value it takes may be the name of a token, a C variable name, a C or C++ data type, a string constant or an integer. String constants are written using the same rules as keyword strings, described above. \begin{indentingcode}{0.4in} grammar token = program parser name = widget default token type = void * header file name = "widget.h" parser stack size = 50 \end{indentingcode} A number of string-valued \index{Configuration parameters}configuration parameters are used to determine file names and variable names. In these parameters, the \index{\#}``\#'', \index{\_dol}``\$'', and ``\index{ \_prc}\%'' characters are used as wild cards. In file name specifications and the specification of the name of your parser, ``\#'' will be replaced by the name of your syntax file. In other function and variable names AnaGram creates while building your parser, ``\$'' will be replaced by the name of your parser. When building enumeration constants for the names of the tokens in your grammar, ``\%'' will be replaced by the name of the token. Note that when entering a Windows/DOS path name as a value for a file name parameter you must quote any backslashes in the path name. For example, \begin{indentingcode}{0.4in} coverage file name = "f:{\bs\bs}sna{\bs\bs}foo.nrc" \end{indentingcode} \subsection{Precedence Declarations} \index{Precedence declarations} AnaGram allows you to resolve shift-reduce conflicts by assigning precedence levels to operators. There are three precedence declarations available, beginning with the keywords \index{Left}\agparam{left}, \index{Right}\agparam{right}, and \index{Nonassoc}\agparam{nonassoc} respectively. Each such declaration consists of the appropriate keyword and a list of tokens enclosed in braces (\bra \ket). All the tokens in the list have the same precedence, higher than tokens in any previous declaration and lower than in any subsequent declaration. If the keyword is \agparam{left}, the tokens will group to the left. If it is \agparam{right}, they will group to the right. If it is \agparam{nonassoc} (for non-associative) no grouping will be assumed. Precedence declarations must be included in a configuration section. Here are precedence declarations appropriate to a simple desk calculator program: \begin{indentingcode}{0.4in} {}[ left \bra '+', '-' \ket left \bra star, '/', '\%' \ket right \bra unary minus \ket ] unary minus = '-' \end{indentingcode} Note that \agcode{unary minus} and \agcode{'-'} can have different precedence. Precedence declarations are one of the few instances in AnaGram where the \index{Statements}\index{Order of statements}order of statements is significant. The use of precedence declarations is discussed in Chapter 9. \subsection{``Sticky'' Declarations} \index{Sticky declaration}\index{Declaration} AnaGram provides another means for resolving shift-reduce conflicts. You may characterize any token as ``sticky''. Then, in the case of a \index{Shift-reduce conflict}\index{Conflicts}shift-reduce conflict where a ``sticky'' token is the last token in the input buffer, the conflict will be resolved by selecting the shift operation. Intuitively, you may think of this as though the ``sticky'' token adheres to and draws in any subsequent input that it can. ``Sticky'' declarations are included in configuration sections. They begin with the keyword \agcode{sticky} followed by a list of tokens, separated by commas inside braces (\bra \ket). Suppose, for instance, you wished to pick up a line of text, skipping any leading space or tab characters. You might write the following syntax: \begin{indentingcode}{0.4in} white space = ' ' + '{\bs}t' text char -> \~{}'{\bs}n':c = do{\us}something(c); line -> leading white space, text char?..., '{\bs}n' leading white space -> -> leading white space, white space \end{indentingcode} Unfortunately, this syntax is ambiguous, since space and tab are legitimate instances of both leading white space and text char. What you really want to do is to skip white space until you find a non-blank character and then you want to accept all characters to the end of the line. There are two ways to address the problem. The first is to define a special token for the first non-blank character and, using it, to write an unambiguous grammar. This approach, while laudable, is tedious and prolix. Instead, use \agparam{sticky} to resolve the problem: \begin{indentingcode}{0.4in} {}[ sticky \bra leading white space \ket ] \end{indentingcode} Now when AnaGram analyzes your grammar, and encounters the ambiguity, it will understand that a blank or tab that could be treated either as leading white space or the as the first text character should be treated as white space. Since \agcode{leading white space} is ``sticky'', any subsequent white space adheres to it. As with conflicts resolved with precedence levels, AnaGram lists all conflicts that it resolves using \agcode{sticky} in the \index{Resolved Conflicts}\index{Window}\agwindow{Resolved Conflicts Table}, so you can verify that the conflicts have been correctly resolved. An important use of sticky tokens is to inhibit the recognition of following \index{Keywords}keywords. Following a sticky token, a keyword, which, according to your grammar, would otherwise be legitimate input, will not be recognized if a shift action is possible for the first character of the keyword. For example, imagine that \agcode{name} has been defined in the conventional way, and there exists a production with name followed immediately by the keyword \agcode{int}. Then if, in your input, the word \agcode{print} were to occur, your grammar would parse it as a name, \agcode{pr}, followed by the keyword \agcode{int}. If you make \agcode{name} sticky, however, the first letter of \agcode{int} will be seen to be an acceptable character for \agcode{name} and the keyword will not be recognized. Your parser will then recognize the \agcode{name} as \agcode{print}. \subsection{Distinguish Keywords Statement} \index{Distinguish keywords}\index{Keywords} Distinguish keywords statements are occasionally needed to prevent keyword recognition. You may, for example, wish to prevent the recognition of the keyword \agcode{int} when it occurs embedded in a word such as \agcode{interval}. Of course, you need to do this only if both the keyword and the other word are both legitimate input at the same point in your grammar. A distinguish keywords statement can prevent recognition of a keyword which is embedded in another word provided at least one character of the other word follows the keyword. The distinguish keywords statement has the form: \begin{indentingcode}{0.4in} distinguish keywords \bra \codemeta{list of character sets} \ket \end{indentingcode} AnaGram compares all the characters in each keyword to the characters included in each character set in turn. If it finds that all the characters in a keyword are members of a particular set, it tells the keyword recognition logic to try to match the keyword only against the longest sequence of characters drawn from the specified set. In other words, in order for a keyword to be recognized, the keyword \emph{must} be followed by a character \emph{not} in the set. The set associated with a keyword is the first one in the list which contains all the characters found in the keyword. If you have more than one \agparam{distinguish keywords} statement in your grammar, the lists are tried in the order in which they appear in the grammar. The purpose of the \agparam{distinguish keywords} statement is to enable your parser to distinguish a keyword from the same sequence of characters embedded within another sequence. Thus suppose that \agcode{int} is a keyword, and, according to your grammar, could appear in the same place as the word \agcode{integral}. If you don't want it to be recognized as a keyword in these circumstances, you would write the following distinguish statement: \begin{indentingcode}{0.4in} distinguish keywords \bra 'a-z'+'A-Z' \ket \end{indentingcode} To also inhibit recognition of \agcode{int} within \agcode{print}, you would combine the use of the distinguish keywords statement with the \agparam{sticky} declaration. \subsection{``Hidden'' Declarations} \index{Hidden declaration}\index{Declaration} AnaGram provides an optional \index{Error diagnosis}error diagnosis feature for your parser (see Chapter 9). The \agparam{hidden} declaration allows you to identify tokens that you do not wish to be used in making up \index{Diagnostic messages}diagnostic messages. These tokens are tokens whose names would not mean anything to your users. The format of a ``hidden'' declaration is the same as that of precedence and ``sticky'' declarations. Within a configuration section, the keyword ``hidden'' is followed by a list of tokens. For example: \begin{indentingcode}{0.4in} {}[ hidden \bra comment head \ket ] comment -> comment head, "*/" comment head -> "/*" -> comment head, \~{}eof \end{indentingcode} This is an AnaGram representation of ANSI standard C comments (non-nested). In this example the token \agcode{comment head} exists only for convenience in writing the grammar and has no particular meaning to an end user. On the other hand, he knows what the word \agcode{comment} refers to. The ``hidden'' attribute will cause AnaGram's diagnostic builder, by backing up the stack until it finds a non-hidden token, to eschew \agcode{comment head} in favor of \agcode{comment}. % XXX eschew obfuscation. how about ``avoid''? \subsection{Disregard Statement} The purpose of the \index{Disregard statement}\index{Statement}\agparam{disregard} statement is to skip over uninteresting \index{White space}white space and comments in your input files. The disregard statement allows you to specify a token that should be passed over in the input to your parser. The statement takes the form: \begin{indentingcode}{0.4in} disregard ws \end{indentingcode} where \agcode{ws} is a token name or character set. Disregard statements may be placed in any configuration section. You may have more than one disregard statement in your grammar. If you do, AnaGram will create a shell production. For example, suppose you write: \begin{indentingcode}{0.4in} {}[ disregard alpha disregard beta ] \end{indentingcode} AnaGram will proceed as though you had written: \begin{indentingcode}{0.4in} gamma -> alpha | beta {}[ disregard gamma ] \end{indentingcode} It frequently happens that you wish your parser to disregard blanks or comments, except that white space within names, numbers, strings, and other elementary constructs is subject to special rules and thus should not be disregarded blindly. In this case, you can use the \agparam{lexeme} statement to declare these constructs off limits for the disregard statement. Within these constructs, the disregard statement will be inoperative and the admissibility of white space will be determined solely by the productions which define these constructs. Outside those productions which define lexemes, you should not generally use a token which is supposed to be disregarded. If you do, your grammar will have conflicts, since the token could satisfy both the explicit usage and the implicit rules set up by the disregard statement. Such conflicts, however, are resolved automatically in favor of your explicit use of the token. The conflicts will appear in the \agwindow{Resolved Conflicts} window. % XXX I'm not sure that's still true. In order to implement the disregard statement AnaGram will redefine some tokens in your grammar. For example, \agcode{+} may be redefined to consist of a simple plus sign followed by optional white space: \begin{indentingcode}{0.4in} '+' -> '+'\%, white space?... \end{indentingcode} The percent sign is used to indicate the original, simple plus sign without the optional white space attached. You will probably notice the percent sign appearing in some windows and traces. In earlier versions of AnaGram, the degree sign, ``\agcode{\degrees}'', was used rather than ``\agcode{\%}''. \subsection{Lexeme Statement} The ``lexeme'' \index{Statement}\index{Lexeme statement}statement is used to fine-tune the disregard statement. The lexeme statement takes the form: \begin{indentingcode}{0.4in} {}[ lexeme \bra \codemeta{nonterminal token list} \ket ] \end{indentingcode} where \textit{nonterminal token list} is a list of nonterminal tokens separated by commas. Lexeme statements may be placed in any configuration section, and there may be any number of them. When you specify that a token is to be disregarded, AnaGram rewrites your grammar so that the token will be passed over whenever it occurs at the beginning of a file or following a lexical unit, or \agterm{lexeme}. If you have no \agparam{lexeme} statement, then the lexemes in your grammar are just the terminal tokens. The \agparam{lexeme} statement allows you to specify that certain nonterminal tokens are also to be treated as lexemes. This means that the disregard token will be skipped following the lexeme, but not between the characters that constitute the lexeme. Lexemes correspond to the tokens that a lexical scanner, if you were using one, would commonly identify and pass to a parser as single tokens. You don't usually wish to disregard white space within these tokens. For example, in a grammar for a conventional programming language where blank characters are to be disregarded, you might include: \begin{indentingcode}{0.4in} {}[ lexeme \bra string, character constant, name, number \ket ] \end{indentingcode} since blank characters must not be overlooked within strings and character constants and should not be permitted within names or numbers. Normally, AnaGram considers the disregard token to be optional; however there are circumstances where treating the disregard token as optional would lead to conflicts: two successive names, or two successive numbers, for example. In this case, you would like to require that the lexemes be separated by instances of the disregard token. To do this, simply set the \index{Distinguish lexemes}\index{Configuration switches} \agparam{distinguish lexemes} configuration switch. When this switch is set, AnaGram will ensure that disregard tokens will be required in those situations where making them optional would lead to conflicts. White space may be used explicitly within definitions of lexeme tokens in your grammar if desired, without causing conflicts. Thus, if you wish to allow embedded space in variable names, you might write: \begin{indentingcode}{0.4in} {}[ disregard space lexeme \bra variable name \ket ] space = ' ' + '{\bs}t' letter = 'a-z' + 'A-Z' digit = '0-9' variable name -> letter -> variable name, letter + digit -> variable name, space..., letter + digit \end{indentingcode} \subsection{Enum Statement} \index{Enum statement}\index{Enumeration}\index{Token} The \agparam{enum} statement follows rules nearly identical to those for C and C++. This makes it possible to copy an enum statement from your syntax file to a program file written in either C or C++, without any need for editing. The only differences are that AnaGram makes no provision for blank lines within the enumeration list, nor does it accept a type name. The \agparam{enum} statement is equivalent to a corresponding set of definition statements. It is especially useful when a parser is accepting token input from another program, a \index{Lexical scanner}lexical scanner, for example. Using the enum statement you may conveniently define all the identification codes for the input tokens. Each entry in an enum statement may be either a name, or a name followed by an ``='' sign and a character representation. If there is a character representation the name is assigned the value of the specified character. Otherwise it is assigned a value one more than that assigned to the previous name. If the first name in the list is not given an explicit value, it will be given the value zero. For example: \begin{indentingcode}{0.4in} {}[ enum \bra eof, a,b,c, blank = '\ ', x, y \ket ] \end{indentingcode} is equivalent to the following definition statements \begin{indentingcode}{0.4in} eof = 0 a = 1 b = 2 c = 3 blank = '\ ' x = 33 y = 34 \end{indentingcode} \subsection{Subgrammar Declarations} \index{Subgrammar declaration}\index{Declaration} A \agparam{subgrammar} declaration can be a useful way to deal with conflicts in certain situations. It tells AnaGram to treat the tokens listed in the declaration as though they were each grammar tokens, each specifying a complete subgrammar in itself, and, in determining shift and reduction actions, to ignore the usage of the tokens in the larger grammar. In some cases it is perfectly reasonable to ignore usage. The most common example occurs when building a lexical scanner for a language such as C as in the example in Section 7.4.4. In this case, you can write a complete grammar for a C token with no difficulty. But if you try to extend it to a sequence of tokens, you get scores of conflicts. This situation arises because you specify that any C token can follow another, when in actual practice, an identifier, for example, cannot follow another identifier without some intervening space or punctuation. It is theoretically possible, but in practice quite awkward, to write a grammar for a sequence of tokens so that there are no conflicts. The subgrammar declaration provides a way around this problem by telling AnaGram that when it is looking for reducing tokens for any rule produced directly or indirectly by a subgrammar token, it should disregard the usage of the token and only consider usage internal to the definition of the subgrammar token, as though the subgrammar token were the start token of the grammar. The subgrammar declaration is made in a configuration section and consists of the keyword \agcode{subgrammar} followed by a list of one or more nonterminal token names, separated by commas and enclosed in braces (\bra \ket). For example: \begin{indentingcode}{0.4in} {}[ subgrammar \bra C token, word \ket ] \end{indentingcode} Since the subgrammar statement changes the way AnaGram determines reducing tokens, it should be used with caution. You should be sure that the conflicts you are eliminating are really inconsequential. \subsection{Reserve Keywords Declaration} \index{Reserve keywords}\index{Keywords}\index{Keyword anomalies} The \agparam{reserve keywords} declaration can be used to specify a list of keywords that are reserved and cannot be used except as explicitly specified in the grammar. This enables AnaGram to avoid issuing meaningless keyword anomaly diagnostics (see \S 7.5). AnaGram does not automatically presume that keywords are also reserved words, since in many grammars there is no need to specify reserved words. The reserve keywords declaration is made in a configuration section and consists of the words \agcode{reserve keywords} followed by a list of one or more keyword strings, separated by commas and enclosed in braces (\bra \ket). For example: \begin{indentingcode}{0.4in} {}[ reserve keywords \bra "int", "char", "float", "double" \ket ] \end{indentingcode} \subsection{Rename Macro Statement} \index{Rename macro}\index{Macros} AnaGram uses a number of macros in its generated code. It is possible, therefore, to run into naming collisions with other components of your program. The \agparam{rename macro} statement allows you to change the name AnaGram uses for a particular macro to avoid these problems. For example, the Windows NT operating system uses \agcode{CONTEXT} structures to perform various internal operations. If you use the context tracking option (see \S 9.5.4) your parser will have a macro called \agcode{CONTEXT}. To avoid the name collision, add the following statement to any configuration section in your grammar: \begin{indentingcode}{0.4in} rename macro CONTEXT AG{\us}CONTEXT \end{indentingcode} Then, simply use \agcode{AG{\us}CONTEXT} where you would otherwise have used \agcode{CONTEXT}.