rustlr 0.4.0

LR Parser Generator with Advanced Options
Documentation
<!doctype html>
<html>
<head>
<meta charset='UTF-8'><meta name='viewport' content='width=device-width initial-scale=1'>
<title>chapter4</title>
</head>
<body><h2 id='chapter-4-automatically-generating-the-ast'>Chapter 4: Automatically Generating the AST</h2>
<p>One of the advantages of writing ambiguous grammars, e.g., <code>E--&gt;E+E</code> instead of <code>E--&gt;E+T</code>, is that it becomes easier to generate reasonable abstract syntax representations automatically.  Extra symbols such as <code>T</code> that are required for unambiguous grammars generally have no meaning at the abstract syntax level and will only lead to convoluted ASTs.  Since version 0.2.8, rustlr is capable of automatically generating the data structures (enums) for the abstract syntax of a language as well as the semantic actions required to create instances of those structures.  For beginners new to writing grammars and parsers, we <strong>do not</strong> recommend starting with an automatically generated AST.  The user must understand clearly the relationship between concrete and abstract syntax and the best way to learn this relationship is by writing ASTs by hand, as demonstrated in the previous two chapters.  Even with Rustlr capable of generating nearly everything one might need from a parser, it is still likely that careful fine tuning will be required.</p>
<p>We redo the enhanced calculator example from <a href='https://cs.hofstra.edu/~cscccl/rustlr_project/chapter2.html' target='_blank'>Chapter 2</a>.  The following grammar is found <a href='https://cs.hofstra.edu/~cscccl/rustlr_project/autocalc/calcauto.grammar'>here</a>.</p>
<pre><code class='language-rust' lang='rust'>lifetime &#39;lt
nonterminals Expr ES
terminals + - * / ( ) = ;
terminals let in
typedterminal int i64
typedterminal var &amp;&#39;lt str
topsym ES
resync ;

left * 500
left / 500
left + 400
left - 400

Expr:Val --&gt; int
Expr:Var --&gt; var
Expr:Letexp --&gt; let var = Expr in Expr
Expr:Plus --&gt; Expr + Expr
Expr:Minus --&gt; Expr - Expr
Expr:Div --&gt; Expr / Expr
Expr:Times --&gt; Expr * Expr
Expr:Neg --&gt; - Expr
# override auto-generated creation of abstract syntax, but type matters
Expr --&gt; ( Expr:e )  { e }
ES:nil --&gt;
ES:cons --&gt; Expr ; ES

