aboutsummaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authorPatrick Walton <[email protected]>2010-11-24 14:42:01 -0800
committerPatrick Walton <[email protected]>2010-11-24 14:42:01 -0800
commit756880a5f5e5df2474a9770d27fa3751b0be4916 (patch)
treea3afab0a271b5d9e10dba1f876ac262f1c6671f9 /src
parentrustc: Parse tag items. Currently segfaults in copy glue. (diff)
downloadrust-756880a5f5e5df2474a9770d27fa3751b0be4916.tar.xz
rust-756880a5f5e5df2474a9770d27fa3751b0be4916.zip
rustc: Parse simple patterns
Diffstat (limited to 'src')
-rw-r--r--src/comp/front/ast.rs10
-rw-r--r--src/comp/front/parser.rs77
2 files changed, 87 insertions, 0 deletions
diff --git a/src/comp/front/ast.rs b/src/comp/front/ast.rs
index 2a590d02..954440dd 100644
--- a/src/comp/front/ast.rs
+++ b/src/comp/front/ast.rs
@@ -39,6 +39,15 @@ type block = spanned[block_];
type block_ = rec(vec[@stmt] stmts,
hashmap[ident,uint] index);
+type pat = spanned[pat_];
+tag pat_ {
+ pat_wild;
+ pat_bind(ident);
+ pat_tag(ident, vec[@pat]);
+}
+
+type arm = rec(@pat pat, block block);
+
tag binop {
add;
sub;
@@ -104,6 +113,7 @@ tag expr_ {
expr_if(@expr, block, option.t[block], ann);
expr_while(@expr, block, ann);
expr_do_while(block, @expr, ann);
+ expr_alt(@expr, vec[arm], ann);
expr_block(block, ann);
expr_assign(@expr /* TODO: @expr|is_lval */, @expr, ann);
expr_field(@expr, ident, ann);
diff --git a/src/comp/front/parser.rs b/src/comp/front/parser.rs
index caff774b..16c6ed11 100644
--- a/src/comp/front/parser.rs
+++ b/src/comp/front/parser.rs
@@ -745,6 +745,38 @@ impure fn parse_do_while_expr(parser p) -> @ast.expr {
ret @spanned(lo, hi, ast.expr_do_while(body, cond, ast.ann_none));
}
+impure fn parse_alt_expr(parser p) -> @ast.expr {
+ auto lo = p.get_span();
+ expect(p, token.ALT);
+ expect(p, token.LPAREN);
+ auto discriminant = parse_expr(p);
+ expect(p, token.RPAREN);
+ expect(p, token.LBRACE);
+
+ let vec[ast.arm] arms = vec();
+ while (p.peek() != token.RBRACE) {
+ alt (p.peek()) {
+ case (token.CASE) {
+ p.bump();
+ expect(p, token.LPAREN);
+ auto pat = parse_pat(p);
+ expect(p, token.RPAREN);
+ auto block = parse_block(p);
+ arms += vec(rec(pat=pat, block=block));
+ }
+ case (token.RBRACE) { /* empty */ }
+ case (?tok) {
+ p.err("expected 'case' or '}' when parsing 'alt' statement " +
+ "but found " + token.to_str(tok));
+ }
+ }
+ }
+
+ auto expr = ast.expr_alt(discriminant, arms, ast.ann_none);
+ auto hi = p.get_span();
+ ret @spanned(lo, hi, expr);
+}
+
impure fn parse_expr(parser p) -> @ast.expr {
alt (p.peek()) {
case (token.LBRACE) {
@@ -761,6 +793,9 @@ impure fn parse_expr(parser p) -> @ast.expr {
case (token.DO) {
ret parse_do_while_expr(p);
}
+ case (token.ALT) {
+ ret parse_alt_expr(p);
+ }
case (_) {
ret parse_assign_expr(p);
}
@@ -777,6 +812,48 @@ impure fn parse_initializer(parser p) -> option.t[@ast.expr] {
ret none[@ast.expr];
}
+impure fn parse_pat(parser p) -> @ast.pat {
+ auto lo = p.get_span();
+
+ auto pat = ast.pat_wild; // FIXME: typestate bug
+ alt (p.peek()) {
+ case (token.UNDERSCORE) { p.bump(); pat = ast.pat_wild; }
+ case (token.QUES) {
+ p.bump();
+ alt (p.peek()) {
+ case (token.IDENT(?id)) { p.bump(); pat = ast.pat_bind(id); }
+ case (?tok) {
+ p.err("expected identifier after '?' in pattern but " +
+ "found " + token.to_str(tok));
+ fail;
+ }
+ }
+ }
+ case (token.IDENT(?id)) {
+ p.bump();
+
+ let vec[@ast.pat] args;
+ alt (p.peek()) {
+ case (token.LPAREN) {
+ auto f = parse_pat;
+ args = parse_seq[@ast.pat](token.LPAREN, token.RPAREN,
+ some(token.COMMA), f, p).node;
+ }
+ case (_) { args = vec(); }
+ }
+
+ pat = ast.pat_tag(id, args);
+ }
+ case (?tok) {
+ p.err("expected pattern but found " + token.to_str(tok));
+ fail;
+ }
+ }
+
+ auto hi = p.get_span();
+ ret @spanned(lo, hi, pat);
+}
+
impure fn parse_let(parser p) -> @ast.decl {
auto lo = p.get_span();