diff options
| author | Tim Chevalier <chevalier@alum.wellesley.edu> | 2011-08-25 17:42:38 -0700 |
|---|---|---|
| committer | Tim Chevalier <chevalier@alum.wellesley.edu> | 2011-08-25 18:28:23 -0700 |
| commit | f841e894437843c142ebd7e0b0a18ed00ed52457 (patch) | |
| tree | fac7eed5de504e4246754bd7eea9a2e25c055090 /src/comp/syntax | |
| parent | d9bc3cb10c4d1e856998c8e35ce7d89e0d74f4d6 (diff) | |
| download | rust-f841e894437843c142ebd7e0b0a18ed00ed52457.tar.gz rust-f841e894437843c142ebd7e0b0a18ed00ed52457.zip | |
Support unchecked blocks
This patch supports the syntax
unchecked {
...
}
to disable purity checking within a block. Presumably it will only be
used within a declared "pure fn". However, there is no checking that it
doesn't occur elsewhere, and it would be harmless for it to do so.
I went with Lindsey's suggestion for the syntax, but it's subject to
change.
This allows you to write code that uses predicates that call arbitrary
Rust functions, but you must declare your intentions by wrapping it in
an unchecked { ... } block. The test case run-pass/unchecked-predicates.rs
demonstrates how to do that.
Diffstat (limited to 'src/comp/syntax')
| -rw-r--r-- | src/comp/syntax/ast.rs | 12 | ||||
| -rw-r--r-- | src/comp/syntax/ast_util.rs | 6 | ||||
| -rw-r--r-- | src/comp/syntax/fold.rs | 3 | ||||
| -rw-r--r-- | src/comp/syntax/parse/parser.rs | 26 | ||||
| -rw-r--r-- | src/comp/syntax/print/pprust.rs | 5 |
5 files changed, 42 insertions, 10 deletions
diff --git a/src/comp/syntax/ast.rs b/src/comp/syntax/ast.rs index 4caddb5e25b..d44baebb6f3 100644 --- a/src/comp/syntax/ast.rs +++ b/src/comp/syntax/ast.rs @@ -82,7 +82,8 @@ tag meta_item_ { type blk = spanned<blk_>; -type blk_ = {stmts: [@stmt], expr: option::t<@expr>, id: node_id}; +type blk_ = {stmts: [@stmt], expr: option::t<@expr>, + id: node_id, rules: check_mode}; type pat = {id: node_id, node: pat_, span: span}; @@ -223,6 +224,15 @@ tag expr_ { expr_uniq(@expr); } +/* +// Says whether this is a block the user marked as +// "unchecked" +tag blk_sort { + blk_unchecked; // declared as "exception to effect-checking rules" + blk_checked; // all typing rules apply +} +*/ + type mac = spanned<mac_>; tag mac_ { diff --git a/src/comp/syntax/ast_util.rs b/src/comp/syntax/ast_util.rs index 48bdfc47b68..517e252dd9c 100644 --- a/src/comp/syntax/ast_util.rs +++ b/src/comp/syntax/ast_util.rs @@ -184,10 +184,14 @@ fn eq_ty(a: &@ty, b: &@ty) -> bool { ret std::box::ptr_eq(a, b); } fn hash_ty(t: &@ty) -> uint { ret t.span.lo << 16u + t.span.hi; } fn block_from_expr(e: @expr) -> blk { - let blk_ = {stmts: [], expr: option::some::<@expr>(e), id: e.id}; + let blk_ = checked_blk([], option::some::<@expr>(e), e.id); ret {node: blk_, span: e.span}; } +fn checked_blk(stmts1: [@stmt], expr1: option::t<@expr>, id1: node_id) + -> blk_ { + ret {stmts: stmts1, expr: expr1, id: id1, rules: checked}; +} fn obj_field_from_anon_obj_field(f: &anon_obj_field) -> obj_field { ret {mut: f.mut, ty: f.ty, ident: f.ident, id: f.id}; diff --git a/src/comp/syntax/fold.rs b/src/comp/syntax/fold.rs index a91b2f31814..770ed307e14 100644 --- a/src/comp/syntax/fold.rs +++ b/src/comp/syntax/fold.rs @@ -255,7 +255,8 @@ fn noop_fold_method(m: &method_, fld: ast_fold) -> method_ { fn noop_fold_block(b: &blk_, fld: ast_fold) -> blk_ { ret {stmts: vec::map(fld.fold_stmt, b.stmts), expr: option::map(fld.fold_expr, b.expr), - id: b.id}; + id: b.id, + rules: b.rules}; } fn noop_fold_stmt(s: &stmt_, fld: ast_fold) -> stmt_ { diff --git a/src/comp/syntax/parse/parser.rs b/src/comp/syntax/parse/parser.rs index f7329bd3b3b..729338b4b30 100644 --- a/src/comp/syntax/parse/parser.rs +++ b/src/comp/syntax/parse/parser.rs @@ -837,7 +837,7 @@ fn parse_bottom_expr(p: &parser) -> @ast::expr { } else if p.peek() == token::BINOP(token::OR) { ret parse_fn_block_expr(p); } else { - let blk = parse_block_tail(p, lo); + let blk = parse_block_tail(p, lo, ast::checked); ret mk_expr(p, blk.span.lo, blk.span.hi, ast::expr_block(blk)); } } else if eat_word(p, "if") { @@ -860,6 +860,10 @@ fn parse_bottom_expr(p: &parser) -> @ast::expr { ret parse_fn_expr(p, ast::proto_block); } else if eat_word(p, "lambda") { ret parse_fn_expr(p, ast::proto_closure); + } else if eat_word(p, "unchecked") { + expect(p, token::LBRACE); + let blk = parse_block_tail(p, lo, ast::unchecked); + ret mk_expr(p, blk.span.lo, blk.span.hi, ast::expr_block(blk)); } else if p.peek() == token::LBRACKET { p.bump(); let mut = parse_mutability(p); @@ -876,7 +880,7 @@ fn parse_bottom_expr(p: &parser) -> @ast::expr { ret mk_mac_expr(p, lo, p.get_hi_pos(), ast::mac_embed_type(ty)) } else if p.peek() == token::POUND_LBRACE { p.bump(); - let blk = ast::mac_embed_block(parse_block_tail(p, lo)); + let blk = ast::mac_embed_block(parse_block_tail(p, lo, ast::checked)); ret mk_mac_expr(p, lo, p.get_hi_pos(), blk); } else if p.peek() == token::ELLIPSIS { p.bump(); @@ -1309,7 +1313,7 @@ fn parse_fn_expr(p: &parser, proto: ast::proto) -> @ast::expr { fn parse_fn_block_expr(p: &parser) -> @ast::expr { let lo = p.get_last_lo_pos(); let decl = parse_fn_block_decl(p); - let body = parse_block_tail(p, lo); + let body = parse_block_tail(p, lo, ast::checked); let _fn = {decl: decl, proto: ast::proto_block, body: body}; ret mk_expr(p, lo, body.span.hi, ast::expr_fn(_fn)); } @@ -1664,12 +1668,20 @@ fn stmt_ends_with_semi(stmt: &ast::stmt) -> bool { fn parse_block(p: &parser) -> ast::blk { let lo = p.get_lo_pos(); - expect(p, token::LBRACE); - be parse_block_tail(p, lo); + if eat_word(p, "unchecked") { + be parse_block_tail(p, lo, ast::unchecked); + } + else { + expect(p, token::LBRACE); + be parse_block_tail(p, lo, ast::checked); + } } +// Precondition: already parsed the '{' or '#{' +// I guess that also means "already parsed the 'impure'" if +// necessary, and this should take a qualifier. // some blocks start with "#{"... -fn parse_block_tail(p: &parser, lo: uint) -> ast::blk { +fn parse_block_tail(p: &parser, lo: uint, s: ast::check_mode) -> ast::blk { let stmts: [@ast::stmt] = []; let expr: option::t<@ast::expr> = none; while p.peek() != token::RBRACE { @@ -1710,7 +1722,7 @@ fn parse_block_tail(p: &parser, lo: uint) -> ast::blk { } let hi = p.get_hi_pos(); p.bump(); - let bloc = {stmts: stmts, expr: expr, id: p.get_id()}; + let bloc = {stmts: stmts, expr: expr, id: p.get_id(), rules: s}; ret spanned(lo, hi, bloc); } diff --git a/src/comp/syntax/print/pprust.rs b/src/comp/syntax/print/pprust.rs index 051c114002b..52ee1eaeacc 100644 --- a/src/comp/syntax/print/pprust.rs +++ b/src/comp/syntax/print/pprust.rs @@ -579,6 +579,11 @@ tag embed_type { block_macro; block_block_fn; block_normal; } fn print_possibly_embedded_block(s: &ps, blk: &ast::blk, embedded: embed_type, indented: uint) { + alt blk.node.rules { + ast::unchecked. { word(s.s, "unchecked"); } + _ {} + } + maybe_print_comment(s, blk.span.lo); let ann_node = node_block(s, blk); s.ann.pre(ann_node); |