lexvalue int Num(n) n
lexvalue var Alphanum(x) x
lexattribute set_line_comment(&quot;#&quot;)
EOF

</code></pre>
<p>Note the following differences between this grammar and the one presented in <a href='https://cs.hofstra.edu/~cscccl/rustlr_project/chapter2.html' target='_blank'>Chapter 2</a>:</p>
<ol>
<li>There are no semantic actions but for one of the rules</li>
<li>There is no &quot;absyntype&quot; or &quot;valuetype&quot; declaration</li>
<li>Only the types of values carried by certain terminal symbols must be declared (with <code>typedterminal</code>).</li>
<li>The non-terminal symbol on the left-hand side of a production rule may carry a label.  This label will become the name of the enum variant to be created.</li>

</ol>
<p>Process the grammar with <strong><code>rustlr calcauto.grammar -genabsyn</code></strong>.   Two files are created.  Besides <strong><a href='https://cs.hofstra.edu/~cscccl/rustlr_project/autocalc/src/calcautoparser.rs'>calcautoparser.rs</a></strong> there will be a <strong><a href='https://cs.hofstra.edu/~cscccl/rustlr_project/autocalc/src/calcauto_ast.rs'>calcauto_ast.rs</a></strong> with the following (principal) contents:</p>
<pre><code class='language-rust' lang='rust'>#[derive(Debug)]
pub enum ES&lt;&#39;lt&gt; {
  cons(LBox&lt;Expr&lt;&#39;lt&gt;&gt;,LBox&lt;ES&lt;&#39;lt&gt;&gt;),
  nil,
  ES_Nothing(&amp;&#39;lt ()),
}
impl&lt;&#39;lt&gt; Default for ES&lt;&#39;lt&gt; { fn default()-&gt;Self { ES::ES_Nothing(&amp;()) } }

#[derive(Debug)]
pub enum Expr&lt;&#39;lt&gt; {
  Neg(LBox&lt;Expr&lt;&#39;lt&gt;&gt;),
  Div(LBox&lt;Expr&lt;&#39;lt&gt;&gt;,LBox&lt;Expr&lt;&#39;lt&gt;&gt;),
  Letexp(&amp;&#39;lt str,LBox&lt;Expr&lt;&#39;lt&gt;&gt;,LBox&lt;Expr&lt;&#39;lt&gt;&gt;),
  Minus(LBox&lt;Expr&lt;&#39;lt&gt;&gt;,LBox&lt;Expr&lt;&#39;lt&gt;&gt;),
  Val(i64),
  Times(LBox&lt;Expr&lt;&#39;lt&gt;&gt;,LBox&lt;Expr&lt;&#39;lt&gt;&gt;),
  Expr_8(LBox&lt;Expr&lt;&#39;lt&gt;&gt;),
  Plus(LBox&lt;Expr&lt;&#39;lt&gt;&gt;,LBox&lt;Expr&lt;&#39;lt&gt;&gt;),
  Var(&amp;&#39;lt str),
  Expr_Nothing(&amp;&#39;lt ()),
}
impl&lt;&#39;lt&gt; Default for Expr&lt;&#39;lt&gt; { fn default()-&gt;Self { Expr::Expr_Nothing(&amp;()) } }

</code></pre>
<p>An enum is created for each non-terminal symbol of the grammar, with the same name as the non-terminal.  There is, essentially, an enum variant for each production rule of the grammar.  The names of the variants are derived from the labels given to the left-hand side nonterminal, or are automatically generated (e.g. <code>Expr_8</code> represents the rule-8 variant of type <code>Expr</code>).<sup class='md-footnote'><a href='#dfref-footnote-1' name='ref-footnote-1'>1</a></sup> The &#39;absyntype&#39; of the grammar will be set to <code>ES</code>, the symbol declared to be &#39;topsym&#39;.  Although the generated parser may not be very readable, rustlr also generated semantic actions that create instances of these enum types.  For example, the rule <code>Expr:Plus --&gt; Expr + Expr</code> will have semantic action equivalent to one created from:</p>
<pre><code>Expr --&gt; Expr:[a] + Expr:[b] {Plus(a,b)}
</code></pre>
<p>Recall from <a href='https://cs.hofstra.edu/~cscccl/rustlr_project/chapter2.html' target='_blank'>Chapter 2</a> that a label of the form <code>[a]</code>means that the semantic value associated with the symbol is enclosed in an <a href='https://docs.rs/rustlr/latest/rustlr/generic_absyn/struct.LBox.html' target='_blank'>LBox</a>.  However, there are cases where one might want to override the automatically generated action, as for the rule <code>Expr --&gt; ( Expr )</code>.  The parentheses are of no use at the abstract syntax level and the most appropriate action would be to return the same value as the expression on the right-hand side.  The automatically generated action would have created an additional LBox.  It is also possible to override the automatic generation of the type of a grammar symbol.  In case of ES, the labels &#39;nil&#39; and &#39;cons&#39; are sufficient for rustlr to create a linked-list data structure.  However, the right-recursive grammar rule is not optimal for LR parsing.  One might wish to use a left-recursive rule and a Rust vector to represent a sequence of expressions.  This can be done by making the following changes to the grammar.  First, change the declaration of the non-terminal symbol <code>ES</code> as follows:</p>
<pre><code>nonterminal ES Vec&lt;LBox&lt;Expr&lt;&#39;lt&gt;&gt;&gt;
</code></pre>
<p>Then replace the two production rules for <code>ES</code> with the following:</p>
<pre><code class='language-rust' lang='rust'>ES --&gt; Expr:[e] ; { vec![e] }
ES --&gt; ES:v Expr:[e] ;  { v.push(e); v }

</code></pre>
<h4 id='adding-new-rules-with---and-'>Adding New Rules with *, + and ?</h4>
<p>A relatively new feature of rustlr (since verion 0.2.8) allows the use of regular-expression style symbols *, + and ? to automatically generate new production rules.  However, this ability is currently rather limited and is only guaranteed to work in the automatic <code>-genabsyn</code> mode.  Another way to achieve the same effects as the above is to use the following alternative grammar declarations:</p>
<pre><code>nonterminal ES Vec&lt;LBox&lt;Expr&lt;&#39;lt&gt;&gt;&gt;
nonterminal ES1 *Expr
ES1 --&gt; Expr:e ; {e}
ES --&gt; ES1+:v { v }
</code></pre>
<p>The special type declaration <strong><code>*Expr</code></strong> means that the type of the nonterminal <code>ES1</code> is copied from the type of <code>Expr</code>, which in this case is automatically generated as <code>Expr&lt;&#39;lt&gt;</code>.  The expression <strong><code>ES1+</code></strong> means a sequence of at least one <code>ES1</code> derivations.  This is done by generating a new non-terminal symbol with associated type <code>Vec&lt;LBox&lt;Expr&lt;&#39;lt&gt;&gt;&gt;</code>. The optional label v will be bound to such a value.  A <strong><code>*</code></strong> would mean zero or more <code>ES1</code> derivations, producing the same vector type,  and a <strong><code>?</code></strong> will mean  one or zero derivations with type <code>Option&lt;LBox&lt;Expr&lt;&#39;lt&gt;&gt;&gt;</code>.  Currently, the *, + and ? symbols can only be placed after exactly one grammar symbol, thus the extra symbol and production for <code>ES1</code> is required.  Additionally, the label given for such an expression cannot be a pattern such as <code>[v]</code>or something enclosed inside <code>@...@</code>.  These restrictions should eventually be eliminated in future releases.</p>
<h5 id='invoking-the-parser'>Invoking the Parser</h5>
<p>Since the grammar also contains lexer generation directives, all we need to do is to write the procedures that interpret the AST (see <a href='https://cs.hofstra.edu/~cscccl/rustlr_project/autocalc/src/main.rs'>main</a>).  The procedure to invoke the parser is the same as described in <a href='https://cs.hofstra.edu/~cscccl/rustlr_project/chapter3.html' target='_blank'>Chapter 3</a>, using the <code>parse_with</code> or <code>parse_train_with</code> functions:</p>
<pre><code class='language-rust' lang='rust'>   let mut scanner = calcautoparser::calcautolexer::from_str(&quot;2*3+1;&quot;);
   let mut parser = calcautoparser::make_parser();
   let result = calcautoparser::parse_with(&amp;mut parser, &amp;mut scanner);
   let tree = result.unwrap_or_else(|x|{println!(&quot;Parsing errors encountered; results are partial..&quot;); x});
   println!(&quot;\nAST: {:?}\n&quot;,&amp;tree);
   
</code></pre>
<p>Please note that all generated enums for the grammar will attempt to derive the Debug trait (as well as implement the Default trait).  </p>
<p>&nbsp;</p>
<h4 id='generating-a-parser-for-c'>Generating a Parser for C</h4>
<p>As a larger example, we applied the <code>-genabsyn</code> feature of rustlr to the ANSI C Yacc grammar published in 1985 by Jeff Lee, which was converted to rustlr syntax and found <a href='https://cs.hofstra.edu/~cscccl/rustlr_project/cparser/cauto.grammar'>here</a>.  The raw grammar produced a shift-reduce conflict caused by the <em>dangling else</em> problem, which we fixed by giving  &#39;else&#39; a higher precedence than &#39;if&#39;.  The raw grammar contained a few other issues we have not addressed, most notably when an identifier should be considered as <code>TYPE_NAME</code>. Manual fine tuning will definitely be required, as one should expect.  However, the generated AST is at least a good starting point.  In forming the enum types, for production rules without a left-hand side label, rustlr will also sometimes use the name of an alpha-numeric terminal symbol to create the name of the enum variant (if it is the first symbol on the right-hand side).</p>
<p>The AST enums are found <a href='https://cs.hofstra.edu/~cscccl/rustlr_project/cparser/src/cauto_ast.rs'>here</a> and the generated parser <a href='https://cs.hofstra.edu/~cscccl/rustlr_project/cparser/src/cautoparser.rs'>here</a>.</p>
<p>&nbsp;</p>



















<p>&nbsp;</p>
<p>&nbsp;</p>
<div class='footnotes-area'  ><hr/>
<div class='footnote-line'><span class='md-fn-count'>1</span> Each enum has a <code>_Nothing(&amp;&#39;lt ())</code> variant.  This is used to implement the Default trait.  The lifetime parameter exists so that all enums can be parameterized with a lifetime, if one was declared for the grammar.  Without the dummy reference one would have to compute a closure over the grammar to determine which enums require lifetimes and which do not: something that&#39;s determined to be too expensive relative to its importance. <a name='dfref-footnote-1' href='#ref-footnote-1' title='back to document' class='reversefootnote' >↩</a></div></div></body>
</html>