2011-06-15 18:19:50 +00:00
|
|
|
|
2011-12-14 00:25:51 +00:00
|
|
|
import core::{vec, str, option, either, result};
|
|
|
|
import std::{io, fs};
|
|
|
|
import option::{some, none};
|
|
|
|
import either::{left, right};
|
2011-09-12 23:13:28 +00:00
|
|
|
import std::map::{hashmap, new_str_hash};
|
2011-07-03 17:39:07 +00:00
|
|
|
import token::can_begin_expr;
|
2011-07-05 09:48:19 +00:00
|
|
|
import codemap::span;
|
|
|
|
import util::interner;
|
2011-09-12 23:13:28 +00:00
|
|
|
import ast::{node_id, spanned};
|
2011-11-17 03:50:50 +00:00
|
|
|
import front::attr;
|
2010-08-18 22:41:13 +00:00
|
|
|
|
2011-12-21 04:12:52 +00:00
|
|
|
tag restriction {
|
|
|
|
UNRESTRICTED;
|
|
|
|
RESTRICT_STMT_EXPR;
|
|
|
|
RESTRICT_NO_CALL_EXPRS;
|
|
|
|
RESTRICT_NO_BAR_OP;
|
|
|
|
}
|
2011-01-24 23:26:10 +00:00
|
|
|
|
2011-06-15 18:19:50 +00:00
|
|
|
tag file_type { CRATE_FILE; SOURCE_FILE; }
|
2011-02-25 01:00:24 +00:00
|
|
|
|
2011-07-27 12:19:39 +00:00
|
|
|
type parse_sess = @{cm: codemap::codemap, mutable next_id: node_id};
|
2011-07-05 09:48:19 +00:00
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn next_node_id(sess: parse_sess) -> node_id {
|
2011-07-27 12:19:39 +00:00
|
|
|
let rv = sess.next_id;
|
2011-07-05 09:48:19 +00:00
|
|
|
sess.next_id += 1;
|
|
|
|
ret rv;
|
|
|
|
}
|
2011-05-15 02:02:30 +00:00
|
|
|
|
2011-06-15 18:19:50 +00:00
|
|
|
type parser =
|
|
|
|
obj {
|
2011-08-19 22:16:48 +00:00
|
|
|
fn peek() -> token::token;
|
|
|
|
fn bump();
|
|
|
|
fn swap(token::token, uint, uint);
|
|
|
|
fn look_ahead(uint) -> token::token;
|
2011-09-12 09:27:30 +00:00
|
|
|
fn fatal(str) -> ! ;
|
2011-09-16 14:57:54 +00:00
|
|
|
fn span_fatal(span, str) -> ! ;
|
2011-09-12 09:27:30 +00:00
|
|
|
fn warn(str);
|
2011-08-19 22:16:48 +00:00
|
|
|
fn restrict(restriction);
|
|
|
|
fn get_restriction() -> restriction;
|
|
|
|
fn get_file_type() -> file_type;
|
|
|
|
fn get_cfg() -> ast::crate_cfg;
|
|
|
|
fn get_span() -> span;
|
|
|
|
fn get_lo_pos() -> uint;
|
|
|
|
fn get_hi_pos() -> uint;
|
|
|
|
fn get_last_lo_pos() -> uint;
|
|
|
|
fn get_last_hi_pos() -> uint;
|
|
|
|
fn get_prec_table() -> @[op_spec];
|
2011-09-02 22:34:58 +00:00
|
|
|
fn get_str(token::str_num) -> str;
|
2011-08-19 22:16:48 +00:00
|
|
|
fn get_reader() -> lexer::reader;
|
|
|
|
fn get_filemap() -> codemap::filemap;
|
2011-09-02 22:34:58 +00:00
|
|
|
fn get_bad_expr_words() -> hashmap<str, ()>;
|
2011-08-19 22:16:48 +00:00
|
|
|
fn get_chpos() -> uint;
|
|
|
|
fn get_byte_pos() -> uint;
|
|
|
|
fn get_id() -> node_id;
|
|
|
|
fn get_sess() -> parse_sess;
|
2010-08-18 22:41:13 +00:00
|
|
|
};
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn new_parser_from_file(sess: parse_sess, cfg: ast::crate_cfg, path: str,
|
2011-08-19 22:16:48 +00:00
|
|
|
chpos: uint, byte_pos: uint, ftype: file_type) ->
|
|
|
|
parser {
|
2011-10-29 04:19:59 +00:00
|
|
|
let src = alt io::read_whole_file_str(path) {
|
|
|
|
result::ok(src) {
|
|
|
|
// FIXME: This copy is unfortunate
|
|
|
|
src
|
|
|
|
}
|
|
|
|
result::err(e) {
|
|
|
|
codemap::emit_error(none, e, sess.cm);
|
|
|
|
fail;
|
|
|
|
}
|
|
|
|
};
|
2011-09-02 22:34:58 +00:00
|
|
|
let filemap = codemap::new_filemap(path, chpos, byte_pos);
|
2011-08-19 22:16:48 +00:00
|
|
|
sess.cm.files += [filemap];
|
2011-09-02 00:27:58 +00:00
|
|
|
let itr = @interner::mk(str::hash, str::eq);
|
2011-07-27 12:19:39 +00:00
|
|
|
let rdr = lexer::new_reader(sess.cm, src, filemap, itr);
|
2011-07-11 23:55:35 +00:00
|
|
|
ret new_parser(sess, cfg, rdr, ftype);
|
|
|
|
}
|
|
|
|
|
2011-12-20 21:38:10 +00:00
|
|
|
fn new_parser_from_source_str(sess: parse_sess, cfg: ast::crate_cfg,
|
|
|
|
name: str, source: str) -> parser {
|
|
|
|
let ftype = SOURCE_FILE;
|
|
|
|
let filemap = codemap::new_filemap(name, 0u, 0u);
|
|
|
|
sess.cm.files += [filemap];
|
|
|
|
let itr = @interner::mk(str::hash, str::eq);
|
|
|
|
let rdr = lexer::new_reader(sess.cm, source, filemap, itr);
|
|
|
|
ret new_parser(sess, cfg, rdr, ftype);
|
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn new_parser(sess: parse_sess, cfg: ast::crate_cfg, rdr: lexer::reader,
|
2011-07-27 12:19:39 +00:00
|
|
|
ftype: file_type) -> parser {
|
|
|
|
obj stdio_parser(sess: parse_sess,
|
|
|
|
cfg: ast::crate_cfg,
|
|
|
|
ftype: file_type,
|
|
|
|
mutable tok: token::token,
|
|
|
|
mutable tok_span: span,
|
|
|
|
mutable last_tok_span: span,
|
2011-08-04 23:20:09 +00:00
|
|
|
mutable buffer: [{tok: token::token, span: span}],
|
2011-07-27 12:19:39 +00:00
|
|
|
mutable restr: restriction,
|
|
|
|
rdr: lexer::reader,
|
2011-08-04 23:20:09 +00:00
|
|
|
precs: @[op_spec],
|
2011-09-02 22:34:58 +00:00
|
|
|
bad_words: hashmap<str, ()>) {
|
2011-06-15 18:19:50 +00:00
|
|
|
fn peek() -> token::token { ret tok; }
|
|
|
|
fn bump() {
|
2011-07-25 15:02:49 +00:00
|
|
|
last_tok_span = tok_span;
|
2011-08-15 23:38:23 +00:00
|
|
|
if vec::len(buffer) == 0u {
|
2011-07-27 12:19:39 +00:00
|
|
|
let next = lexer::next_token(rdr);
|
2011-07-26 12:06:02 +00:00
|
|
|
tok = next.tok;
|
2011-08-22 04:44:41 +00:00
|
|
|
tok_span = ast_util::mk_sp(next.chpos, rdr.get_chpos());
|
2011-07-25 15:02:49 +00:00
|
|
|
} else {
|
2011-08-15 23:38:23 +00:00
|
|
|
let next = vec::pop(buffer);
|
2011-07-26 12:06:02 +00:00
|
|
|
tok = next.tok;
|
|
|
|
tok_span = next.span;
|
2011-07-25 15:02:49 +00:00
|
|
|
}
|
|
|
|
}
|
2011-08-16 16:07:26 +00:00
|
|
|
fn swap(next: token::token, lo: uint, hi: uint) {
|
|
|
|
tok = next;
|
2011-08-22 04:44:41 +00:00
|
|
|
tok_span = ast_util::mk_sp(lo, hi);
|
2011-08-16 16:07:26 +00:00
|
|
|
}
|
2011-07-27 12:19:39 +00:00
|
|
|
fn look_ahead(distance: uint) -> token::token {
|
2011-08-15 23:38:23 +00:00
|
|
|
while vec::len(buffer) < distance {
|
2011-07-27 12:19:39 +00:00
|
|
|
let next = lexer::next_token(rdr);
|
2011-08-22 04:44:41 +00:00
|
|
|
let sp = ast_util::mk_sp(next.chpos, rdr.get_chpos());
|
2011-08-19 22:16:48 +00:00
|
|
|
buffer = [{tok: next.tok, span: sp}] + buffer;
|
2011-07-25 15:02:49 +00:00
|
|
|
}
|
2011-08-19 22:16:48 +00:00
|
|
|
ret buffer[distance - 1u].tok;
|
2011-06-15 18:19:50 +00:00
|
|
|
}
|
2011-09-12 09:27:30 +00:00
|
|
|
fn fatal(m: str) -> ! {
|
2011-09-16 14:57:54 +00:00
|
|
|
self.span_fatal(self.get_span(), m);
|
|
|
|
}
|
|
|
|
fn span_fatal(sp: span, m: str) -> ! {
|
|
|
|
codemap::emit_error(some(sp), m, sess.cm);
|
2011-07-05 09:48:19 +00:00
|
|
|
fail;
|
|
|
|
}
|
2011-09-12 09:27:30 +00:00
|
|
|
fn warn(m: str) {
|
2011-09-02 22:34:58 +00:00
|
|
|
codemap::emit_warning(some(self.get_span()), m, sess.cm);
|
2011-07-05 09:48:19 +00:00
|
|
|
}
|
2011-07-27 12:19:39 +00:00
|
|
|
fn restrict(r: restriction) { restr = r; }
|
2011-06-24 17:04:08 +00:00
|
|
|
fn get_restriction() -> restriction { ret restr; }
|
2011-07-25 15:02:49 +00:00
|
|
|
fn get_span() -> span { ret tok_span; }
|
|
|
|
fn get_lo_pos() -> uint { ret tok_span.lo; }
|
|
|
|
fn get_hi_pos() -> uint { ret tok_span.hi; }
|
|
|
|
fn get_last_lo_pos() -> uint { ret last_tok_span.lo; }
|
|
|
|
fn get_last_hi_pos() -> uint { ret last_tok_span.hi; }
|
2011-06-15 18:19:50 +00:00
|
|
|
fn get_file_type() -> file_type { ret ftype; }
|
2011-06-30 01:36:52 +00:00
|
|
|
fn get_cfg() -> ast::crate_cfg { ret cfg; }
|
2011-08-04 23:20:09 +00:00
|
|
|
fn get_prec_table() -> @[op_spec] { ret precs; }
|
2011-09-02 22:34:58 +00:00
|
|
|
fn get_str(i: token::str_num) -> str {
|
2011-08-27 07:43:22 +00:00
|
|
|
ret interner::get(*rdr.get_interner(), i);
|
2011-06-15 18:19:50 +00:00
|
|
|
}
|
|
|
|
fn get_reader() -> lexer::reader { ret rdr; }
|
|
|
|
fn get_filemap() -> codemap::filemap { ret rdr.get_filemap(); }
|
2011-09-02 22:34:58 +00:00
|
|
|
fn get_bad_expr_words() -> hashmap<str, ()> { ret bad_words; }
|
2011-06-15 18:19:50 +00:00
|
|
|
fn get_chpos() -> uint { ret rdr.get_chpos(); }
|
2011-07-16 06:01:10 +00:00
|
|
|
fn get_byte_pos() -> uint { ret rdr.get_byte_pos(); }
|
2011-07-20 00:52:34 +00:00
|
|
|
fn get_id() -> node_id { ret next_node_id(sess); }
|
2011-07-05 09:48:19 +00:00
|
|
|
fn get_sess() -> parse_sess { ret sess; }
|
2011-02-25 01:00:24 +00:00
|
|
|
}
|
2011-07-27 12:19:39 +00:00
|
|
|
let tok0 = lexer::next_token(rdr);
|
2011-08-22 04:44:41 +00:00
|
|
|
let span0 = ast_util::mk_sp(tok0.chpos, rdr.get_chpos());
|
2011-08-19 22:16:48 +00:00
|
|
|
ret stdio_parser(sess, cfg, ftype, tok0.tok, span0, span0, [],
|
2011-07-27 12:19:39 +00:00
|
|
|
UNRESTRICTED, rdr, prec_table(), bad_expr_word_table());
|
2011-05-13 19:54:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// These are the words that shouldn't be allowed as value identifiers,
|
|
|
|
// because, if used at the start of a line, they will cause the line to be
|
|
|
|
// interpreted as a specific kind of statement, which would be confusing.
|
2011-09-02 22:34:58 +00:00
|
|
|
fn bad_expr_word_table() -> hashmap<str, ()> {
|
2011-07-27 12:19:39 +00:00
|
|
|
let words = new_str_hash();
|
2011-12-14 14:42:35 +00:00
|
|
|
for word in ["mod", "if", "else", "while", "do", "alt", "for", "break",
|
|
|
|
"cont", "ret", "be", "fail", "type", "resource", "check",
|
2012-01-10 17:51:15 +00:00
|
|
|
"assert", "claim", "native", "fn", "pure",
|
2011-12-14 14:42:35 +00:00
|
|
|
"unsafe", "block", "import", "export", "let", "const",
|
2012-01-10 23:29:50 +00:00
|
|
|
"log", "tag", "obj", "copy", "sendfn", "impl", "iface",
|
|
|
|
"enum"] {
|
2011-12-14 14:42:35 +00:00
|
|
|
words.insert(word, ());
|
|
|
|
}
|
|
|
|
words
|
2010-09-01 20:24:14 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn unexpected(p: parser, t: token::token) -> ! {
|
2011-12-19 08:48:28 +00:00
|
|
|
let s: str = "unexpected token: '" + token::to_str(p.get_reader(), t) +
|
|
|
|
"'";
|
2011-06-22 19:55:26 +00:00
|
|
|
p.fatal(s);
|
2010-12-10 01:11:52 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn expect(p: parser, t: token::token) {
|
2011-07-27 12:19:39 +00:00
|
|
|
if p.peek() == t {
|
2010-09-01 20:24:14 +00:00
|
|
|
p.bump();
|
|
|
|
} else {
|
2011-12-19 08:48:28 +00:00
|
|
|
let s: str = "expecting '";
|
2011-08-27 07:43:22 +00:00
|
|
|
s += token::to_str(p.get_reader(), t);
|
2011-12-19 08:48:28 +00:00
|
|
|
s += "' but found '";
|
2011-08-27 07:43:22 +00:00
|
|
|
s += token::to_str(p.get_reader(), p.peek());
|
2011-12-19 08:48:28 +00:00
|
|
|
p.fatal(s + "'");
|
2010-09-01 20:24:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn expect_gt(p: parser) {
|
2011-08-16 16:07:26 +00:00
|
|
|
if p.peek() == token::GT {
|
|
|
|
p.bump();
|
|
|
|
} else if p.peek() == token::BINOP(token::LSR) {
|
|
|
|
p.swap(token::GT, p.get_lo_pos() + 1u, p.get_hi_pos());
|
|
|
|
} else if p.peek() == token::BINOP(token::ASR) {
|
2011-08-19 22:16:48 +00:00
|
|
|
p.swap(token::BINOP(token::LSR), p.get_lo_pos() + 1u, p.get_hi_pos());
|
2011-08-16 16:07:26 +00:00
|
|
|
} else {
|
2011-09-02 22:34:58 +00:00
|
|
|
let s: str = "expecting ";
|
2011-08-27 07:43:22 +00:00
|
|
|
s += token::to_str(p.get_reader(), token::GT);
|
2011-09-02 22:34:58 +00:00
|
|
|
s += ", found ";
|
2011-08-27 07:43:22 +00:00
|
|
|
s += token::to_str(p.get_reader(), p.peek());
|
2011-08-16 16:07:26 +00:00
|
|
|
p.fatal(s);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-01-05 14:35:37 +00:00
|
|
|
fn spanned<T: copy>(lo: uint, hi: uint, node: T) -> spanned<T> {
|
2011-08-22 04:44:41 +00:00
|
|
|
ret {node: node, span: ast_util::mk_sp(lo, hi)};
|
2010-10-06 01:21:44 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_ident(p: parser) -> ast::ident {
|
2011-07-27 12:19:39 +00:00
|
|
|
alt p.peek() {
|
2011-09-02 22:34:58 +00:00
|
|
|
token::IDENT(i, _) { p.bump(); ret p.get_str(i); }
|
|
|
|
_ { p.fatal("expecting ident"); }
|
2010-09-01 20:24:14 +00:00
|
|
|
}
|
|
|
|
}
|
2011-06-15 18:19:50 +00:00
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_value_ident(p: parser) -> ast::ident {
|
2011-05-13 19:54:32 +00:00
|
|
|
check_bad_word(p);
|
|
|
|
ret parse_ident(p);
|
|
|
|
}
|
2010-09-01 20:24:14 +00:00
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn eat(p: parser, tok: token::token) -> bool {
|
2011-07-27 12:19:39 +00:00
|
|
|
ret if p.peek() == tok { p.bump(); true } else { false };
|
2011-07-27 07:13:00 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn is_word(p: parser, word: str) -> bool {
|
2011-07-27 12:19:39 +00:00
|
|
|
ret alt p.peek() {
|
2011-09-02 00:27:58 +00:00
|
|
|
token::IDENT(sid, false) { str::eq(word, p.get_str(sid)) }
|
2011-07-27 12:19:39 +00:00
|
|
|
_ { false }
|
2011-06-15 18:19:50 +00:00
|
|
|
};
|
2011-05-13 19:30:08 +00:00
|
|
|
}
|
2011-06-15 18:19:50 +00:00
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn eat_word(p: parser, word: str) -> bool {
|
2011-07-27 12:19:39 +00:00
|
|
|
alt p.peek() {
|
|
|
|
token::IDENT(sid, false) {
|
2011-09-02 00:27:58 +00:00
|
|
|
if str::eq(word, p.get_str(sid)) {
|
2011-07-27 12:19:39 +00:00
|
|
|
p.bump();
|
|
|
|
ret true;
|
|
|
|
} else { ret false; }
|
|
|
|
}
|
|
|
|
_ { ret false; }
|
2011-05-13 19:30:08 +00:00
|
|
|
}
|
|
|
|
}
|
2011-06-15 18:19:50 +00:00
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn expect_word(p: parser, word: str) {
|
2011-07-27 12:19:39 +00:00
|
|
|
if !eat_word(p, word) {
|
2011-09-02 22:34:58 +00:00
|
|
|
p.fatal("expecting " + word + ", found " +
|
|
|
|
token::to_str(p.get_reader(), p.peek()));
|
2011-05-13 19:30:08 +00:00
|
|
|
}
|
|
|
|
}
|
2011-06-15 18:19:50 +00:00
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn check_bad_word(p: parser) {
|
2011-07-27 12:19:39 +00:00
|
|
|
alt p.peek() {
|
|
|
|
token::IDENT(sid, false) {
|
|
|
|
let w = p.get_str(sid);
|
2011-08-27 07:43:22 +00:00
|
|
|
if p.get_bad_expr_words().contains_key(w) {
|
2011-09-02 22:34:58 +00:00
|
|
|
p.fatal("found " + w + " in expression position");
|
2011-05-13 19:54:32 +00:00
|
|
|
}
|
2011-07-27 12:19:39 +00:00
|
|
|
}
|
|
|
|
_ { }
|
2011-05-13 19:54:32 +00:00
|
|
|
}
|
|
|
|
}
|
2011-01-11 02:18:16 +00:00
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_ty_fn(proto: ast::proto, p: parser) -> ast::ty_ {
|
2011-12-22 16:49:54 +00:00
|
|
|
fn parse_fn_input_ty(p: parser) -> ast::arg {
|
2011-09-12 09:27:30 +00:00
|
|
|
let mode = parse_arg_mode(p);
|
2011-12-22 16:49:54 +00:00
|
|
|
let name = if is_plain_ident(p) && p.look_ahead(1u) == token::COLON {
|
|
|
|
let name = parse_value_ident(p);
|
2011-07-27 12:19:39 +00:00
|
|
|
p.bump();
|
2011-12-22 16:49:54 +00:00
|
|
|
name
|
|
|
|
} else { "" };
|
|
|
|
ret {mode: mode, ty: parse_ty(p, false), ident: name, id: p.get_id()};
|
2010-11-05 22:23:03 +00:00
|
|
|
}
|
2011-07-27 12:19:39 +00:00
|
|
|
let inputs =
|
2011-11-21 15:36:11 +00:00
|
|
|
parse_seq(token::LPAREN, token::RPAREN, seq_sep(token::COMMA),
|
2011-07-27 12:19:39 +00:00
|
|
|
parse_fn_input_ty, p);
|
2011-07-20 00:52:34 +00:00
|
|
|
// FIXME: there's no syntax for this right now anyway
|
|
|
|
// auto constrs = parse_constrs(~[], p);
|
2011-08-19 22:16:48 +00:00
|
|
|
let constrs: [@ast::constr] = [];
|
2011-11-23 09:56:10 +00:00
|
|
|
let (ret_style, ret_ty) = parse_ret_ty(p);
|
2011-12-30 04:07:55 +00:00
|
|
|
ret ast::ty_fn(proto, {inputs: inputs.node, output: ret_ty,
|
|
|
|
purity: ast::impure_fn, cf: ret_style,
|
|
|
|
constraints: constrs});
|
2011-02-19 01:30:57 +00:00
|
|
|
}
|
|
|
|
|
2011-12-22 07:45:18 +00:00
|
|
|
fn parse_ty_methods(p: parser, allow_tps: bool) -> [ast::ty_method] {
|
|
|
|
parse_seq(token::LBRACE, token::RBRACE, seq_sep_none(), {|p|
|
2011-07-27 12:19:39 +00:00
|
|
|
let flo = p.get_lo_pos();
|
2011-12-30 04:07:55 +00:00
|
|
|
expect_word(p, "fn");
|
2011-07-27 12:19:39 +00:00
|
|
|
let ident = parse_value_ident(p);
|
2011-12-22 07:45:18 +00:00
|
|
|
let tps = allow_tps ? parse_ty_params(p) : [];
|
2011-12-30 04:07:55 +00:00
|
|
|
let f = parse_ty_fn(ast::proto_bare, p), fhi = p.get_last_hi_pos();
|
2011-05-12 15:24:54 +00:00
|
|
|
expect(p, token::SEMI);
|
2011-07-27 12:19:39 +00:00
|
|
|
alt f {
|
2011-12-30 04:07:55 +00:00
|
|
|
ast::ty_fn(_, d) {
|
2011-12-22 07:45:18 +00:00
|
|
|
{ident: ident, decl: d, tps: tps,
|
|
|
|
span: ast_util::mk_sp(flo, fhi)}
|
2011-07-27 12:19:39 +00:00
|
|
|
}
|
2010-12-15 01:42:12 +00:00
|
|
|
}
|
2011-12-22 07:45:18 +00:00
|
|
|
}, p).node
|
2010-12-15 01:42:12 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_mt(p: parser) -> ast::mt {
|
2011-07-27 12:19:39 +00:00
|
|
|
let mut = parse_mutability(p);
|
2011-08-15 19:06:10 +00:00
|
|
|
let t = parse_ty(p, false);
|
2011-07-27 12:19:39 +00:00
|
|
|
ret {ty: t, mut: mut};
|
2011-03-18 00:39:47 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_ty_field(p: parser) -> ast::ty_field {
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_lo_pos();
|
|
|
|
let mut = parse_mutability(p);
|
|
|
|
let id = parse_ident(p);
|
2011-07-25 15:37:58 +00:00
|
|
|
expect(p, token::COLON);
|
2011-08-15 19:06:10 +00:00
|
|
|
let ty = parse_ty(p, false);
|
2011-07-27 12:19:39 +00:00
|
|
|
ret spanned(lo, ty.span.hi, {ident: id, mt: {ty: ty, mut: mut}});
|
2011-07-25 15:37:58 +00:00
|
|
|
}
|
2011-06-15 18:19:50 +00:00
|
|
|
|
2011-06-11 02:12:42 +00:00
|
|
|
// if i is the jth ident in args, return j
|
|
|
|
// otherwise, fail
|
2011-09-12 09:27:30 +00:00
|
|
|
fn ident_index(p: parser, args: [ast::arg], i: ast::ident) -> uint {
|
2011-07-27 12:19:39 +00:00
|
|
|
let j = 0u;
|
2011-08-16 04:54:52 +00:00
|
|
|
for a: ast::arg in args { if a.ident == i { ret j; } j += 1u; }
|
2011-09-02 22:34:58 +00:00
|
|
|
p.fatal("Unbound variable " + i + " in constraint arg");
|
2011-06-11 02:12:42 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_type_constr_arg(p: parser) -> @ast::ty_constr_arg {
|
2011-07-27 12:19:39 +00:00
|
|
|
let sp = p.get_span();
|
|
|
|
let carg = ast::carg_base;
|
2011-07-20 00:52:34 +00:00
|
|
|
expect(p, token::BINOP(token::STAR));
|
2011-07-27 12:19:39 +00:00
|
|
|
if p.peek() == token::DOT {
|
2011-07-20 00:52:34 +00:00
|
|
|
// "*..." notation for record fields
|
|
|
|
p.bump();
|
2011-11-30 12:38:38 +00:00
|
|
|
let pth = parse_path(p);
|
2011-07-20 00:52:34 +00:00
|
|
|
carg = ast::carg_ident(pth);
|
|
|
|
}
|
|
|
|
// No literals yet, I guess?
|
2011-07-27 12:19:39 +00:00
|
|
|
ret @{node: carg, span: sp};
|
2011-07-20 00:52:34 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_constr_arg(args: [ast::arg], p: parser) -> @ast::constr_arg {
|
2011-07-27 12:19:39 +00:00
|
|
|
let sp = p.get_span();
|
|
|
|
let carg = ast::carg_base;
|
|
|
|
if p.peek() == token::BINOP(token::STAR) {
|
2011-03-04 22:15:19 +00:00
|
|
|
p.bump();
|
|
|
|
} else {
|
2011-07-27 12:19:39 +00:00
|
|
|
let i: ast::ident = parse_value_ident(p);
|
2011-06-11 02:12:42 +00:00
|
|
|
carg = ast::carg_ident(ident_index(p, args, i));
|
2011-03-04 22:15:19 +00:00
|
|
|
}
|
2011-07-27 12:19:39 +00:00
|
|
|
ret @{node: carg, span: sp};
|
2011-03-04 22:15:19 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_ty_constr(fn_args: [ast::arg], p: parser) -> @ast::constr {
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_lo_pos();
|
|
|
|
let path = parse_path(p);
|
2011-08-04 23:20:09 +00:00
|
|
|
let args: {node: [@ast::constr_arg], span: span} =
|
2011-11-21 15:36:11 +00:00
|
|
|
parse_seq(token::LPAREN, token::RPAREN, seq_sep(token::COMMA),
|
2011-10-28 14:41:56 +00:00
|
|
|
{|p| parse_constr_arg(fn_args, p)}, p);
|
2011-06-11 02:12:42 +00:00
|
|
|
ret @spanned(lo, args.span.hi,
|
2011-07-27 12:19:39 +00:00
|
|
|
{path: path, args: args.node, id: p.get_id()});
|
2011-03-04 22:15:19 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_constr_in_type(p: parser) -> @ast::ty_constr {
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_lo_pos();
|
|
|
|
let path = parse_path(p);
|
2011-08-04 23:20:09 +00:00
|
|
|
let args: [@ast::ty_constr_arg] =
|
2011-11-21 15:36:11 +00:00
|
|
|
parse_seq(token::LPAREN, token::RPAREN, seq_sep(token::COMMA),
|
2011-07-27 12:19:39 +00:00
|
|
|
parse_type_constr_arg, p).node;
|
|
|
|
let hi = p.get_lo_pos();
|
|
|
|
let tc: ast::ty_constr_ = {path: path, args: args, id: p.get_id()};
|
2011-07-20 00:52:34 +00:00
|
|
|
ret @spanned(lo, hi, tc);
|
2011-03-04 22:15:19 +00:00
|
|
|
}
|
|
|
|
|
2011-07-20 00:52:34 +00:00
|
|
|
|
2012-01-05 14:35:37 +00:00
|
|
|
fn parse_constrs<T: copy>(pser: block(parser) -> @ast::constr_general<T>,
|
2011-11-18 11:39:20 +00:00
|
|
|
p: parser) ->
|
2011-09-12 09:27:30 +00:00
|
|
|
[@ast::constr_general<T>] {
|
2011-08-19 22:16:48 +00:00
|
|
|
let constrs: [@ast::constr_general<T>] = [];
|
2011-07-27 12:19:39 +00:00
|
|
|
while true {
|
|
|
|
let constr = pser(p);
|
2011-08-19 22:16:48 +00:00
|
|
|
constrs += [constr];
|
2011-07-27 12:19:39 +00:00
|
|
|
if p.peek() == token::COMMA { p.bump(); } else { break; }
|
2011-06-15 18:19:50 +00:00
|
|
|
}
|
2011-07-20 00:52:34 +00:00
|
|
|
constrs
|
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_type_constraints(p: parser) -> [@ast::ty_constr] {
|
2011-07-20 00:52:34 +00:00
|
|
|
ret parse_constrs(parse_constr_in_type, p);
|
2011-03-04 22:15:19 +00:00
|
|
|
}
|
|
|
|
|
2012-01-05 10:09:19 +00:00
|
|
|
fn parse_ty_postfix(orig_t: ast::ty_, p: parser, colons_before_params: bool,
|
|
|
|
lo: uint) -> @ast::ty {
|
2011-08-10 14:50:00 +00:00
|
|
|
if colons_before_params && p.peek() == token::MOD_SEP {
|
2011-08-15 19:06:10 +00:00
|
|
|
p.bump();
|
|
|
|
expect(p, token::LT);
|
|
|
|
} else if !colons_before_params && p.peek() == token::LT {
|
|
|
|
p.bump();
|
2012-01-05 10:09:19 +00:00
|
|
|
} else { ret @spanned(lo, p.get_last_hi_pos(), orig_t); }
|
2011-06-16 17:12:56 +00:00
|
|
|
|
2011-08-15 19:06:10 +00:00
|
|
|
// If we're here, we have explicit type parameter instantiation.
|
2011-10-28 14:41:56 +00:00
|
|
|
let seq = parse_seq_to_gt(some(token::COMMA), {|p| parse_ty(p, false)},
|
|
|
|
p);
|
2011-08-15 19:06:10 +00:00
|
|
|
|
|
|
|
alt orig_t {
|
|
|
|
ast::ty_path(pth, ann) {
|
2012-01-05 10:09:19 +00:00
|
|
|
ret @spanned(lo, p.get_last_hi_pos(),
|
|
|
|
ast::ty_path(@spanned(lo, p.get_last_hi_pos(),
|
2011-11-30 12:38:38 +00:00
|
|
|
{global: pth.node.global,
|
|
|
|
idents: pth.node.idents,
|
|
|
|
types: seq}), ann));
|
2011-08-15 19:06:10 +00:00
|
|
|
}
|
2011-09-02 22:34:58 +00:00
|
|
|
_ { p.fatal("type parameter instantiation only allowed for paths"); }
|
2011-06-16 17:12:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-11-23 09:56:10 +00:00
|
|
|
fn parse_ret_ty(p: parser) -> (ast::ret_style, @ast::ty) {
|
2011-09-14 08:46:40 +00:00
|
|
|
ret if eat(p, token::RARROW) {
|
|
|
|
let lo = p.get_lo_pos();
|
|
|
|
if eat(p, token::NOT) {
|
|
|
|
(ast::noreturn, @spanned(lo, p.get_last_hi_pos(), ast::ty_bot))
|
2011-11-23 09:56:10 +00:00
|
|
|
} else { (ast::return_val, parse_ty(p, false)) }
|
2011-09-14 08:38:23 +00:00
|
|
|
} else {
|
2011-09-14 08:46:40 +00:00
|
|
|
let pos = p.get_lo_pos();
|
|
|
|
(ast::return_val, @spanned(pos, pos, ast::ty_nil))
|
2011-05-15 02:02:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_ty(p: parser, colons_before_params: bool) -> @ast::ty {
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_lo_pos();
|
|
|
|
let t: ast::ty_;
|
2011-03-28 18:46:31 +00:00
|
|
|
// FIXME: do something with this
|
2011-02-11 02:58:58 +00:00
|
|
|
|
2011-09-02 22:34:58 +00:00
|
|
|
if eat_word(p, "bool") {
|
2011-06-15 18:19:50 +00:00
|
|
|
t = ast::ty_bool;
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "int") {
|
2011-12-07 20:06:12 +00:00
|
|
|
t = ast::ty_int(ast::ty_i);
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "uint") {
|
2011-12-07 20:06:12 +00:00
|
|
|
t = ast::ty_uint(ast::ty_u);
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "float") {
|
2011-12-07 20:06:12 +00:00
|
|
|
t = ast::ty_float(ast::ty_f);
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "str") {
|
2011-09-02 23:45:00 +00:00
|
|
|
t = ast::ty_str;
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "char") {
|
2011-12-07 20:06:12 +00:00
|
|
|
t = ast::ty_int(ast::ty_char);
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "i8") {
|
2011-12-07 20:06:12 +00:00
|
|
|
t = ast::ty_int(ast::ty_i8);
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "i16") {
|
2011-12-07 20:06:12 +00:00
|
|
|
t = ast::ty_int(ast::ty_i16);
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "i32") {
|
2011-12-07 20:06:12 +00:00
|
|
|
t = ast::ty_int(ast::ty_i32);
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "i64") {
|
2011-12-07 20:06:12 +00:00
|
|
|
t = ast::ty_int(ast::ty_i64);
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "u8") {
|
2011-12-07 20:06:12 +00:00
|
|
|
t = ast::ty_uint(ast::ty_u8);
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "u16") {
|
2011-12-07 20:06:12 +00:00
|
|
|
t = ast::ty_uint(ast::ty_u16);
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "u32") {
|
2011-12-07 20:06:12 +00:00
|
|
|
t = ast::ty_uint(ast::ty_u32);
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "u64") {
|
2011-12-07 20:06:12 +00:00
|
|
|
t = ast::ty_uint(ast::ty_u64);
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "f32") {
|
2011-12-07 20:06:12 +00:00
|
|
|
t = ast::ty_float(ast::ty_f32);
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "f64") {
|
2011-12-07 20:06:12 +00:00
|
|
|
t = ast::ty_float(ast::ty_f64);
|
2011-08-19 22:16:48 +00:00
|
|
|
} else if p.peek() == token::LPAREN {
|
2011-05-13 19:30:08 +00:00
|
|
|
p.bump();
|
2011-08-15 10:18:27 +00:00
|
|
|
if p.peek() == token::RPAREN {
|
|
|
|
p.bump();
|
|
|
|
t = ast::ty_nil;
|
|
|
|
} else {
|
2011-08-19 22:16:48 +00:00
|
|
|
let ts = [parse_ty(p, false)];
|
2011-08-15 10:18:27 +00:00
|
|
|
while p.peek() == token::COMMA {
|
|
|
|
p.bump();
|
2011-08-19 22:16:48 +00:00
|
|
|
ts += [parse_ty(p, false)];
|
2011-08-15 10:18:27 +00:00
|
|
|
}
|
2011-08-15 23:38:23 +00:00
|
|
|
if vec::len(ts) == 1u {
|
2011-08-19 22:16:48 +00:00
|
|
|
t = ts[0].node;
|
|
|
|
} else { t = ast::ty_tup(ts); }
|
2011-07-27 12:19:39 +00:00
|
|
|
expect(p, token::RPAREN);
|
2011-02-19 01:30:57 +00:00
|
|
|
}
|
2011-08-19 22:16:48 +00:00
|
|
|
} else if p.peek() == token::AT {
|
2011-05-13 19:30:08 +00:00
|
|
|
p.bump();
|
2011-11-02 10:42:31 +00:00
|
|
|
t = ast::ty_box(parse_mt(p));
|
2011-09-20 23:07:09 +00:00
|
|
|
} else if p.peek() == token::TILDE {
|
|
|
|
p.bump();
|
2011-11-02 10:42:31 +00:00
|
|
|
t = ast::ty_uniq(parse_mt(p));
|
2011-08-19 22:16:48 +00:00
|
|
|
} else if p.peek() == token::BINOP(token::STAR) {
|
2011-06-03 18:34:19 +00:00
|
|
|
p.bump();
|
2011-11-02 10:42:31 +00:00
|
|
|
t = ast::ty_ptr(parse_mt(p));
|
2011-08-19 22:16:48 +00:00
|
|
|
} else if p.peek() == token::LBRACE {
|
2011-07-27 12:19:39 +00:00
|
|
|
let elems =
|
2011-11-21 15:36:11 +00:00
|
|
|
parse_seq(token::LBRACE, token::RBRACE, seq_sep_opt(token::COMMA),
|
2011-07-27 12:19:39 +00:00
|
|
|
parse_ty_field, p);
|
2011-12-08 12:29:43 +00:00
|
|
|
if vec::len(elems.node) == 0u { unexpected(p, token::RBRACE); }
|
2011-11-02 10:42:31 +00:00
|
|
|
let hi = elems.span.hi;
|
2011-07-25 15:37:58 +00:00
|
|
|
t = ast::ty_rec(elems.node);
|
2011-07-27 12:19:39 +00:00
|
|
|
if p.peek() == token::COLON {
|
2011-07-25 15:37:58 +00:00
|
|
|
p.bump();
|
2011-11-02 10:42:31 +00:00
|
|
|
t = ast::ty_constr(@spanned(lo, hi, t),
|
2011-07-25 15:37:58 +00:00
|
|
|
parse_type_constraints(p));
|
|
|
|
}
|
2011-08-19 22:16:48 +00:00
|
|
|
} else if p.peek() == token::LBRACKET {
|
2011-07-18 15:41:35 +00:00
|
|
|
expect(p, token::LBRACKET);
|
2011-08-18 21:11:06 +00:00
|
|
|
t = ast::ty_vec(parse_mt(p));
|
2011-07-18 15:41:35 +00:00
|
|
|
expect(p, token::RBRACKET);
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "fn") {
|
2011-10-14 21:57:21 +00:00
|
|
|
let proto = parse_fn_ty_proto(p);
|
2011-10-10 19:42:27 +00:00
|
|
|
t = parse_ty_fn(proto, p);
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "block") {
|
2011-08-18 08:01:39 +00:00
|
|
|
t = parse_ty_fn(ast::proto_block, p);
|
2012-01-10 14:49:15 +00:00
|
|
|
} else if eat_word(p, "lambda") {
|
2012-01-10 17:51:15 +00:00
|
|
|
//(breaks prettyprinting!) p.warn("lambda is deprecated, use fn@");
|
2012-01-10 14:49:15 +00:00
|
|
|
t = parse_ty_fn(ast::proto_box, p);
|
2011-12-14 05:43:36 +00:00
|
|
|
} else if eat_word(p, "sendfn") {
|
2012-01-10 17:51:15 +00:00
|
|
|
//(breaks prettyprinting!) p.warn("sendfn is deprecated, use fn~");
|
2012-01-10 14:49:15 +00:00
|
|
|
t = parse_ty_fn(ast::proto_uniq, p);
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "obj") {
|
2011-12-22 07:45:18 +00:00
|
|
|
t = ast::ty_obj(parse_ty_methods(p, false));
|
2011-08-19 22:16:48 +00:00
|
|
|
} else if p.peek() == token::MOD_SEP || is_ident(p.peek()) {
|
2011-07-27 12:19:39 +00:00
|
|
|
let path = parse_path(p);
|
2011-06-19 20:41:21 +00:00
|
|
|
t = ast::ty_path(path, p.get_id());
|
2011-09-02 22:34:58 +00:00
|
|
|
} else { p.fatal("expecting type"); }
|
2012-01-05 10:09:19 +00:00
|
|
|
ret parse_ty_postfix(t, p, colons_before_params, lo);
|
2010-09-21 23:22:32 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_arg_mode(p: parser) -> ast::mode {
|
2011-10-06 10:26:12 +00:00
|
|
|
if eat(p, token::BINOP(token::AND)) { ast::by_mut_ref }
|
|
|
|
else if eat(p, token::BINOP(token::MINUS)) { ast::by_move }
|
|
|
|
else if eat(p, token::ANDAND) { ast::by_ref }
|
2011-11-16 11:32:38 +00:00
|
|
|
else if eat(p, token::BINOP(token::PLUS)) {
|
|
|
|
if eat(p, token::BINOP(token::PLUS)) { ast::by_val }
|
|
|
|
else { ast::by_copy }
|
|
|
|
}
|
2011-10-06 10:26:12 +00:00
|
|
|
else { ast::mode_infer }
|
2011-08-12 19:58:37 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_arg(p: parser) -> ast::arg {
|
|
|
|
let m = parse_arg_mode(p);
|
2011-08-12 19:58:37 +00:00
|
|
|
let i = parse_value_ident(p);
|
|
|
|
expect(p, token::COLON);
|
|
|
|
let t = parse_ty(p, false);
|
|
|
|
ret {mode: m, ty: t, ident: i, id: p.get_id()};
|
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_fn_block_arg(p: parser) -> ast::arg {
|
2011-08-12 19:58:37 +00:00
|
|
|
let m = parse_arg_mode(p);
|
|
|
|
let i = parse_value_ident(p);
|
2011-12-09 08:42:09 +00:00
|
|
|
let t = eat(p, token::COLON) ? parse_ty(p, false) :
|
|
|
|
@spanned(p.get_lo_pos(), p.get_hi_pos(), ast::ty_infer);
|
2011-07-27 12:19:39 +00:00
|
|
|
ret {mode: m, ty: t, ident: i, id: p.get_id()};
|
2010-09-21 23:22:32 +00:00
|
|
|
}
|
|
|
|
|
2012-01-05 14:35:37 +00:00
|
|
|
fn parse_seq_to_before_gt<T: copy>(sep: option::t<token::token>,
|
2011-11-18 11:39:20 +00:00
|
|
|
f: block(parser) -> T,
|
|
|
|
p: parser) -> [T] {
|
2011-08-16 16:07:26 +00:00
|
|
|
let first = true;
|
2011-08-19 22:16:48 +00:00
|
|
|
let v = [];
|
|
|
|
while p.peek() != token::GT && p.peek() != token::BINOP(token::LSR) &&
|
|
|
|
p.peek() != token::BINOP(token::ASR) {
|
2011-08-16 16:07:26 +00:00
|
|
|
alt sep {
|
|
|
|
some(t) { if first { first = false; } else { expect(p, t); } }
|
|
|
|
_ { }
|
|
|
|
}
|
2011-08-19 22:16:48 +00:00
|
|
|
v += [f(p)];
|
2011-08-16 16:07:26 +00:00
|
|
|
}
|
|
|
|
|
2011-08-16 16:03:58 +00:00
|
|
|
ret v;
|
|
|
|
}
|
|
|
|
|
2012-01-05 14:35:37 +00:00
|
|
|
fn parse_seq_to_gt<T: copy>(sep: option::t<token::token>,
|
2011-11-18 11:39:20 +00:00
|
|
|
f: block(parser) -> T, p: parser) -> [T] {
|
2011-08-16 16:03:58 +00:00
|
|
|
let v = parse_seq_to_before_gt(sep, f, p);
|
2011-08-16 16:07:26 +00:00
|
|
|
expect_gt(p);
|
|
|
|
|
|
|
|
ret v;
|
|
|
|
}
|
|
|
|
|
2012-01-05 14:35:37 +00:00
|
|
|
fn parse_seq_lt_gt<T: copy>(sep: option::t<token::token>,
|
2011-11-18 11:39:20 +00:00
|
|
|
f: block(parser) -> T,
|
|
|
|
p: parser) -> spanned<[T]> {
|
2011-08-16 16:03:58 +00:00
|
|
|
let lo = p.get_lo_pos();
|
|
|
|
expect(p, token::LT);
|
2011-08-13 07:09:25 +00:00
|
|
|
let result = parse_seq_to_before_gt::<T>(sep, f, p);
|
2011-08-16 16:03:58 +00:00
|
|
|
let hi = p.get_hi_pos();
|
|
|
|
expect_gt(p);
|
|
|
|
ret spanned(lo, hi, result);
|
|
|
|
}
|
|
|
|
|
2012-01-05 14:35:37 +00:00
|
|
|
fn parse_seq_to_end<T: copy>(ket: token::token, sep: seq_sep,
|
2011-11-18 11:39:20 +00:00
|
|
|
f: block(parser) -> T, p: parser) -> [T] {
|
2011-07-27 12:19:39 +00:00
|
|
|
let val = parse_seq_to_before_end(ket, sep, f, p);
|
2011-07-25 12:04:37 +00:00
|
|
|
p.bump();
|
|
|
|
ret val;
|
|
|
|
}
|
|
|
|
|
2011-11-21 15:36:11 +00:00
|
|
|
type seq_sep = {
|
|
|
|
sep: option::t<token::token>,
|
|
|
|
trailing_opt: bool // is trailing separator optional?
|
|
|
|
};
|
|
|
|
|
|
|
|
fn seq_sep(t: token::token) -> seq_sep {
|
|
|
|
ret {sep: option::some(t), trailing_opt: false};
|
|
|
|
}
|
|
|
|
fn seq_sep_opt(t: token::token) -> seq_sep {
|
|
|
|
ret {sep: option::some(t), trailing_opt: true};
|
|
|
|
}
|
|
|
|
fn seq_sep_none() -> seq_sep {
|
|
|
|
ret {sep: option::none, trailing_opt: false};
|
|
|
|
}
|
|
|
|
|
2012-01-05 14:35:37 +00:00
|
|
|
fn parse_seq_to_before_end<T: copy>(ket: token::token,
|
2011-11-21 15:36:11 +00:00
|
|
|
sep: seq_sep,
|
2011-11-18 11:39:20 +00:00
|
|
|
f: block(parser) -> T, p: parser) -> [T] {
|
2011-07-27 12:19:39 +00:00
|
|
|
let first: bool = true;
|
2011-08-19 22:16:48 +00:00
|
|
|
let v: [T] = [];
|
2011-07-27 12:19:39 +00:00
|
|
|
while p.peek() != ket {
|
2011-11-21 15:36:11 +00:00
|
|
|
alt sep.sep {
|
2011-07-27 12:19:39 +00:00
|
|
|
some(t) { if first { first = false; } else { expect(p, t); } }
|
|
|
|
_ { }
|
2011-07-06 00:57:34 +00:00
|
|
|
}
|
2011-11-21 15:36:11 +00:00
|
|
|
if sep.trailing_opt && p.peek() == ket { break; }
|
2011-08-19 22:16:48 +00:00
|
|
|
v += [f(p)];
|
2011-07-06 00:57:34 +00:00
|
|
|
}
|
|
|
|
ret v;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-01-05 14:35:37 +00:00
|
|
|
fn parse_seq<T: copy>(bra: token::token, ket: token::token,
|
2011-11-21 15:36:11 +00:00
|
|
|
sep: seq_sep, f: block(parser) -> T,
|
2011-11-18 11:39:20 +00:00
|
|
|
p: parser) -> spanned<[T]> {
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_lo_pos();
|
2011-07-06 00:57:34 +00:00
|
|
|
expect(p, bra);
|
2011-08-13 07:09:25 +00:00
|
|
|
let result = parse_seq_to_before_end::<T>(ket, sep, f, p);
|
2011-07-27 12:19:39 +00:00
|
|
|
let hi = p.get_hi_pos();
|
2011-07-25 12:04:37 +00:00
|
|
|
p.bump();
|
2011-07-06 00:57:34 +00:00
|
|
|
ret spanned(lo, hi, result);
|
|
|
|
}
|
|
|
|
|
2011-10-07 14:22:53 +00:00
|
|
|
fn lit_from_token(p: parser, tok: token::token) -> ast::lit_ {
|
|
|
|
alt tok {
|
2011-12-07 20:06:12 +00:00
|
|
|
token::LIT_INT(i, it) { ast::lit_int(i, it) }
|
|
|
|
token::LIT_UINT(u, ut) { ast::lit_uint(u, ut) }
|
|
|
|
token::LIT_FLOAT(s, ft) { ast::lit_float(p.get_str(s), ft) }
|
2011-10-07 14:22:53 +00:00
|
|
|
token::LIT_STR(s) { ast::lit_str(p.get_str(s)) }
|
|
|
|
token::LPAREN. { expect(p, token::RPAREN); ast::lit_nil }
|
|
|
|
_ { unexpected(p, tok); }
|
|
|
|
}
|
|
|
|
}
|
2011-07-06 00:57:34 +00:00
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_lit(p: parser) -> ast::lit {
|
2011-07-27 12:19:39 +00:00
|
|
|
let sp = p.get_span();
|
2011-10-07 14:22:53 +00:00
|
|
|
let lit = if eat_word(p, "true") {
|
|
|
|
ast::lit_bool(true)
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "false") {
|
2011-10-07 14:22:53 +00:00
|
|
|
ast::lit_bool(false)
|
2011-05-13 19:30:08 +00:00
|
|
|
} else {
|
2011-10-07 14:22:53 +00:00
|
|
|
let tok = p.peek();
|
|
|
|
p.bump();
|
|
|
|
lit_from_token(p, tok)
|
|
|
|
};
|
2011-07-27 12:19:39 +00:00
|
|
|
ret {node: lit, span: sp};
|
2010-08-18 22:41:13 +00:00
|
|
|
}
|
2010-08-12 17:29:23 +00:00
|
|
|
|
2011-07-27 12:19:39 +00:00
|
|
|
fn is_ident(t: token::token) -> bool {
|
|
|
|
alt t { token::IDENT(_, _) { ret true; } _ { } }
|
2011-01-14 01:42:28 +00:00
|
|
|
ret false;
|
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn is_plain_ident(p: parser) -> bool {
|
2011-07-27 07:13:00 +00:00
|
|
|
ret alt p.peek() { token::IDENT(_, false) { true } _ { false } };
|
|
|
|
}
|
|
|
|
|
2011-11-30 12:38:38 +00:00
|
|
|
fn parse_path(p: parser) -> @ast::path {
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_lo_pos();
|
2011-12-20 15:33:55 +00:00
|
|
|
let global = eat(p, token::MOD_SEP), ids = [parse_ident(p)];
|
|
|
|
while p.look_ahead(1u) != token::LT && eat(p, token::MOD_SEP) {
|
|
|
|
ids += [parse_ident(p)];
|
2011-01-14 01:42:28 +00:00
|
|
|
}
|
2011-12-20 15:33:55 +00:00
|
|
|
ret @spanned(lo, p.get_last_hi_pos(),
|
|
|
|
{global: global, idents: ids, types: []});
|
2011-07-27 12:19:39 +00:00
|
|
|
}
|
|
|
|
|
2011-12-20 15:33:55 +00:00
|
|
|
fn parse_path_and_ty_param_substs(p: parser, colons: bool) -> @ast::path {
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_lo_pos();
|
|
|
|
let path = parse_path(p);
|
2011-12-20 15:33:55 +00:00
|
|
|
if colons ? eat(p, token::MOD_SEP) : p.peek() == token::LT {
|
|
|
|
let seq = parse_seq_lt_gt(some(token::COMMA),
|
|
|
|
{|p| parse_ty(p, false)}, p);
|
|
|
|
@spanned(lo, seq.span.hi, {types: seq.node with path.node})
|
|
|
|
} else { path }
|
2010-10-05 00:25:52 +00:00
|
|
|
}
|
2010-09-28 01:25:02 +00:00
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_mutability(p: parser) -> ast::mutability {
|
2011-09-02 22:34:58 +00:00
|
|
|
if eat_word(p, "mutable") {
|
2011-11-16 22:41:32 +00:00
|
|
|
ast::mut
|
2011-11-16 21:52:08 +00:00
|
|
|
} else if eat_word(p, "const") {
|
|
|
|
ast::maybe_mut
|
|
|
|
} else {
|
|
|
|
ast::imm
|
2010-10-12 23:30:44 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_field(p: parser, sep: token::token) -> ast::field {
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_lo_pos();
|
|
|
|
let m = parse_mutability(p);
|
|
|
|
let i = parse_ident(p);
|
2011-07-25 15:37:58 +00:00
|
|
|
expect(p, sep);
|
2011-07-27 12:19:39 +00:00
|
|
|
let e = parse_expr(p);
|
|
|
|
ret spanned(lo, e.span.hi, {mut: m, ident: i, expr: e});
|
2010-12-14 23:32:13 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn mk_expr(p: parser, lo: uint, hi: uint, node: ast::expr_) -> @ast::expr {
|
2011-08-22 04:44:41 +00:00
|
|
|
ret @{id: p.get_id(), node: node, span: ast_util::mk_sp(lo, hi)};
|
2011-06-21 20:16:40 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn mk_mac_expr(p: parser, lo: uint, hi: uint, m: ast::mac_) -> @ast::expr {
|
2011-07-27 12:19:39 +00:00
|
|
|
ret @{id: p.get_id(),
|
2011-08-22 04:44:41 +00:00
|
|
|
node: ast::expr_mac({node: m, span: ast_util::mk_sp(lo, hi)}),
|
|
|
|
span: ast_util::mk_sp(lo, hi)};
|
2011-07-08 23:35:09 +00:00
|
|
|
}
|
|
|
|
|
2011-10-21 09:32:26 +00:00
|
|
|
fn is_bar(t: token::token) -> bool {
|
|
|
|
alt t { token::BINOP(token::OR.) | token::OROR. { true } _ { false } }
|
|
|
|
}
|
|
|
|
|
2011-12-21 22:31:31 +00:00
|
|
|
fn mk_lit_u32(p: parser, i: u32) -> @ast::expr {
|
|
|
|
let span = p.get_span();
|
|
|
|
|
|
|
|
let lv_lit = @{node: ast::lit_uint(i as u64, ast::ty_u32),
|
|
|
|
span: span};
|
|
|
|
|
|
|
|
ret @{id: p.get_id(), node: ast::expr_lit(lv_lit), span: span};
|
|
|
|
}
|
|
|
|
|
2012-01-04 06:03:07 +00:00
|
|
|
// We don't allow single-entry tuples in the true AST; that indicates a
|
|
|
|
// parenthesized expression. However, we preserve them temporarily while
|
|
|
|
// parsing because `(while{...})+3` parses differently from `while{...}+3`.
|
|
|
|
//
|
|
|
|
// To reflect the fact that the @ast::expr is not a true expr that should be
|
|
|
|
// part of the AST, we wrap such expressions in the pexpr tag. They
|
|
|
|
// can then be converted to true expressions by a call to `to_expr()`.
|
|
|
|
tag pexpr {
|
|
|
|
pexpr(@ast::expr);
|
|
|
|
}
|
|
|
|
|
|
|
|
fn mk_pexpr(p: parser, lo: uint, hi: uint, node: ast::expr_) -> pexpr {
|
|
|
|
ret pexpr(mk_expr(p, lo, hi, node));
|
|
|
|
}
|
|
|
|
|
|
|
|
fn to_expr(e: pexpr) -> @ast::expr {
|
|
|
|
alt e.node {
|
2012-01-10 01:15:17 +00:00
|
|
|
ast::expr_tup(es) if vec::len(es) == 1u { es[0u] }
|
2012-01-04 06:03:07 +00:00
|
|
|
_ { *e }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn parse_bottom_expr(p: parser) -> pexpr {
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_lo_pos();
|
|
|
|
let hi = p.get_hi_pos();
|
2011-06-15 18:19:50 +00:00
|
|
|
|
2011-08-13 01:26:23 +00:00
|
|
|
let ex: ast::expr_;
|
2011-07-27 12:19:39 +00:00
|
|
|
if p.peek() == token::LPAREN {
|
2011-05-13 19:30:08 +00:00
|
|
|
p.bump();
|
2011-08-15 10:18:27 +00:00
|
|
|
if p.peek() == token::RPAREN {
|
2011-07-27 12:19:39 +00:00
|
|
|
hi = p.get_hi_pos();
|
|
|
|
p.bump();
|
|
|
|
let lit = @spanned(lo, hi, ast::lit_nil);
|
2012-01-04 06:03:07 +00:00
|
|
|
ret mk_pexpr(p, lo, hi, ast::expr_lit(lit));
|
2011-05-13 19:30:08 +00:00
|
|
|
}
|
2011-08-19 22:16:48 +00:00
|
|
|
let es = [parse_expr(p)];
|
|
|
|
while p.peek() == token::COMMA { p.bump(); es += [parse_expr(p)]; }
|
2011-05-13 19:30:08 +00:00
|
|
|
hi = p.get_hi_pos();
|
|
|
|
expect(p, token::RPAREN);
|
2012-01-04 06:03:07 +00:00
|
|
|
|
|
|
|
// Note: we retain the expr_tup() even for simple
|
|
|
|
// parenthesized expressions, but only for a "little while".
|
|
|
|
// This is so that wrappers around parse_bottom_expr()
|
|
|
|
// can tell whether the expression was parenthesized or not,
|
|
|
|
// which affects expr_is_complete().
|
|
|
|
ret mk_pexpr(p, lo, hi, ast::expr_tup(es));
|
2011-08-19 22:16:48 +00:00
|
|
|
} else if p.peek() == token::LBRACE {
|
2011-07-25 15:37:58 +00:00
|
|
|
p.bump();
|
2011-09-02 22:34:58 +00:00
|
|
|
if is_word(p, "mutable") ||
|
2011-07-27 12:19:39 +00:00
|
|
|
is_plain_ident(p) && p.look_ahead(1u) == token::COLON {
|
2011-08-19 22:16:48 +00:00
|
|
|
let fields = [parse_field(p, token::COLON)];
|
2011-07-27 12:19:39 +00:00
|
|
|
let base = none;
|
2011-07-25 15:37:58 +00:00
|
|
|
while p.peek() != token::RBRACE {
|
2011-09-02 22:34:58 +00:00
|
|
|
if eat_word(p, "with") { base = some(parse_expr(p)); break; }
|
2011-07-25 15:37:58 +00:00
|
|
|
expect(p, token::COMMA);
|
2011-11-21 15:36:11 +00:00
|
|
|
if p.peek() == token::RBRACE {
|
|
|
|
// record ends by an optional trailing comma
|
|
|
|
break;
|
|
|
|
}
|
2011-08-19 22:16:48 +00:00
|
|
|
fields += [parse_field(p, token::COLON)];
|
2011-07-25 15:37:58 +00:00
|
|
|
}
|
|
|
|
hi = p.get_hi_pos();
|
|
|
|
expect(p, token::RBRACE);
|
|
|
|
ex = ast::expr_rec(fields, base);
|
2011-10-21 09:32:26 +00:00
|
|
|
} else if is_bar(p.peek()) {
|
2012-01-04 06:03:07 +00:00
|
|
|
ret pexpr(parse_fn_block_expr(p));
|
2011-07-25 15:37:58 +00:00
|
|
|
} else {
|
2011-10-07 22:51:55 +00:00
|
|
|
let blk = parse_block_tail(p, lo, ast::default_blk);
|
2012-01-04 06:03:07 +00:00
|
|
|
ret mk_pexpr(p, blk.span.lo, blk.span.hi, ast::expr_block(blk));
|
2011-07-25 15:37:58 +00:00
|
|
|
}
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "if") {
|
2012-01-04 06:03:07 +00:00
|
|
|
ret pexpr(parse_if_expr(p));
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "for") {
|
2012-01-04 06:03:07 +00:00
|
|
|
ret pexpr(parse_for_expr(p));
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "while") {
|
2012-01-04 06:03:07 +00:00
|
|
|
ret pexpr(parse_while_expr(p));
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "do") {
|
2012-01-04 06:03:07 +00:00
|
|
|
ret pexpr(parse_do_while_expr(p));
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "alt") {
|
2012-01-04 06:03:07 +00:00
|
|
|
ret pexpr(parse_alt_expr(p));
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "fn") {
|
2011-12-14 05:43:36 +00:00
|
|
|
let proto = parse_fn_ty_proto(p);
|
2012-01-04 06:03:07 +00:00
|
|
|
ret pexpr(parse_fn_expr(p, proto));
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "block") {
|
2012-01-04 06:03:07 +00:00
|
|
|
ret pexpr(parse_fn_expr(p, ast::proto_block));
|
2012-01-10 14:49:15 +00:00
|
|
|
} else if eat_word(p, "lambda") {
|
2012-01-10 17:51:15 +00:00
|
|
|
//(breaks prettyprinting!) p.warn("lambda is deprecated, use fn@");
|
2012-01-10 14:49:15 +00:00
|
|
|
ret pexpr(parse_fn_expr(p, ast::proto_box));
|
2011-12-14 05:43:36 +00:00
|
|
|
} else if eat_word(p, "sendfn") {
|
2012-01-10 17:51:15 +00:00
|
|
|
//(breaks prettyprinting!) p.warn("sendfn is deprecated, use fn~");
|
2012-01-10 14:49:15 +00:00
|
|
|
ret pexpr(parse_fn_expr(p, ast::proto_uniq));
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "unchecked") {
|
2012-01-04 06:03:07 +00:00
|
|
|
ret pexpr(parse_block_expr(p, lo, ast::unchecked_blk));
|
2011-10-06 23:42:27 +00:00
|
|
|
} else if eat_word(p, "unsafe") {
|
2012-01-04 06:03:07 +00:00
|
|
|
ret pexpr(parse_block_expr(p, lo, ast::unsafe_blk));
|
2011-08-19 22:16:48 +00:00
|
|
|
} else if p.peek() == token::LBRACKET {
|
2011-05-16 22:25:10 +00:00
|
|
|
p.bump();
|
2011-07-27 12:19:39 +00:00
|
|
|
let mut = parse_mutability(p);
|
|
|
|
let es =
|
2011-11-21 15:36:11 +00:00
|
|
|
parse_seq_to_end(token::RBRACKET, seq_sep(token::COMMA),
|
|
|
|
parse_expr, p);
|
2011-08-18 18:37:19 +00:00
|
|
|
ex = ast::expr_vec(es, mut);
|
2011-08-19 22:16:48 +00:00
|
|
|
} else if p.peek() == token::POUND_LT {
|
2011-06-30 01:07:04 +00:00
|
|
|
p.bump();
|
2011-08-15 19:06:10 +00:00
|
|
|
let ty = parse_ty(p, false);
|
2011-06-30 01:07:04 +00:00
|
|
|
expect(p, token::GT);
|
2011-07-27 12:19:39 +00:00
|
|
|
|
2011-07-08 23:35:09 +00:00
|
|
|
/* hack: early return to take advantage of specialized function */
|
2012-01-04 06:03:07 +00:00
|
|
|
ret pexpr(mk_mac_expr(p, lo, p.get_hi_pos(),
|
|
|
|
ast::mac_embed_type(ty)));
|
2011-08-19 22:16:48 +00:00
|
|
|
} else if p.peek() == token::POUND_LBRACE {
|
2011-06-30 01:07:04 +00:00
|
|
|
p.bump();
|
2011-10-06 23:42:27 +00:00
|
|
|
let blk = ast::mac_embed_block(
|
2011-10-07 22:51:55 +00:00
|
|
|
parse_block_tail(p, lo, ast::default_blk));
|
2012-01-04 06:03:07 +00:00
|
|
|
ret pexpr(mk_mac_expr(p, lo, p.get_hi_pos(), blk));
|
2011-08-19 22:16:48 +00:00
|
|
|
} else if p.peek() == token::ELLIPSIS {
|
2011-07-21 23:47:47 +00:00
|
|
|
p.bump();
|
2012-01-04 06:03:07 +00:00
|
|
|
ret pexpr(mk_mac_expr(p, lo, p.get_hi_pos(), ast::mac_ellipsis));
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "obj") {
|
2011-05-13 18:00:26 +00:00
|
|
|
// Anonymous object
|
2011-05-07 00:08:41 +00:00
|
|
|
|
2011-06-17 00:33:31 +00:00
|
|
|
// Only make people type () if they're actually adding new fields
|
2011-08-12 14:15:18 +00:00
|
|
|
let fields: option::t<[ast::anon_obj_field]> = none;
|
2011-07-27 12:19:39 +00:00
|
|
|
if p.peek() == token::LPAREN {
|
2011-06-15 12:20:04 +00:00
|
|
|
p.bump();
|
2011-06-15 18:19:50 +00:00
|
|
|
fields =
|
2011-11-21 15:36:11 +00:00
|
|
|
some(parse_seq_to_end(token::RPAREN, seq_sep(token::COMMA),
|
2011-07-27 12:19:39 +00:00
|
|
|
parse_anon_obj_field, p));
|
2011-05-07 00:08:41 +00:00
|
|
|
}
|
2011-08-19 22:16:48 +00:00
|
|
|
let meths: [@ast::method] = [];
|
2011-08-12 14:15:18 +00:00
|
|
|
let inner_obj: option::t<@ast::expr> = none;
|
2011-05-11 02:52:22 +00:00
|
|
|
expect(p, token::LBRACE);
|
2011-07-27 12:19:39 +00:00
|
|
|
while p.peek() != token::RBRACE {
|
2011-09-02 22:34:58 +00:00
|
|
|
if eat_word(p, "with") {
|
2011-07-28 21:43:19 +00:00
|
|
|
inner_obj = some(parse_expr(p));
|
2011-12-16 10:37:38 +00:00
|
|
|
} else { meths += [parse_method(p, false)]; }
|
2011-05-07 00:08:41 +00:00
|
|
|
}
|
|
|
|
hi = p.get_hi_pos();
|
2011-05-11 02:52:22 +00:00
|
|
|
expect(p, token::RBRACE);
|
2011-05-13 18:00:26 +00:00
|
|
|
// fields and methods may be *additional* or *overriding* fields
|
2011-07-28 21:43:19 +00:00
|
|
|
// and methods if there's a inner_obj, or they may be the *only*
|
|
|
|
// fields and methods if there's no inner_obj.
|
2011-05-07 00:08:41 +00:00
|
|
|
|
|
|
|
// We don't need to pull ".node" out of fields because it's not a
|
|
|
|
// "spanned".
|
2011-07-29 13:01:06 +00:00
|
|
|
let ob = {fields: fields, methods: meths, inner_obj: inner_obj};
|
2011-07-21 22:08:15 +00:00
|
|
|
ex = ast::expr_anon_obj(ob);
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "bind") {
|
2011-07-27 12:19:39 +00:00
|
|
|
let e = parse_expr_res(p, RESTRICT_NO_CALL_EXPRS);
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_expr_opt(p: parser) -> option::t<@ast::expr> {
|
2011-07-27 12:19:39 +00:00
|
|
|
alt p.peek() {
|
|
|
|
token::UNDERSCORE. { p.bump(); ret none; }
|
|
|
|
_ { ret some(parse_expr(p)); }
|
2011-02-15 01:46:28 +00:00
|
|
|
}
|
|
|
|
}
|
2011-07-27 12:19:39 +00:00
|
|
|
let es =
|
2011-11-21 15:36:11 +00:00
|
|
|
parse_seq(token::LPAREN, token::RPAREN, seq_sep(token::COMMA),
|
2011-07-27 12:19:39 +00:00
|
|
|
parse_expr_opt, p);
|
2011-05-13 19:30:08 +00:00
|
|
|
hi = es.span.hi;
|
2011-06-21 20:16:40 +00:00
|
|
|
ex = ast::expr_bind(e, es.node);
|
2011-08-19 22:16:48 +00:00
|
|
|
} else if p.peek() == token::POUND {
|
2011-07-27 12:19:39 +00:00
|
|
|
let ex_ext = parse_syntax_ext(p);
|
2011-07-08 23:35:09 +00:00
|
|
|
hi = ex_ext.span.hi;
|
2011-06-15 20:27:39 +00:00
|
|
|
ex = ex_ext.node;
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "fail") {
|
2011-07-27 12:19:39 +00:00
|
|
|
if can_begin_expr(p.peek()) {
|
|
|
|
let e = parse_expr(p);
|
2011-07-03 17:39:07 +00:00
|
|
|
hi = e.span.hi;
|
|
|
|
ex = ast::expr_fail(some(e));
|
2011-07-27 12:19:39 +00:00
|
|
|
} else { ex = ast::expr_fail(none); }
|
2011-12-23 01:53:53 +00:00
|
|
|
} else if eat_word(p, "log") {
|
2011-12-22 02:05:08 +00:00
|
|
|
expect(p, token::LPAREN);
|
2011-12-21 22:31:31 +00:00
|
|
|
let lvl = parse_expr(p);
|
2011-12-22 02:05:08 +00:00
|
|
|
expect(p, token::COMMA);
|
2011-12-21 23:43:38 +00:00
|
|
|
let e = parse_expr(p);
|
2011-12-21 22:31:31 +00:00
|
|
|
ex = ast::expr_log(2, lvl, e);
|
2011-12-22 02:05:08 +00:00
|
|
|
hi = p.get_hi_pos();
|
|
|
|
expect(p, token::RPAREN);
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "assert") {
|
2011-07-27 12:19:39 +00:00
|
|
|
let e = parse_expr(p);
|
2011-06-21 20:16:40 +00:00
|
|
|
ex = ast::expr_assert(e);
|
2011-07-08 23:35:09 +00:00
|
|
|
hi = e.span.hi;
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "check") {
|
2011-07-13 22:44:09 +00:00
|
|
|
/* Should be a predicate (pure boolean function) applied to
|
2011-05-13 19:30:08 +00:00
|
|
|
arguments that are all either slot variables or literals.
|
|
|
|
but the typechecker enforces that. */
|
2011-06-15 18:19:50 +00:00
|
|
|
|
2011-07-27 12:19:39 +00:00
|
|
|
let e = parse_expr(p);
|
2011-07-08 23:35:09 +00:00
|
|
|
hi = e.span.hi;
|
2011-10-06 23:42:27 +00:00
|
|
|
ex = ast::expr_check(ast::checked_expr, e);
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "claim") {
|
2011-06-28 23:29:37 +00:00
|
|
|
/* Same rules as check, except that if check-claims
|
|
|
|
is enabled (a command-line flag), then the parser turns
|
|
|
|
claims into check */
|
2011-07-13 22:44:09 +00:00
|
|
|
|
2011-07-27 12:19:39 +00:00
|
|
|
let e = parse_expr(p);
|
2011-07-08 23:35:09 +00:00
|
|
|
hi = e.span.hi;
|
2011-10-06 23:42:27 +00:00
|
|
|
ex = ast::expr_check(ast::claimed_expr, e);
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "ret") {
|
2011-07-27 12:19:39 +00:00
|
|
|
if can_begin_expr(p.peek()) {
|
|
|
|
let e = parse_expr(p);
|
2011-07-13 22:00:59 +00:00
|
|
|
hi = e.span.hi;
|
|
|
|
ex = ast::expr_ret(some(e));
|
2011-07-27 12:19:39 +00:00
|
|
|
} else { ex = ast::expr_ret(none); }
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "break") {
|
2011-06-21 20:16:40 +00:00
|
|
|
ex = ast::expr_break;
|
2011-07-08 23:35:09 +00:00
|
|
|
hi = p.get_hi_pos();
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "cont") {
|
2011-06-21 20:16:40 +00:00
|
|
|
ex = ast::expr_cont;
|
2011-07-08 23:35:09 +00:00
|
|
|
hi = p.get_hi_pos();
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "be") {
|
2011-07-27 12:19:39 +00:00
|
|
|
let e = parse_expr(p);
|
2011-08-19 22:16:48 +00:00
|
|
|
|
2011-05-13 19:30:08 +00:00
|
|
|
// FIXME: Is this the right place for this check?
|
2011-12-22 12:47:30 +00:00
|
|
|
if /*check*/ast_util::is_call_expr(e) {
|
2011-04-08 16:44:20 +00:00
|
|
|
hi = e.span.hi;
|
2011-06-21 20:16:40 +00:00
|
|
|
ex = ast::expr_be(e);
|
2011-09-02 22:34:58 +00:00
|
|
|
} else { p.fatal("Non-call expression in tail call"); }
|
|
|
|
} else if eat_word(p, "copy") {
|
2011-08-15 22:35:40 +00:00
|
|
|
let e = parse_expr(p);
|
|
|
|
ex = ast::expr_copy(e);
|
|
|
|
hi = e.span.hi;
|
2011-08-19 22:16:48 +00:00
|
|
|
} else if p.peek() == token::MOD_SEP ||
|
2011-09-02 22:34:58 +00:00
|
|
|
is_ident(p.peek()) && !is_word(p, "true") &&
|
|
|
|
!is_word(p, "false") {
|
2011-05-13 19:54:32 +00:00
|
|
|
check_bad_word(p);
|
2011-12-20 15:33:55 +00:00
|
|
|
let pth = parse_path_and_ty_param_substs(p, true);
|
2011-05-13 19:30:08 +00:00
|
|
|
hi = pth.span.hi;
|
2011-06-21 20:16:40 +00:00
|
|
|
ex = ast::expr_path(pth);
|
2011-05-13 19:30:08 +00:00
|
|
|
} else {
|
2011-07-27 12:19:39 +00:00
|
|
|
let lit = parse_lit(p);
|
2011-05-13 19:30:08 +00:00
|
|
|
hi = lit.span.hi;
|
2011-06-21 20:16:40 +00:00
|
|
|
ex = ast::expr_lit(@lit);
|
2010-09-28 01:25:02 +00:00
|
|
|
}
|
2012-01-04 06:03:07 +00:00
|
|
|
ret mk_pexpr(p, lo, hi, ex);
|
2010-09-28 01:25:02 +00:00
|
|
|
}
|
|
|
|
|
2011-10-06 23:42:27 +00:00
|
|
|
fn parse_block_expr(p: parser,
|
|
|
|
lo: uint,
|
|
|
|
blk_mode: ast::blk_check_mode) -> @ast::expr {
|
|
|
|
expect(p, token::LBRACE);
|
|
|
|
let blk = parse_block_tail(p, lo, blk_mode);
|
|
|
|
ret mk_expr(p, blk.span.lo, blk.span.hi, ast::expr_block(blk));
|
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_syntax_ext(p: parser) -> @ast::expr {
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_lo_pos();
|
2011-06-15 20:27:39 +00:00
|
|
|
expect(p, token::POUND);
|
2011-06-16 19:19:27 +00:00
|
|
|
ret parse_syntax_ext_naked(p, lo);
|
2011-06-15 20:27:39 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_syntax_ext_naked(p: parser, lo: uint) -> @ast::expr {
|
2011-12-20 15:33:55 +00:00
|
|
|
alt p.peek() {
|
|
|
|
token::IDENT(_, _) {}
|
|
|
|
_ { p.fatal("expected a syntax expander name"); }
|
2011-07-02 04:59:33 +00:00
|
|
|
}
|
2011-12-20 15:33:55 +00:00
|
|
|
let pth = parse_path(p);
|
2011-07-28 00:36:37 +00:00
|
|
|
//temporary for a backwards-compatible cycle:
|
2011-11-21 15:36:11 +00:00
|
|
|
let sep = seq_sep(token::COMMA);
|
2011-08-19 22:16:48 +00:00
|
|
|
let es =
|
|
|
|
if p.peek() == token::LPAREN {
|
2011-11-21 15:36:11 +00:00
|
|
|
parse_seq(token::LPAREN, token::RPAREN, sep, parse_expr, p)
|
2011-08-19 22:16:48 +00:00
|
|
|
} else {
|
2011-11-21 15:36:11 +00:00
|
|
|
parse_seq(token::LBRACKET, token::RBRACKET, sep, parse_expr, p)
|
2011-08-19 22:16:48 +00:00
|
|
|
};
|
2011-07-27 12:19:39 +00:00
|
|
|
let hi = es.span.hi;
|
2011-08-19 22:16:48 +00:00
|
|
|
let e = mk_expr(p, es.span.lo, hi, ast::expr_vec(es.node, ast::imm));
|
2011-07-28 00:36:37 +00:00
|
|
|
ret mk_mac_expr(p, lo, hi, ast::mac_invoc(pth, e, none));
|
2011-02-24 04:48:01 +00:00
|
|
|
}
|
|
|
|
|
2012-01-04 06:03:07 +00:00
|
|
|
fn parse_dot_or_call_expr(p: parser) -> pexpr {
|
2011-09-15 07:39:24 +00:00
|
|
|
let b = parse_bottom_expr(p);
|
2011-12-21 04:12:52 +00:00
|
|
|
parse_dot_or_call_expr_with(p, b)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn permits_call(p: parser) -> bool {
|
|
|
|
ret p.get_restriction() != RESTRICT_NO_CALL_EXPRS;
|
2011-06-14 13:20:04 +00:00
|
|
|
}
|
|
|
|
|
2012-01-04 06:03:07 +00:00
|
|
|
fn parse_dot_or_call_expr_with(p: parser, e0: pexpr) -> pexpr {
|
|
|
|
let e = e0;
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = e.span.lo;
|
|
|
|
let hi = e.span.hi;
|
2011-12-21 04:12:52 +00:00
|
|
|
while !expr_is_complete(p, e) {
|
2011-07-27 12:19:39 +00:00
|
|
|
alt p.peek() {
|
2011-12-21 04:12:52 +00:00
|
|
|
// expr(...)
|
2012-01-10 01:15:17 +00:00
|
|
|
token::LPAREN. if permits_call(p) {
|
2011-12-21 04:12:52 +00:00
|
|
|
let es = parse_seq(token::LPAREN, token::RPAREN,
|
|
|
|
seq_sep(token::COMMA), parse_expr, p);
|
|
|
|
hi = es.span.hi;
|
2012-01-04 06:03:07 +00:00
|
|
|
let nd = ast::expr_call(to_expr(e), es.node, false);
|
|
|
|
e = mk_pexpr(p, lo, hi, nd);
|
2011-12-21 04:12:52 +00:00
|
|
|
}
|
|
|
|
|
2012-01-04 06:03:07 +00:00
|
|
|
// expr {|| ... }
|
2012-01-10 01:15:17 +00:00
|
|
|
token::LBRACE. if is_bar(p.look_ahead(1u)) && permits_call(p) {
|
2011-12-21 04:12:52 +00:00
|
|
|
p.bump();
|
|
|
|
let blk = parse_fn_block_expr(p);
|
|
|
|
alt e.node {
|
|
|
|
ast::expr_call(f, args, false) {
|
2012-01-04 06:03:07 +00:00
|
|
|
e = pexpr(@{node: ast::expr_call(f, args + [blk], true)
|
|
|
|
with *to_expr(e)});
|
2011-12-21 04:12:52 +00:00
|
|
|
}
|
|
|
|
_ {
|
2012-01-04 06:03:07 +00:00
|
|
|
e = mk_pexpr(p, lo, p.get_last_hi_pos(),
|
|
|
|
ast::expr_call(to_expr(e), [blk], true));
|
2011-12-21 04:12:52 +00:00
|
|
|
}
|
2010-12-15 00:59:13 +00:00
|
|
|
}
|
2011-07-27 12:19:39 +00:00
|
|
|
}
|
2011-12-21 04:12:52 +00:00
|
|
|
|
|
|
|
// expr[...]
|
2011-08-19 19:01:01 +00:00
|
|
|
token::LBRACKET. {
|
|
|
|
p.bump();
|
|
|
|
let ix = parse_expr(p);
|
|
|
|
hi = ix.span.hi;
|
|
|
|
expect(p, token::RBRACKET);
|
2012-01-04 06:03:07 +00:00
|
|
|
e = mk_pexpr(p, lo, hi, ast::expr_index(to_expr(e), ix));
|
2011-08-19 19:01:01 +00:00
|
|
|
}
|
2011-12-21 04:12:52 +00:00
|
|
|
|
|
|
|
// expr.f
|
2011-07-27 12:19:39 +00:00
|
|
|
token::DOT. {
|
|
|
|
p.bump();
|
|
|
|
alt p.peek() {
|
|
|
|
token::IDENT(i, _) {
|
|
|
|
hi = p.get_hi_pos();
|
2010-09-28 17:30:34 +00:00
|
|
|
p.bump();
|
2011-12-19 09:21:31 +00:00
|
|
|
let tys = if eat(p, token::MOD_SEP) {
|
|
|
|
expect(p, token::LT);
|
|
|
|
parse_seq_to_gt(some(token::COMMA),
|
|
|
|
{|p| parse_ty(p, false)}, p)
|
|
|
|
} else { [] };
|
2012-01-04 06:03:07 +00:00
|
|
|
e = mk_pexpr(p, lo, hi,
|
|
|
|
ast::expr_field(to_expr(e),
|
|
|
|
p.get_str(i),
|
|
|
|
tys));
|
2011-07-27 12:19:39 +00:00
|
|
|
}
|
|
|
|
t { unexpected(p, t); }
|
2010-09-28 17:30:34 +00:00
|
|
|
}
|
2011-07-27 12:19:39 +00:00
|
|
|
}
|
2011-12-21 04:12:52 +00:00
|
|
|
|
2011-07-27 12:19:39 +00:00
|
|
|
_ { ret e; }
|
2010-09-28 17:30:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
ret e;
|
|
|
|
}
|
2010-09-28 01:25:02 +00:00
|
|
|
|
2012-01-04 06:03:07 +00:00
|
|
|
fn parse_prefix_expr(p: parser) -> pexpr {
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_lo_pos();
|
|
|
|
let hi = p.get_hi_pos();
|
2011-06-15 18:19:50 +00:00
|
|
|
|
2011-08-24 23:45:22 +00:00
|
|
|
let ex;
|
2011-07-27 12:19:39 +00:00
|
|
|
alt p.peek() {
|
|
|
|
token::NOT. {
|
|
|
|
p.bump();
|
2012-01-04 06:03:07 +00:00
|
|
|
let e = to_expr(parse_prefix_expr(p));
|
2011-07-27 12:19:39 +00:00
|
|
|
hi = e.span.hi;
|
|
|
|
ex = ast::expr_unary(ast::not, e);
|
|
|
|
}
|
|
|
|
token::BINOP(b) {
|
|
|
|
alt b {
|
|
|
|
token::MINUS. {
|
2010-09-28 21:01:21 +00:00
|
|
|
p.bump();
|
2012-01-04 06:03:07 +00:00
|
|
|
let e = to_expr(parse_prefix_expr(p));
|
2011-04-08 16:44:20 +00:00
|
|
|
hi = e.span.hi;
|
2011-07-27 12:19:39 +00:00
|
|
|
ex = ast::expr_unary(ast::neg, e);
|
|
|
|
}
|
|
|
|
token::STAR. {
|
2010-09-28 17:30:34 +00:00
|
|
|
p.bump();
|
2012-01-04 06:03:07 +00:00
|
|
|
let e = to_expr(parse_prefix_expr(p));
|
2011-04-08 16:44:20 +00:00
|
|
|
hi = e.span.hi;
|
2011-07-27 12:19:39 +00:00
|
|
|
ex = ast::expr_unary(ast::deref, e);
|
|
|
|
}
|
|
|
|
_ { ret parse_dot_or_call_expr(p); }
|
2010-09-28 17:30:34 +00:00
|
|
|
}
|
2011-07-27 12:19:39 +00:00
|
|
|
}
|
|
|
|
token::AT. {
|
|
|
|
p.bump();
|
|
|
|
let m = parse_mutability(p);
|
2012-01-04 06:03:07 +00:00
|
|
|
let e = to_expr(parse_prefix_expr(p));
|
2011-07-27 12:19:39 +00:00
|
|
|
hi = e.span.hi;
|
|
|
|
ex = ast::expr_unary(ast::box(m), e);
|
|
|
|
}
|
2011-09-21 01:06:47 +00:00
|
|
|
token::TILDE. {
|
|
|
|
p.bump();
|
|
|
|
let m = parse_mutability(p);
|
2012-01-04 06:03:07 +00:00
|
|
|
let e = to_expr(parse_prefix_expr(p));
|
2011-09-21 01:06:47 +00:00
|
|
|
hi = e.span.hi;
|
|
|
|
ex = ast::expr_unary(ast::uniq(m), e);
|
|
|
|
}
|
2011-07-27 12:19:39 +00:00
|
|
|
_ { ret parse_dot_or_call_expr(p); }
|
2010-09-28 17:30:34 +00:00
|
|
|
}
|
2012-01-04 06:03:07 +00:00
|
|
|
ret mk_pexpr(p, lo, hi, ex);
|
2010-09-28 17:30:34 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_ternary(p: parser) -> @ast::expr {
|
2011-07-27 12:19:39 +00:00
|
|
|
let cond_expr = parse_binops(p);
|
|
|
|
if p.peek() == token::QUES {
|
2011-06-23 22:15:50 +00:00
|
|
|
p.bump();
|
2011-07-27 12:19:39 +00:00
|
|
|
let then_expr = parse_expr(p);
|
2011-06-23 22:15:50 +00:00
|
|
|
expect(p, token::COLON);
|
2011-07-27 12:19:39 +00:00
|
|
|
let else_expr = parse_expr(p);
|
2011-06-23 22:15:50 +00:00
|
|
|
ret mk_expr(p, cond_expr.span.lo, else_expr.span.hi,
|
|
|
|
ast::expr_ternary(cond_expr, then_expr, else_expr));
|
2011-07-27 12:19:39 +00:00
|
|
|
} else { ret cond_expr; }
|
2011-06-23 22:15:50 +00:00
|
|
|
}
|
|
|
|
|
2011-07-27 12:19:39 +00:00
|
|
|
type op_spec = {tok: token::token, op: ast::binop, prec: int};
|
2011-03-07 14:29:06 +00:00
|
|
|
|
2011-06-15 18:19:50 +00:00
|
|
|
|
2011-03-04 06:22:43 +00:00
|
|
|
// FIXME make this a const, don't store it in parser state
|
2011-08-04 23:20:09 +00:00
|
|
|
fn prec_table() -> @[op_spec] {
|
2011-08-19 22:16:48 +00:00
|
|
|
ret @[{tok: token::BINOP(token::STAR), op: ast::mul, prec: 11},
|
|
|
|
{tok: token::BINOP(token::SLASH), op: ast::div, prec: 11},
|
|
|
|
{tok: token::BINOP(token::PERCENT), op: ast::rem, prec: 11},
|
|
|
|
{tok: token::BINOP(token::PLUS), op: ast::add, prec: 10},
|
|
|
|
{tok: token::BINOP(token::MINUS), op: ast::sub, prec: 10},
|
|
|
|
{tok: token::BINOP(token::LSL), op: ast::lsl, prec: 9},
|
|
|
|
{tok: token::BINOP(token::LSR), op: ast::lsr, prec: 9},
|
|
|
|
{tok: token::BINOP(token::ASR), op: ast::asr, prec: 9},
|
|
|
|
{tok: token::BINOP(token::AND), op: ast::bitand, prec: 8},
|
|
|
|
{tok: token::BINOP(token::CARET), op: ast::bitxor, prec: 6},
|
|
|
|
{tok: token::BINOP(token::OR), op: ast::bitor, prec: 6},
|
|
|
|
// 'as' sits between here with 5
|
|
|
|
{tok: token::LT, op: ast::lt, prec: 4},
|
|
|
|
{tok: token::LE, op: ast::le, prec: 4},
|
|
|
|
{tok: token::GE, op: ast::ge, prec: 4},
|
|
|
|
{tok: token::GT, op: ast::gt, prec: 4},
|
|
|
|
{tok: token::EQEQ, op: ast::eq, prec: 3},
|
|
|
|
{tok: token::NE, op: ast::ne, prec: 3},
|
|
|
|
{tok: token::ANDAND, op: ast::and, prec: 2},
|
|
|
|
{tok: token::OROR, op: ast::or, prec: 1}];
|
2011-05-12 15:24:54 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_binops(p: parser) -> @ast::expr {
|
2011-03-07 14:29:06 +00:00
|
|
|
ret parse_more_binops(p, parse_prefix_expr(p), 0);
|
2010-09-28 17:30:34 +00:00
|
|
|
}
|
|
|
|
|
2011-07-27 12:19:39 +00:00
|
|
|
const unop_prec: int = 100;
|
2011-06-15 18:19:50 +00:00
|
|
|
|
2011-07-27 12:19:39 +00:00
|
|
|
const as_prec: int = 5;
|
|
|
|
const ternary_prec: int = 0;
|
2011-05-13 19:30:08 +00:00
|
|
|
|
2012-01-04 06:03:07 +00:00
|
|
|
fn parse_more_binops(p: parser, plhs: pexpr, min_prec: int) ->
|
2011-07-27 12:19:39 +00:00
|
|
|
@ast::expr {
|
2012-01-04 06:03:07 +00:00
|
|
|
let lhs = to_expr(plhs);
|
|
|
|
if expr_is_complete(p, plhs) { ret lhs; }
|
2011-07-27 12:19:39 +00:00
|
|
|
let peeked = p.peek();
|
2011-12-02 12:42:51 +00:00
|
|
|
if peeked == token::BINOP(token::OR) &&
|
|
|
|
p.get_restriction() == RESTRICT_NO_BAR_OP { ret lhs; }
|
2011-08-16 04:54:52 +00:00
|
|
|
for cur: op_spec in *p.get_prec_table() {
|
2011-07-27 12:19:39 +00:00
|
|
|
if cur.prec > min_prec && cur.tok == peeked {
|
2011-03-07 14:29:06 +00:00
|
|
|
p.bump();
|
2011-12-02 12:57:12 +00:00
|
|
|
let expr = parse_prefix_expr(p);
|
2011-10-07 14:22:53 +00:00
|
|
|
let rhs = parse_more_binops(p, expr, cur.prec);
|
2012-01-04 06:03:07 +00:00
|
|
|
let bin = mk_pexpr(p, lhs.span.lo, rhs.span.hi,
|
2011-10-07 14:22:53 +00:00
|
|
|
ast::expr_binary(cur.op, lhs, rhs));
|
2011-06-21 20:16:40 +00:00
|
|
|
ret parse_more_binops(p, bin, min_prec);
|
2010-09-28 17:30:34 +00:00
|
|
|
}
|
|
|
|
}
|
2011-09-02 22:34:58 +00:00
|
|
|
if as_prec > min_prec && eat_word(p, "as") {
|
2011-08-15 19:06:10 +00:00
|
|
|
let rhs = parse_ty(p, true);
|
2011-07-27 12:19:39 +00:00
|
|
|
let _as =
|
2012-01-04 06:03:07 +00:00
|
|
|
mk_pexpr(p, lhs.span.lo, rhs.span.hi, ast::expr_cast(lhs, rhs));
|
2011-06-21 20:16:40 +00:00
|
|
|
ret parse_more_binops(p, _as, min_prec);
|
2011-05-13 19:30:08 +00:00
|
|
|
}
|
2011-03-07 14:29:06 +00:00
|
|
|
ret lhs;
|
2010-09-28 01:25:02 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_assign_expr(p: parser) -> @ast::expr {
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_lo_pos();
|
|
|
|
let lhs = parse_ternary(p);
|
|
|
|
alt p.peek() {
|
|
|
|
token::EQ. {
|
|
|
|
p.bump();
|
|
|
|
let rhs = parse_expr(p);
|
|
|
|
ret mk_expr(p, lo, rhs.span.hi, ast::expr_assign(lhs, rhs));
|
|
|
|
}
|
|
|
|
token::BINOPEQ(op) {
|
|
|
|
p.bump();
|
|
|
|
let rhs = parse_expr(p);
|
|
|
|
let aop = ast::add;
|
|
|
|
alt op {
|
|
|
|
token::PLUS. { aop = ast::add; }
|
|
|
|
token::MINUS. { aop = ast::sub; }
|
|
|
|
token::STAR. { aop = ast::mul; }
|
|
|
|
token::SLASH. { aop = ast::div; }
|
|
|
|
token::PERCENT. { aop = ast::rem; }
|
|
|
|
token::CARET. { aop = ast::bitxor; }
|
|
|
|
token::AND. { aop = ast::bitand; }
|
|
|
|
token::OR. { aop = ast::bitor; }
|
|
|
|
token::LSL. { aop = ast::lsl; }
|
|
|
|
token::LSR. { aop = ast::lsr; }
|
|
|
|
token::ASR. { aop = ast::asr; }
|
|
|
|
}
|
|
|
|
ret mk_expr(p, lo, rhs.span.hi, ast::expr_assign_op(aop, lhs, rhs));
|
|
|
|
}
|
|
|
|
token::LARROW. {
|
|
|
|
p.bump();
|
|
|
|
let rhs = parse_expr(p);
|
|
|
|
ret mk_expr(p, lo, rhs.span.hi, ast::expr_move(lhs, rhs));
|
|
|
|
}
|
|
|
|
token::DARROW. {
|
|
|
|
p.bump();
|
|
|
|
let rhs = parse_expr(p);
|
|
|
|
ret mk_expr(p, lo, rhs.span.hi, ast::expr_swap(lhs, rhs));
|
|
|
|
}
|
|
|
|
_ {/* fall through */ }
|
2010-10-19 23:33:11 +00:00
|
|
|
}
|
|
|
|
ret lhs;
|
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_if_expr_1(p: parser) ->
|
2011-07-27 12:19:39 +00:00
|
|
|
{cond: @ast::expr,
|
|
|
|
then: ast::blk,
|
2011-08-12 14:15:18 +00:00
|
|
|
els: option::t<@ast::expr>,
|
2011-07-27 12:19:39 +00:00
|
|
|
lo: uint,
|
|
|
|
hi: uint} {
|
|
|
|
let lo = p.get_last_lo_pos();
|
|
|
|
let cond = parse_expr(p);
|
|
|
|
let thn = parse_block(p);
|
2011-08-12 14:15:18 +00:00
|
|
|
let els: option::t<@ast::expr> = none;
|
2011-07-27 12:19:39 +00:00
|
|
|
let hi = thn.span.hi;
|
2011-09-02 22:34:58 +00:00
|
|
|
if eat_word(p, "else") {
|
2011-07-27 12:19:39 +00:00
|
|
|
let elexpr = parse_else_expr(p);
|
2011-05-13 19:30:08 +00:00
|
|
|
els = some(elexpr);
|
|
|
|
hi = elexpr.span.hi;
|
2010-10-04 22:55:12 +00:00
|
|
|
}
|
2011-07-27 12:19:39 +00:00
|
|
|
ret {cond: cond, then: thn, els: els, lo: lo, hi: hi};
|
2011-06-16 18:56:34 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_if_expr(p: parser) -> @ast::expr {
|
2011-09-02 22:34:58 +00:00
|
|
|
if eat_word(p, "check") {
|
2011-07-27 12:19:39 +00:00
|
|
|
let q = parse_if_expr_1(p);
|
|
|
|
ret mk_expr(p, q.lo, q.hi, ast::expr_if_check(q.cond, q.then, q.els));
|
|
|
|
} else {
|
|
|
|
let q = parse_if_expr_1(p);
|
2011-07-26 12:06:02 +00:00
|
|
|
ret mk_expr(p, q.lo, q.hi, ast::expr_if(q.cond, q.then, q.els));
|
2011-06-16 18:56:34 +00:00
|
|
|
}
|
2010-10-04 22:55:12 +00:00
|
|
|
}
|
|
|
|
|
2011-12-08 19:47:01 +00:00
|
|
|
// Parses:
|
|
|
|
//
|
2011-12-14 05:43:36 +00:00
|
|
|
// CC := [copy ID*; move ID*]
|
2011-12-08 19:47:01 +00:00
|
|
|
//
|
|
|
|
// where any part is optional and trailing ; is permitted.
|
2011-12-18 05:12:30 +00:00
|
|
|
fn parse_capture_clause(p: parser) -> @ast::capture_clause {
|
2011-12-08 19:47:01 +00:00
|
|
|
fn expect_opt_trailing_semi(p: parser) {
|
|
|
|
if !eat(p, token::SEMI) {
|
2011-12-10 00:56:48 +00:00
|
|
|
if p.peek() != token::RBRACKET {
|
2011-12-08 19:47:01 +00:00
|
|
|
p.fatal("expecting ; or ]");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-12-18 05:12:30 +00:00
|
|
|
fn eat_ident_list(p: parser) -> [@ast::capture_item] {
|
2011-12-08 19:47:01 +00:00
|
|
|
let res = [];
|
|
|
|
while true {
|
|
|
|
alt p.peek() {
|
|
|
|
token::IDENT(_, _) {
|
2011-12-18 05:12:30 +00:00
|
|
|
let id = p.get_id();
|
|
|
|
let sp = ast_util::mk_sp(p.get_lo_pos(), p.get_hi_pos());
|
|
|
|
let ident = parse_ident(p);
|
|
|
|
res += [@{id:id, name:ident, span:sp}];
|
2011-12-08 19:47:01 +00:00
|
|
|
if !eat(p, token::COMMA) {
|
|
|
|
ret res;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
_ { ret res; }
|
|
|
|
}
|
|
|
|
}
|
2011-12-10 00:56:48 +00:00
|
|
|
std::util::unreachable();
|
2011-12-08 19:47:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
let copies = [];
|
|
|
|
let moves = [];
|
|
|
|
|
2011-12-10 00:56:48 +00:00
|
|
|
if eat(p, token::LBRACKET) {
|
|
|
|
while !eat(p, token::RBRACKET) {
|
2011-12-14 05:43:36 +00:00
|
|
|
if eat_word(p, "copy") {
|
2011-12-09 16:16:04 +00:00
|
|
|
copies += eat_ident_list(p);
|
|
|
|
expect_opt_trailing_semi(p);
|
|
|
|
} else if eat_word(p, "move") {
|
|
|
|
moves += eat_ident_list(p);
|
|
|
|
expect_opt_trailing_semi(p);
|
|
|
|
} else {
|
|
|
|
let s: str = "expecting send, copy, or move clause";
|
|
|
|
p.fatal(s);
|
|
|
|
}
|
2011-12-08 19:47:01 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-12-18 05:12:30 +00:00
|
|
|
ret @{copies: copies, moves: moves};
|
2011-12-08 19:47:01 +00:00
|
|
|
}
|
|
|
|
|
2011-12-14 05:43:36 +00:00
|
|
|
fn parse_fn_expr(p: parser, proto: ast::proto) -> @ast::expr {
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_last_lo_pos();
|
2011-12-18 05:12:30 +00:00
|
|
|
let capture_clause = parse_capture_clause(p);
|
2011-12-30 04:07:55 +00:00
|
|
|
let decl = parse_fn_decl(p, ast::impure_fn);
|
2011-07-27 12:19:39 +00:00
|
|
|
let body = parse_block(p);
|
2011-12-22 16:49:54 +00:00
|
|
|
ret mk_expr(p, lo, body.span.hi,
|
2011-12-30 04:07:55 +00:00
|
|
|
ast::expr_fn(proto, decl, body, capture_clause));
|
2011-12-14 00:52:33 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_fn_block_expr(p: parser) -> @ast::expr {
|
2011-08-12 19:58:37 +00:00
|
|
|
let lo = p.get_last_lo_pos();
|
|
|
|
let decl = parse_fn_block_decl(p);
|
2011-10-07 22:51:55 +00:00
|
|
|
let body = parse_block_tail(p, lo, ast::default_blk);
|
2011-12-20 19:03:21 +00:00
|
|
|
ret mk_expr(p, lo, body.span.hi, ast::expr_fn_block(decl, body));
|
2011-08-12 19:58:37 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_else_expr(p: parser) -> @ast::expr {
|
2011-09-02 22:34:58 +00:00
|
|
|
if eat_word(p, "if") {
|
2011-05-13 19:30:08 +00:00
|
|
|
ret parse_if_expr(p);
|
|
|
|
} else {
|
2011-07-27 12:19:39 +00:00
|
|
|
let blk = parse_block(p);
|
2011-06-21 20:16:40 +00:00
|
|
|
ret mk_expr(p, blk.span.lo, blk.span.hi, ast::expr_block(blk));
|
2011-01-30 19:15:22 +00:00
|
|
|
}
|
2010-10-04 22:55:12 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_for_expr(p: parser) -> @ast::expr {
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_last_lo_pos();
|
|
|
|
let decl = parse_local(p, false);
|
2011-09-02 22:34:58 +00:00
|
|
|
expect_word(p, "in");
|
2011-07-27 12:19:39 +00:00
|
|
|
let seq = parse_expr(p);
|
2011-09-15 07:39:24 +00:00
|
|
|
let body = parse_block_no_value(p);
|
2011-07-27 12:19:39 +00:00
|
|
|
let hi = body.span.hi;
|
2011-10-21 11:42:26 +00:00
|
|
|
ret mk_expr(p, lo, hi, ast::expr_for(decl, seq, body));
|
2011-01-20 21:11:47 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_while_expr(p: parser) -> @ast::expr {
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_last_lo_pos();
|
|
|
|
let cond = parse_expr(p);
|
2011-09-15 07:39:24 +00:00
|
|
|
let body = parse_block_no_value(p);
|
2011-07-27 12:19:39 +00:00
|
|
|
let hi = body.span.hi;
|
2011-06-21 20:16:40 +00:00
|
|
|
ret mk_expr(p, lo, hi, ast::expr_while(cond, body));
|
2010-11-03 18:05:15 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_do_while_expr(p: parser) -> @ast::expr {
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_last_lo_pos();
|
2011-09-15 07:39:24 +00:00
|
|
|
let body = parse_block_no_value(p);
|
2011-09-02 22:34:58 +00:00
|
|
|
expect_word(p, "while");
|
2011-07-27 12:19:39 +00:00
|
|
|
let cond = parse_expr(p);
|
|
|
|
let hi = cond.span.hi;
|
2011-06-21 20:16:40 +00:00
|
|
|
ret mk_expr(p, lo, hi, ast::expr_do_while(body, cond));
|
2010-11-03 18:05:15 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_alt_expr(p: parser) -> @ast::expr {
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_last_lo_pos();
|
|
|
|
let discriminant = parse_expr(p);
|
2011-05-12 15:24:54 +00:00
|
|
|
expect(p, token::LBRACE);
|
2011-08-19 22:16:48 +00:00
|
|
|
let arms: [ast::arm] = [];
|
2011-07-27 12:19:39 +00:00
|
|
|
while p.peek() != token::RBRACE {
|
|
|
|
let pats = parse_pats(p);
|
2011-08-22 12:38:48 +00:00
|
|
|
let guard = none;
|
2012-01-10 01:17:02 +00:00
|
|
|
if eat_word(p, "if") { guard = some(parse_expr(p)); }
|
2011-07-27 12:19:39 +00:00
|
|
|
let blk = parse_block(p);
|
2011-08-22 12:38:48 +00:00
|
|
|
arms += [{pats: pats, guard: guard, body: blk}];
|
2011-07-27 12:19:39 +00:00
|
|
|
}
|
|
|
|
let hi = p.get_hi_pos();
|
2010-11-25 00:29:44 +00:00
|
|
|
p.bump();
|
2011-06-21 20:16:40 +00:00
|
|
|
ret mk_expr(p, lo, hi, ast::expr_alt(discriminant, arms));
|
2010-11-24 22:42:01 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_expr(p: parser) -> @ast::expr {
|
2011-01-24 23:26:10 +00:00
|
|
|
ret parse_expr_res(p, UNRESTRICTED);
|
2011-01-01 01:28:43 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_expr_res(p: parser, r: restriction) -> @ast::expr {
|
2011-07-27 12:19:39 +00:00
|
|
|
let old = p.get_restriction();
|
2011-01-24 23:26:10 +00:00
|
|
|
p.restrict(r);
|
2011-07-27 12:19:39 +00:00
|
|
|
let e = parse_assign_expr(p);
|
2011-01-01 01:28:43 +00:00
|
|
|
p.restrict(old);
|
|
|
|
ret e;
|
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_initializer(p: parser) -> option::t<ast::initializer> {
|
2011-07-27 12:19:39 +00:00
|
|
|
alt p.peek() {
|
|
|
|
token::EQ. {
|
|
|
|
p.bump();
|
|
|
|
ret some({op: ast::init_assign, expr: parse_expr(p)});
|
|
|
|
}
|
|
|
|
token::LARROW. {
|
|
|
|
p.bump();
|
|
|
|
ret some({op: ast::init_move, expr: parse_expr(p)});
|
|
|
|
}
|
|
|
|
// Now that the the channel is the first argument to receive,
|
|
|
|
// combining it with an initializer doesn't really make sense.
|
|
|
|
// case (token::RECV) {
|
|
|
|
// p.bump();
|
|
|
|
// ret some(rec(op = ast::init_recv,
|
|
|
|
// expr = parse_expr(p)));
|
|
|
|
// }
|
|
|
|
_ {
|
|
|
|
ret none;
|
|
|
|
}
|
2010-10-12 01:20:25 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_pats(p: parser) -> [@ast::pat] {
|
2011-08-19 22:16:48 +00:00
|
|
|
let pats = [];
|
2011-07-27 12:19:39 +00:00
|
|
|
while true {
|
2011-08-19 22:16:48 +00:00
|
|
|
pats += [parse_pat(p)];
|
2011-07-27 12:19:39 +00:00
|
|
|
if p.peek() == token::BINOP(token::OR) { p.bump(); } else { break; }
|
2011-07-08 14:27:55 +00:00
|
|
|
}
|
|
|
|
ret pats;
|
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_pat(p: parser) -> @ast::pat {
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_lo_pos();
|
|
|
|
let hi = p.get_hi_pos();
|
|
|
|
let pat;
|
|
|
|
alt p.peek() {
|
|
|
|
token::UNDERSCORE. { p.bump(); pat = ast::pat_wild; }
|
|
|
|
token::AT. {
|
|
|
|
p.bump();
|
|
|
|
let sub = parse_pat(p);
|
|
|
|
pat = ast::pat_box(sub);
|
|
|
|
hi = sub.span.hi;
|
|
|
|
}
|
2011-09-23 18:15:17 +00:00
|
|
|
token::TILDE. {
|
|
|
|
p.bump();
|
|
|
|
let sub = parse_pat(p);
|
|
|
|
pat = ast::pat_uniq(sub);
|
|
|
|
hi = sub.span.hi;
|
|
|
|
}
|
2011-07-27 12:19:39 +00:00
|
|
|
token::LBRACE. {
|
|
|
|
p.bump();
|
2011-08-19 22:16:48 +00:00
|
|
|
let fields = [];
|
2011-07-27 12:19:39 +00:00
|
|
|
let etc = false;
|
|
|
|
let first = true;
|
|
|
|
while p.peek() != token::RBRACE {
|
|
|
|
if first { first = false; } else { expect(p, token::COMMA); }
|
|
|
|
|
|
|
|
if p.peek() == token::UNDERSCORE {
|
|
|
|
p.bump();
|
|
|
|
if p.peek() != token::RBRACE {
|
2011-09-02 22:34:58 +00:00
|
|
|
p.fatal("expecting }, found " +
|
|
|
|
token::to_str(p.get_reader(), p.peek()));
|
2011-07-11 12:13:20 +00:00
|
|
|
}
|
2011-07-27 12:19:39 +00:00
|
|
|
etc = true;
|
|
|
|
break;
|
|
|
|
}
|
2011-07-11 22:20:33 +00:00
|
|
|
|
2011-07-27 12:19:39 +00:00
|
|
|
let fieldname = parse_ident(p);
|
|
|
|
let subpat;
|
|
|
|
if p.peek() == token::COLON {
|
|
|
|
p.bump();
|
|
|
|
subpat = parse_pat(p);
|
|
|
|
} else {
|
2011-08-26 00:00:12 +00:00
|
|
|
if p.get_bad_expr_words().contains_key(fieldname) {
|
2011-09-02 22:34:58 +00:00
|
|
|
p.fatal("found " + fieldname + " in binding position");
|
2011-07-11 12:13:20 +00:00
|
|
|
}
|
2011-12-08 10:56:16 +00:00
|
|
|
subpat = @{id: p.get_id(),
|
|
|
|
node: ast::pat_bind(fieldname, none),
|
|
|
|
span: ast_util::mk_sp(lo, hi)};
|
2011-07-11 12:13:20 +00:00
|
|
|
}
|
2011-08-19 22:16:48 +00:00
|
|
|
fields += [{ident: fieldname, pat: subpat}];
|
2011-07-11 12:13:20 +00:00
|
|
|
}
|
2011-07-27 12:19:39 +00:00
|
|
|
hi = p.get_hi_pos();
|
|
|
|
p.bump();
|
|
|
|
pat = ast::pat_rec(fields, etc);
|
|
|
|
}
|
2011-08-15 11:15:19 +00:00
|
|
|
token::LPAREN. {
|
|
|
|
p.bump();
|
|
|
|
if p.peek() == token::RPAREN {
|
|
|
|
hi = p.get_hi_pos();
|
|
|
|
p.bump();
|
2011-12-02 12:42:51 +00:00
|
|
|
let lit = @{node: ast::lit_nil, span: ast_util::mk_sp(lo, hi)};
|
|
|
|
let expr = mk_expr(p, lo, hi, ast::expr_lit(lit));
|
|
|
|
pat = ast::pat_lit(expr);
|
2011-08-15 11:15:19 +00:00
|
|
|
} else {
|
2011-08-19 22:16:48 +00:00
|
|
|
let fields = [parse_pat(p)];
|
2011-08-15 11:15:19 +00:00
|
|
|
while p.peek() == token::COMMA {
|
|
|
|
p.bump();
|
2011-08-19 22:16:48 +00:00
|
|
|
fields += [parse_pat(p)];
|
2011-08-15 11:15:19 +00:00
|
|
|
}
|
2011-08-15 23:38:23 +00:00
|
|
|
if vec::len(fields) == 1u { expect(p, token::COMMA); }
|
2011-08-15 11:15:19 +00:00
|
|
|
hi = p.get_hi_pos();
|
|
|
|
expect(p, token::RPAREN);
|
|
|
|
pat = ast::pat_tup(fields);
|
|
|
|
}
|
2011-09-01 05:34:41 +00:00
|
|
|
}
|
2011-07-27 12:19:39 +00:00
|
|
|
tok {
|
2011-09-02 22:34:58 +00:00
|
|
|
if !is_ident(tok) || is_word(p, "true") || is_word(p, "false") {
|
2011-12-02 12:42:51 +00:00
|
|
|
let val = parse_expr_res(p, RESTRICT_NO_BAR_OP);
|
2011-09-28 19:07:33 +00:00
|
|
|
if eat_word(p, "to") {
|
2011-12-02 12:42:51 +00:00
|
|
|
let end = parse_expr_res(p, RESTRICT_NO_BAR_OP);
|
2011-09-28 19:07:33 +00:00
|
|
|
hi = end.span.hi;
|
2011-12-02 12:42:51 +00:00
|
|
|
pat = ast::pat_range(val, end);
|
2011-09-28 19:07:33 +00:00
|
|
|
} else {
|
2011-12-02 12:42:51 +00:00
|
|
|
hi = val.span.hi;
|
|
|
|
pat = ast::pat_lit(val);
|
2011-09-28 19:07:33 +00:00
|
|
|
}
|
2011-08-19 22:16:48 +00:00
|
|
|
} else if is_plain_ident(p) &&
|
|
|
|
alt p.look_ahead(1u) {
|
|
|
|
token::DOT. | token::LPAREN. | token::LBRACKET. {
|
|
|
|
false
|
|
|
|
}
|
|
|
|
_ { true }
|
|
|
|
} {
|
2011-07-27 12:19:39 +00:00
|
|
|
hi = p.get_hi_pos();
|
2011-12-08 10:56:16 +00:00
|
|
|
let name = parse_value_ident(p);
|
|
|
|
let sub = eat(p, token::AT) ? some(parse_pat(p)) : none;
|
|
|
|
pat = ast::pat_bind(name, sub);
|
2011-07-27 12:19:39 +00:00
|
|
|
} else {
|
2011-12-20 15:33:55 +00:00
|
|
|
let tag_path = parse_path_and_ty_param_substs(p, true);
|
2011-07-27 12:19:39 +00:00
|
|
|
hi = tag_path.span.hi;
|
2011-08-04 23:20:09 +00:00
|
|
|
let args: [@ast::pat];
|
2011-07-27 12:19:39 +00:00
|
|
|
alt p.peek() {
|
|
|
|
token::LPAREN. {
|
|
|
|
let a =
|
|
|
|
parse_seq(token::LPAREN, token::RPAREN,
|
2011-11-21 15:36:11 +00:00
|
|
|
seq_sep(token::COMMA), parse_pat, p);
|
2011-07-27 12:19:39 +00:00
|
|
|
args = a.node;
|
|
|
|
hi = a.span.hi;
|
|
|
|
}
|
2011-08-19 22:16:48 +00:00
|
|
|
token::DOT. { args = []; p.bump(); }
|
2011-07-27 12:19:39 +00:00
|
|
|
_ { expect(p, token::LPAREN); fail; }
|
2011-05-13 19:30:08 +00:00
|
|
|
}
|
2011-07-27 12:19:39 +00:00
|
|
|
pat = ast::pat_tag(tag_path, args);
|
2010-11-24 22:42:01 +00:00
|
|
|
}
|
2011-07-27 12:19:39 +00:00
|
|
|
}
|
2010-11-24 22:42:01 +00:00
|
|
|
}
|
2011-08-22 04:44:41 +00:00
|
|
|
ret @{id: p.get_id(), node: pat, span: ast_util::mk_sp(lo, hi)};
|
2010-11-24 22:42:01 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_local(p: parser, allow_init: bool) -> @ast::local {
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_lo_pos();
|
2011-07-28 10:01:45 +00:00
|
|
|
let pat = parse_pat(p);
|
2011-08-10 19:51:50 +00:00
|
|
|
let ty = @spanned(lo, lo, ast::ty_infer);
|
2011-08-15 19:06:10 +00:00
|
|
|
if eat(p, token::COLON) { ty = parse_ty(p, false); }
|
2011-07-27 12:19:39 +00:00
|
|
|
let init = if allow_init { parse_initializer(p) } else { none };
|
2011-07-27 07:13:00 +00:00
|
|
|
ret @spanned(lo, p.get_last_hi_pos(),
|
2011-08-19 22:16:48 +00:00
|
|
|
{ty: ty, pat: pat, init: init, id: p.get_id()});
|
2010-10-19 01:19:16 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_let(p: parser) -> @ast::decl {
|
2011-09-15 09:42:56 +00:00
|
|
|
fn parse_let_style(p: parser) -> ast::let_style {
|
|
|
|
eat(p, token::BINOP(token::AND)) ? ast::let_ref : ast::let_copy
|
|
|
|
}
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_lo_pos();
|
2011-09-15 09:42:56 +00:00
|
|
|
let locals = [(parse_let_style(p), parse_local(p, true))];
|
|
|
|
while eat(p, token::COMMA) {
|
|
|
|
locals += [(parse_let_style(p), parse_local(p, true))];
|
2011-07-26 08:27:26 +00:00
|
|
|
}
|
|
|
|
ret @spanned(lo, p.get_last_hi_pos(), ast::decl_local(locals));
|
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_stmt(p: parser) -> @ast::stmt {
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_lo_pos();
|
2011-09-02 22:34:58 +00:00
|
|
|
if eat_word(p, "let") {
|
2011-07-27 12:19:39 +00:00
|
|
|
let decl = parse_let(p);
|
2011-06-19 20:41:21 +00:00
|
|
|
ret @spanned(lo, decl.span.hi, ast::stmt_decl(decl, p.get_id()));
|
2011-05-13 19:30:08 +00:00
|
|
|
} else {
|
2011-07-27 12:19:39 +00:00
|
|
|
let item_attrs;
|
|
|
|
alt parse_outer_attrs_or_ext(p) {
|
2011-08-19 22:16:48 +00:00
|
|
|
none. { item_attrs = []; }
|
2011-07-27 12:19:39 +00:00
|
|
|
some(left(attrs)) { item_attrs = attrs; }
|
|
|
|
some(right(ext)) {
|
|
|
|
ret @spanned(lo, ext.span.hi, ast::stmt_expr(ext, p.get_id()));
|
|
|
|
}
|
2011-06-15 20:27:39 +00:00
|
|
|
}
|
|
|
|
|
2011-12-21 04:12:52 +00:00
|
|
|
alt parse_item(p, item_attrs) {
|
2011-07-29 13:01:06 +00:00
|
|
|
some(i) {
|
2011-07-27 12:19:39 +00:00
|
|
|
let hi = i.span.hi;
|
|
|
|
let decl = @spanned(lo, hi, ast::decl_item(i));
|
|
|
|
ret @spanned(lo, hi, ast::stmt_decl(decl, p.get_id()));
|
|
|
|
}
|
2011-12-21 04:12:52 +00:00
|
|
|
none() { /* fallthrough */ }
|
2010-10-04 22:55:12 +00:00
|
|
|
}
|
2011-12-21 04:12:52 +00:00
|
|
|
|
|
|
|
// If we have attributes then we should have an item
|
|
|
|
if vec::len(item_attrs) > 0u {
|
|
|
|
ret p.fatal("expected item");
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remainder are line-expr stmts.
|
|
|
|
let e = parse_expr_res(p, RESTRICT_STMT_EXPR);
|
|
|
|
ret @spanned(lo, e.span.hi, ast::stmt_expr(e, p.get_id()));
|
2010-09-21 23:22:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-01-04 06:03:07 +00:00
|
|
|
fn expr_is_complete(p: parser, e: pexpr) -> bool {
|
2011-12-29 20:03:39 +00:00
|
|
|
log(debug, ("expr_is_complete", p.get_restriction(),
|
2012-01-04 06:03:07 +00:00
|
|
|
print::pprust::expr_to_str(*e),
|
|
|
|
expr_requires_semi_to_be_stmt(*e)));
|
2011-12-21 04:12:52 +00:00
|
|
|
ret p.get_restriction() == RESTRICT_STMT_EXPR &&
|
2012-01-04 06:03:07 +00:00
|
|
|
!expr_requires_semi_to_be_stmt(*e);
|
2011-12-21 04:12:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn expr_requires_semi_to_be_stmt(e: @ast::expr) -> bool {
|
2011-09-15 07:39:24 +00:00
|
|
|
alt e.node {
|
2011-12-29 20:03:39 +00:00
|
|
|
ast::expr_if(_, _, _) | ast::expr_if_check(_, _, _)
|
|
|
|
| ast::expr_alt(_, _) | ast::expr_block(_)
|
|
|
|
| ast::expr_do_while(_, _) | ast::expr_while(_, _)
|
|
|
|
| ast::expr_for(_, _, _)
|
|
|
|
| ast::expr_call(_, _, true) {
|
|
|
|
false
|
2011-09-15 07:39:24 +00:00
|
|
|
}
|
|
|
|
_ { true }
|
2011-09-13 13:50:03 +00:00
|
|
|
}
|
2011-09-15 07:39:24 +00:00
|
|
|
}
|
2011-09-13 13:50:03 +00:00
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn stmt_ends_with_semi(stmt: ast::stmt) -> bool {
|
2011-07-27 12:19:39 +00:00
|
|
|
alt stmt.node {
|
|
|
|
ast::stmt_decl(d, _) {
|
|
|
|
ret alt d.node {
|
|
|
|
ast::decl_local(_) { true }
|
|
|
|
ast::decl_item(_) { false }
|
2010-12-31 22:33:49 +00:00
|
|
|
}
|
2011-07-27 12:19:39 +00:00
|
|
|
}
|
|
|
|
ast::stmt_expr(e, _) {
|
2011-12-21 04:12:52 +00:00
|
|
|
ret expr_requires_semi_to_be_stmt(e);
|
2011-07-27 12:19:39 +00:00
|
|
|
}
|
2012-01-04 22:16:41 +00:00
|
|
|
ast::stmt_semi(e, _) {
|
|
|
|
ret false;
|
|
|
|
}
|
2010-11-30 01:11:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_block(p: parser) -> ast::blk {
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_lo_pos();
|
2011-09-02 22:34:58 +00:00
|
|
|
if eat_word(p, "unchecked") {
|
2011-10-07 22:51:55 +00:00
|
|
|
expect(p, token::LBRACE);
|
2011-10-06 23:42:27 +00:00
|
|
|
be parse_block_tail(p, lo, ast::unchecked_blk);
|
|
|
|
} else if eat_word(p, "unsafe") {
|
2011-10-07 22:51:55 +00:00
|
|
|
expect(p, token::LBRACE);
|
2011-10-06 23:42:27 +00:00
|
|
|
be parse_block_tail(p, lo, ast::unsafe_blk);
|
2011-09-02 22:34:58 +00:00
|
|
|
} else {
|
2011-08-26 00:42:38 +00:00
|
|
|
expect(p, token::LBRACE);
|
2011-10-07 22:51:55 +00:00
|
|
|
be parse_block_tail(p, lo, ast::default_blk);
|
2011-08-26 00:42:38 +00:00
|
|
|
}
|
2011-06-30 01:07:04 +00:00
|
|
|
}
|
|
|
|
|
2011-09-15 07:39:24 +00:00
|
|
|
fn parse_block_no_value(p: parser) -> ast::blk {
|
2011-12-29 20:03:39 +00:00
|
|
|
// We parse blocks that cannot have a value the same as any other block;
|
|
|
|
// the type checker will make sure that the tail expression (if any) has
|
|
|
|
// unit type.
|
|
|
|
ret parse_block(p);
|
2011-09-15 07:39:24 +00:00
|
|
|
}
|
|
|
|
|
2011-08-26 00:42:38 +00:00
|
|
|
// Precondition: already parsed the '{' or '#{'
|
|
|
|
// I guess that also means "already parsed the 'impure'" if
|
|
|
|
// necessary, and this should take a qualifier.
|
2011-07-13 22:44:09 +00:00
|
|
|
// some blocks start with "#{"...
|
2011-10-06 23:42:27 +00:00
|
|
|
fn parse_block_tail(p: parser, lo: uint, s: ast::blk_check_mode) -> ast::blk {
|
2011-11-23 19:57:34 +00:00
|
|
|
let view_items = [], stmts = [], expr = none;
|
|
|
|
while is_word(p, "import") { view_items += [parse_view_item(p)]; }
|
2011-07-27 12:19:39 +00:00
|
|
|
while p.peek() != token::RBRACE {
|
|
|
|
alt p.peek() {
|
|
|
|
token::SEMI. {
|
|
|
|
p.bump(); // empty
|
|
|
|
}
|
|
|
|
_ {
|
|
|
|
let stmt = parse_stmt(p);
|
2012-01-04 22:16:41 +00:00
|
|
|
alt stmt.node {
|
|
|
|
ast::stmt_expr(e, stmt_id) { // Expression without semicolon:
|
2011-07-27 12:19:39 +00:00
|
|
|
alt p.peek() {
|
2012-01-04 22:16:41 +00:00
|
|
|
token::SEMI. {
|
|
|
|
p.bump();
|
|
|
|
stmts += [@{node: ast::stmt_semi(e, stmt_id) with *stmt}];
|
|
|
|
}
|
|
|
|
token::RBRACE. {
|
|
|
|
expr = some(e);
|
|
|
|
}
|
2011-07-27 12:19:39 +00:00
|
|
|
t {
|
|
|
|
if stmt_ends_with_semi(*stmt) {
|
2011-12-19 08:48:28 +00:00
|
|
|
p.fatal("expected ';' or '}' after expression but \
|
|
|
|
found '" + token::to_str(p.get_reader(), t) +
|
|
|
|
"'");
|
2010-11-30 01:11:03 +00:00
|
|
|
}
|
2011-08-19 22:16:48 +00:00
|
|
|
stmts += [stmt];
|
2011-07-27 12:19:39 +00:00
|
|
|
}
|
2010-11-30 01:11:03 +00:00
|
|
|
}
|
2011-07-27 12:19:39 +00:00
|
|
|
}
|
2012-01-04 22:16:41 +00:00
|
|
|
|
|
|
|
_ { // All other kinds of statements:
|
2011-08-19 22:16:48 +00:00
|
|
|
stmts += [stmt];
|
2011-07-27 12:19:39 +00:00
|
|
|
|
2011-12-07 21:12:05 +00:00
|
|
|
if stmt_ends_with_semi(*stmt) {
|
2011-07-27 12:19:39 +00:00
|
|
|
expect(p, token::SEMI);
|
|
|
|
}
|
|
|
|
}
|
2010-11-30 01:11:03 +00:00
|
|
|
}
|
2011-07-27 12:19:39 +00:00
|
|
|
}
|
2010-11-30 01:11:03 +00:00
|
|
|
}
|
|
|
|
}
|
2011-07-27 12:19:39 +00:00
|
|
|
let hi = p.get_hi_pos();
|
2011-03-24 15:33:20 +00:00
|
|
|
p.bump();
|
2011-11-23 19:57:34 +00:00
|
|
|
let bloc = {view_items: view_items, stmts: stmts, expr: expr,
|
|
|
|
id: p.get_id(), rules: s};
|
2011-06-14 20:08:30 +00:00
|
|
|
ret spanned(lo, hi, bloc);
|
2010-09-21 23:22:32 +00:00
|
|
|
}
|
|
|
|
|
2011-11-18 11:39:20 +00:00
|
|
|
fn parse_ty_param(p: parser) -> ast::ty_param {
|
2011-12-28 16:50:12 +00:00
|
|
|
let bounds = [];
|
|
|
|
let ident = parse_ident(p);
|
|
|
|
if eat(p, token::COLON) {
|
|
|
|
while p.peek() != token::COMMA && p.peek() != token::GT {
|
|
|
|
if eat_word(p, "send") { bounds += [ast::bound_send]; }
|
|
|
|
else if eat_word(p, "copy") { bounds += [ast::bound_copy]; }
|
|
|
|
else { bounds += [ast::bound_iface(parse_ty(p, false))]; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ret {ident: ident, id: p.get_id(), bounds: @bounds};
|
2011-07-28 17:22:59 +00:00
|
|
|
}
|
2010-11-25 02:01:20 +00:00
|
|
|
|
2011-11-18 11:39:20 +00:00
|
|
|
fn parse_ty_params(p: parser) -> [ast::ty_param] {
|
2011-12-20 15:33:55 +00:00
|
|
|
if eat(p, token::LT) {
|
|
|
|
parse_seq_to_gt(some(token::COMMA), parse_ty_param, p)
|
|
|
|
} else { [] }
|
2010-11-25 01:15:54 +00:00
|
|
|
}
|
|
|
|
|
2011-12-30 04:07:55 +00:00
|
|
|
fn parse_fn_decl(p: parser, purity: ast::purity)
|
2011-12-22 16:49:54 +00:00
|
|
|
-> ast::fn_decl {
|
2011-08-12 14:15:18 +00:00
|
|
|
let inputs: ast::spanned<[ast::arg]> =
|
2011-11-21 15:36:11 +00:00
|
|
|
parse_seq(token::LPAREN, token::RPAREN, seq_sep(token::COMMA),
|
|
|
|
parse_arg, p);
|
2011-07-27 07:13:00 +00:00
|
|
|
// Use the args list to translate each bound variable
|
|
|
|
// mentioned in a constraint to an arg index.
|
|
|
|
// Seems weird to do this in the parser, but I'm not sure how else to.
|
2011-08-19 22:16:48 +00:00
|
|
|
let constrs = [];
|
2011-07-27 12:19:39 +00:00
|
|
|
if p.peek() == token::COLON {
|
2011-07-20 00:52:34 +00:00
|
|
|
p.bump();
|
2011-10-28 14:41:56 +00:00
|
|
|
constrs = parse_constrs({|x| parse_ty_constr(inputs.node, x) }, p);
|
2011-07-20 00:52:34 +00:00
|
|
|
}
|
2011-11-23 09:56:10 +00:00
|
|
|
let (ret_style, ret_ty) = parse_ret_ty(p);
|
2011-12-30 04:07:55 +00:00
|
|
|
ret {inputs: inputs.node,
|
2011-09-14 08:46:40 +00:00
|
|
|
output: ret_ty,
|
|
|
|
purity: purity,
|
|
|
|
cf: ret_style,
|
|
|
|
constraints: constrs};
|
2011-07-27 12:19:39 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_fn_block_decl(p: parser) -> ast::fn_decl {
|
2011-12-09 08:42:09 +00:00
|
|
|
let inputs = eat(p, token::OROR) ? [] :
|
|
|
|
parse_seq(token::BINOP(token::OR), token::BINOP(token::OR),
|
|
|
|
seq_sep(token::COMMA), parse_fn_block_arg, p).node;
|
|
|
|
let output = eat(p, token::RARROW) ? parse_ty(p, false) :
|
|
|
|
@spanned(p.get_lo_pos(), p.get_hi_pos(), ast::ty_infer);
|
2011-12-30 04:07:55 +00:00
|
|
|
ret {inputs: inputs,
|
2011-12-09 08:42:09 +00:00
|
|
|
output: output,
|
2011-08-12 19:58:37 +00:00
|
|
|
purity: ast::impure_fn,
|
2011-09-14 08:38:23 +00:00
|
|
|
cf: ast::return_val,
|
2011-08-19 22:16:48 +00:00
|
|
|
constraints: []};
|
2011-08-12 19:58:37 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_fn_header(p: parser) -> {ident: ast::ident, tps: [ast::ty_param]} {
|
2011-07-27 12:19:39 +00:00
|
|
|
let id = parse_value_ident(p);
|
2011-11-18 11:39:20 +00:00
|
|
|
let ty_params = parse_ty_params(p);
|
2011-07-27 12:19:39 +00:00
|
|
|
ret {ident: id, tps: ty_params};
|
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn mk_item(p: parser, lo: uint, hi: uint, ident: ast::ident, node: ast::item_,
|
|
|
|
attrs: [ast::attribute]) -> @ast::item {
|
2011-07-27 12:19:39 +00:00
|
|
|
ret @{ident: ident,
|
|
|
|
attrs: attrs,
|
|
|
|
id: p.get_id(),
|
|
|
|
node: node,
|
2011-08-22 04:44:41 +00:00
|
|
|
span: ast_util::mk_sp(lo, hi)};
|
2011-07-27 12:19:39 +00:00
|
|
|
}
|
|
|
|
|
2011-12-30 04:07:55 +00:00
|
|
|
fn parse_item_fn(p: parser, purity: ast::purity,
|
2011-12-22 16:49:54 +00:00
|
|
|
attrs: [ast::attribute]) -> @ast::item {
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_last_lo_pos();
|
|
|
|
let t = parse_fn_header(p);
|
2011-12-30 04:07:55 +00:00
|
|
|
let decl = parse_fn_decl(p, purity);
|
2011-12-22 16:49:54 +00:00
|
|
|
let body = parse_block(p);
|
|
|
|
ret mk_item(p, lo, body.span.hi, t.ident,
|
|
|
|
ast::item_fn(decl, t.tps, body), attrs);
|
2011-07-27 12:19:39 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_obj_field(p: parser) -> ast::obj_field {
|
2011-07-27 12:19:39 +00:00
|
|
|
let mut = parse_mutability(p);
|
|
|
|
let ident = parse_value_ident(p);
|
2011-07-27 07:13:00 +00:00
|
|
|
expect(p, token::COLON);
|
2011-08-15 19:06:10 +00:00
|
|
|
let ty = parse_ty(p, false);
|
2011-07-27 12:19:39 +00:00
|
|
|
ret {mut: mut, ty: ty, ident: ident, id: p.get_id()};
|
2010-12-14 23:32:13 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_anon_obj_field(p: parser) -> ast::anon_obj_field {
|
2011-07-27 12:19:39 +00:00
|
|
|
let mut = parse_mutability(p);
|
|
|
|
let ident = parse_value_ident(p);
|
2011-07-27 07:13:00 +00:00
|
|
|
expect(p, token::COLON);
|
2011-08-15 19:06:10 +00:00
|
|
|
let ty = parse_ty(p, false);
|
2011-06-17 00:33:31 +00:00
|
|
|
expect(p, token::EQ);
|
2011-07-27 12:19:39 +00:00
|
|
|
let expr = parse_expr(p);
|
|
|
|
ret {mut: mut, ty: ty, expr: expr, ident: ident, id: p.get_id()};
|
2011-06-17 00:33:31 +00:00
|
|
|
}
|
|
|
|
|
2011-12-16 10:37:38 +00:00
|
|
|
fn parse_method(p: parser, allow_tps: bool) -> @ast::method {
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_lo_pos();
|
2011-12-30 04:07:55 +00:00
|
|
|
expect_word(p, "fn");
|
2011-07-27 12:19:39 +00:00
|
|
|
let ident = parse_value_ident(p);
|
2011-12-16 10:37:38 +00:00
|
|
|
let tps = allow_tps ? parse_ty_params(p) : [];
|
2011-12-30 04:07:55 +00:00
|
|
|
let decl = parse_fn_decl(p, ast::impure_fn);
|
2011-12-22 16:49:54 +00:00
|
|
|
let body = parse_block(p);
|
|
|
|
@{ident: ident, tps: tps, decl: decl, body: body,
|
|
|
|
id: p.get_id(), span: ast_util::mk_sp(lo, body.span.hi)}
|
2010-12-14 23:32:13 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_item_obj(p: parser, attrs: [ast::attribute]) -> @ast::item {
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_last_lo_pos();
|
|
|
|
let ident = parse_value_ident(p);
|
2011-11-18 11:39:20 +00:00
|
|
|
let ty_params = parse_ty_params(p);
|
2011-08-12 14:15:18 +00:00
|
|
|
let fields: ast::spanned<[ast::obj_field]> =
|
2011-11-21 15:36:11 +00:00
|
|
|
parse_seq(token::LPAREN, token::RPAREN, seq_sep(token::COMMA),
|
2011-07-27 12:19:39 +00:00
|
|
|
parse_obj_field, p);
|
2011-08-19 22:16:48 +00:00
|
|
|
let meths: [@ast::method] = [];
|
2011-05-12 15:24:54 +00:00
|
|
|
expect(p, token::LBRACE);
|
2011-12-16 10:37:38 +00:00
|
|
|
while p.peek() != token::RBRACE { meths += [parse_method(p, false)]; }
|
2011-07-27 12:19:39 +00:00
|
|
|
let hi = p.get_hi_pos();
|
2011-05-12 15:24:54 +00:00
|
|
|
expect(p, token::RBRACE);
|
2011-07-29 11:36:57 +00:00
|
|
|
let ob: ast::_obj = {fields: fields.node, methods: meths};
|
2011-07-27 12:19:39 +00:00
|
|
|
ret mk_item(p, lo, hi, ident, ast::item_obj(ob, ty_params, p.get_id()),
|
|
|
|
attrs);
|
2010-09-21 23:22:32 +00:00
|
|
|
}
|
|
|
|
|
2011-12-20 15:33:55 +00:00
|
|
|
fn parse_item_iface(p: parser, attrs: [ast::attribute]) -> @ast::item {
|
2011-12-16 09:42:28 +00:00
|
|
|
let lo = p.get_last_lo_pos(), ident = parse_ident(p),
|
2011-12-22 07:45:18 +00:00
|
|
|
tps = parse_ty_params(p), meths = parse_ty_methods(p, true);
|
2011-12-20 15:33:55 +00:00
|
|
|
ret mk_item(p, lo, p.get_last_hi_pos(), ident,
|
|
|
|
ast::item_iface(tps, meths), attrs);
|
|
|
|
}
|
|
|
|
|
2012-01-03 14:29:28 +00:00
|
|
|
// Parses three variants (with the initial params always optional):
|
|
|
|
// impl <T: copy> of to_str for [T] { ... }
|
|
|
|
// impl name<T> of to_str for [T] { ... }
|
|
|
|
// impl name<T> for [T] { ... }
|
2011-12-20 15:33:55 +00:00
|
|
|
fn parse_item_impl(p: parser, attrs: [ast::attribute]) -> @ast::item {
|
2012-01-03 14:29:28 +00:00
|
|
|
let lo = p.get_last_lo_pos();
|
2011-12-20 15:33:55 +00:00
|
|
|
fn wrap_path(p: parser, pt: @ast::path) -> @ast::ty {
|
|
|
|
@{node: ast::ty_path(pt, p.get_id()), span: pt.span}
|
|
|
|
}
|
2012-01-03 14:29:28 +00:00
|
|
|
let (ident, tps) = if !is_word(p, "of") {
|
|
|
|
if p.peek() == token::LT { (none, parse_ty_params(p)) }
|
|
|
|
else { (some(parse_ident(p)), parse_ty_params(p)) }
|
|
|
|
} else { (none, []) };
|
|
|
|
let ifce = if eat_word(p, "of") {
|
2011-12-20 15:33:55 +00:00
|
|
|
let path = parse_path_and_ty_param_substs(p, false);
|
2012-01-03 14:29:28 +00:00
|
|
|
if option::is_none(ident) {
|
|
|
|
ident = some(path.node.idents[vec::len(path.node.idents) - 1u]);
|
|
|
|
}
|
|
|
|
some(wrap_path(p, path))
|
|
|
|
} else { none };
|
|
|
|
let ident = alt ident {
|
|
|
|
some(name) { name }
|
|
|
|
none. { expect_word(p, "of"); fail; }
|
2011-12-20 15:33:55 +00:00
|
|
|
};
|
2011-12-16 09:42:28 +00:00
|
|
|
expect_word(p, "for");
|
|
|
|
let ty = parse_ty(p, false), meths = [];
|
2011-12-13 12:19:56 +00:00
|
|
|
expect(p, token::LBRACE);
|
2011-12-16 10:37:38 +00:00
|
|
|
while !eat(p, token::RBRACE) { meths += [parse_method(p, true)]; }
|
2011-12-16 09:42:28 +00:00
|
|
|
ret mk_item(p, lo, p.get_last_hi_pos(), ident,
|
2011-12-20 15:33:55 +00:00
|
|
|
ast::item_impl(tps, ifce, ty, meths), attrs);
|
2011-12-13 12:19:56 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_item_res(p: parser, attrs: [ast::attribute]) -> @ast::item {
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_last_lo_pos();
|
|
|
|
let ident = parse_value_ident(p);
|
2011-11-18 11:39:20 +00:00
|
|
|
let ty_params = parse_ty_params(p);
|
2011-06-24 16:10:40 +00:00
|
|
|
expect(p, token::LPAREN);
|
2011-07-27 12:19:39 +00:00
|
|
|
let arg_ident = parse_value_ident(p);
|
2011-07-27 07:13:00 +00:00
|
|
|
expect(p, token::COLON);
|
2011-08-15 19:06:10 +00:00
|
|
|
let t = parse_ty(p, false);
|
2011-06-24 16:10:40 +00:00
|
|
|
expect(p, token::RPAREN);
|
2011-09-15 07:39:24 +00:00
|
|
|
let dtor = parse_block_no_value(p);
|
2011-07-27 12:19:39 +00:00
|
|
|
let decl =
|
2011-12-30 04:07:55 +00:00
|
|
|
{inputs:
|
2011-10-06 10:26:12 +00:00
|
|
|
[{mode: ast::by_ref, ty: t, ident: arg_ident,
|
|
|
|
id: p.get_id()}],
|
2011-07-27 12:19:39 +00:00
|
|
|
output: @spanned(lo, lo, ast::ty_nil),
|
|
|
|
purity: ast::impure_fn,
|
2011-09-14 08:38:23 +00:00
|
|
|
cf: ast::return_val,
|
2011-08-19 22:16:48 +00:00
|
|
|
constraints: []};
|
2011-06-24 16:10:40 +00:00
|
|
|
ret mk_item(p, lo, dtor.span.hi, ident,
|
2011-12-22 16:49:54 +00:00
|
|
|
ast::item_res(decl, ty_params, dtor, p.get_id(), p.get_id()),
|
|
|
|
attrs);
|
2011-06-24 16:10:40 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_mod_items(p: parser, term: token::token,
|
|
|
|
first_item_attrs: [ast::attribute]) -> ast::_mod {
|
2011-07-27 12:48:34 +00:00
|
|
|
// Shouldn't be any view items since we've already parsed an item attr
|
|
|
|
let view_items =
|
2011-08-19 22:16:48 +00:00
|
|
|
if vec::len(first_item_attrs) == 0u { parse_view(p) } else { [] };
|
|
|
|
let items: [@ast::item] = [];
|
2011-07-27 12:19:39 +00:00
|
|
|
let initial_attrs = first_item_attrs;
|
|
|
|
while p.peek() != term {
|
|
|
|
let attrs = initial_attrs + parse_outer_attributes(p);
|
2011-08-19 22:16:48 +00:00
|
|
|
initial_attrs = [];
|
2011-07-27 12:19:39 +00:00
|
|
|
alt parse_item(p, attrs) {
|
2011-08-19 22:16:48 +00:00
|
|
|
some(i) { items += [i]; }
|
2011-07-27 12:19:39 +00:00
|
|
|
_ {
|
2011-12-19 08:48:28 +00:00
|
|
|
p.fatal("expected item but found '" +
|
|
|
|
token::to_str(p.get_reader(), p.peek()) + "'");
|
2011-07-27 12:19:39 +00:00
|
|
|
}
|
2011-06-14 12:46:46 +00:00
|
|
|
}
|
2010-10-19 01:19:16 +00:00
|
|
|
}
|
2011-07-27 12:19:39 +00:00
|
|
|
ret {view_items: view_items, items: items};
|
2011-01-11 02:18:16 +00:00
|
|
|
}
|
2010-10-19 01:19:16 +00:00
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_item_const(p: parser, attrs: [ast::attribute]) -> @ast::item {
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_last_lo_pos();
|
|
|
|
let id = parse_value_ident(p);
|
2011-07-27 07:13:00 +00:00
|
|
|
expect(p, token::COLON);
|
2011-08-15 19:06:10 +00:00
|
|
|
let ty = parse_ty(p, false);
|
2011-05-12 15:24:54 +00:00
|
|
|
expect(p, token::EQ);
|
2011-07-27 12:19:39 +00:00
|
|
|
let e = parse_expr(p);
|
|
|
|
let hi = p.get_hi_pos();
|
2011-05-12 15:24:54 +00:00
|
|
|
expect(p, token::SEMI);
|
2011-06-16 09:53:06 +00:00
|
|
|
ret mk_item(p, lo, hi, id, ast::item_const(ty, e), attrs);
|
2010-12-09 22:37:50 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_item_mod(p: parser, attrs: [ast::attribute]) -> @ast::item {
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_last_lo_pos();
|
|
|
|
let id = parse_ident(p);
|
2011-05-12 15:24:54 +00:00
|
|
|
expect(p, token::LBRACE);
|
2011-07-27 12:19:39 +00:00
|
|
|
let inner_attrs = parse_inner_attrs_and_next(p);
|
|
|
|
let first_item_outer_attrs = inner_attrs.next;
|
|
|
|
let m = parse_mod_items(p, token::RBRACE, first_item_outer_attrs);
|
|
|
|
let hi = p.get_hi_pos();
|
2011-05-12 15:24:54 +00:00
|
|
|
expect(p, token::RBRACE);
|
2011-07-26 12:06:02 +00:00
|
|
|
ret mk_item(p, lo, hi, id, ast::item_mod(m), attrs + inner_attrs.inner);
|
2010-09-23 20:15:51 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_item_native_type(p: parser, attrs: [ast::attribute]) ->
|
2011-07-27 12:19:39 +00:00
|
|
|
@ast::native_item {
|
|
|
|
let t = parse_type_decl(p);
|
|
|
|
let hi = p.get_hi_pos();
|
2011-05-12 15:24:54 +00:00
|
|
|
expect(p, token::SEMI);
|
2011-07-27 12:19:39 +00:00
|
|
|
ret @{ident: t.ident,
|
|
|
|
attrs: attrs,
|
|
|
|
node: ast::native_item_ty,
|
|
|
|
id: p.get_id(),
|
2011-08-22 04:44:41 +00:00
|
|
|
span: ast_util::mk_sp(t.lo, hi)};
|
2011-07-27 12:19:39 +00:00
|
|
|
}
|
|
|
|
|
2011-10-11 21:52:38 +00:00
|
|
|
fn parse_item_native_fn(p: parser, attrs: [ast::attribute],
|
|
|
|
purity: ast::purity) -> @ast::native_item {
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_last_lo_pos();
|
|
|
|
let t = parse_fn_header(p);
|
2011-12-30 04:07:55 +00:00
|
|
|
let decl = parse_fn_decl(p, purity);
|
2011-07-27 12:19:39 +00:00
|
|
|
let hi = p.get_hi_pos();
|
2011-05-12 15:24:54 +00:00
|
|
|
expect(p, token::SEMI);
|
2011-07-27 12:19:39 +00:00
|
|
|
ret @{ident: t.ident,
|
|
|
|
attrs: attrs,
|
2011-11-14 13:06:39 +00:00
|
|
|
node: ast::native_item_fn(decl, t.tps),
|
2011-07-27 12:19:39 +00:00
|
|
|
id: p.get_id(),
|
2011-08-22 04:44:41 +00:00
|
|
|
span: ast_util::mk_sp(lo, hi)};
|
2011-02-04 16:10:04 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_native_item(p: parser, attrs: [ast::attribute]) ->
|
2011-07-27 12:19:39 +00:00
|
|
|
@ast::native_item {
|
2011-09-02 22:34:58 +00:00
|
|
|
if eat_word(p, "type") {
|
2011-07-05 19:38:53 +00:00
|
|
|
ret parse_item_native_type(p, attrs);
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "fn") {
|
2011-10-11 21:52:38 +00:00
|
|
|
ret parse_item_native_fn(p, attrs, ast::impure_fn);
|
2011-11-22 18:51:05 +00:00
|
|
|
} else if eat_word(p, "pure") {
|
|
|
|
expect_word(p, "fn");
|
|
|
|
ret parse_item_native_fn(p, attrs, ast::pure_fn);
|
2011-10-11 21:52:38 +00:00
|
|
|
} else if eat_word(p, "unsafe") {
|
|
|
|
expect_word(p, "fn");
|
|
|
|
ret parse_item_native_fn(p, attrs, ast::unsafe_fn);
|
2011-07-29 18:51:18 +00:00
|
|
|
} else { unexpected(p, p.peek()); }
|
2011-02-02 15:43:57 +00:00
|
|
|
}
|
|
|
|
|
2011-11-20 18:15:40 +00:00
|
|
|
fn parse_native_mod_items(p: parser, first_item_attrs: [ast::attribute]) ->
|
2011-08-19 22:16:48 +00:00
|
|
|
ast::native_mod {
|
2011-07-27 12:48:34 +00:00
|
|
|
// Shouldn't be any view items since we've already parsed an item attr
|
|
|
|
let view_items =
|
2011-08-15 23:38:23 +00:00
|
|
|
if vec::len(first_item_attrs) == 0u {
|
2011-07-27 12:19:39 +00:00
|
|
|
parse_native_view(p)
|
2011-08-19 22:16:48 +00:00
|
|
|
} else { [] };
|
|
|
|
let items: [@ast::native_item] = [];
|
2011-07-27 12:19:39 +00:00
|
|
|
let initial_attrs = first_item_attrs;
|
|
|
|
while p.peek() != token::RBRACE {
|
|
|
|
let attrs = initial_attrs + parse_outer_attributes(p);
|
2011-08-19 22:16:48 +00:00
|
|
|
initial_attrs = [];
|
|
|
|
items += [parse_native_item(p, attrs)];
|
2011-07-05 19:38:53 +00:00
|
|
|
}
|
2011-11-20 18:15:40 +00:00
|
|
|
ret {view_items: view_items,
|
2011-07-27 12:19:39 +00:00
|
|
|
items: items};
|
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_item_native_mod(p: parser, attrs: [ast::attribute]) -> @ast::item {
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_last_lo_pos();
|
2011-09-02 22:34:58 +00:00
|
|
|
expect_word(p, "mod");
|
2011-07-27 12:19:39 +00:00
|
|
|
let id = parse_ident(p);
|
2011-05-12 15:24:54 +00:00
|
|
|
expect(p, token::LBRACE);
|
2011-07-27 12:19:39 +00:00
|
|
|
let more_attrs = parse_inner_attrs_and_next(p);
|
|
|
|
let inner_attrs = more_attrs.inner;
|
|
|
|
let first_item_outer_attrs = more_attrs.next;
|
2011-11-20 18:15:40 +00:00
|
|
|
let m = parse_native_mod_items(p, first_item_outer_attrs);
|
2011-07-27 12:19:39 +00:00
|
|
|
let hi = p.get_hi_pos();
|
2011-05-12 15:24:54 +00:00
|
|
|
expect(p, token::RBRACE);
|
2011-07-05 19:38:53 +00:00
|
|
|
ret mk_item(p, lo, hi, id, ast::item_native_mod(m), attrs + inner_attrs);
|
2011-02-01 18:40:04 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_type_decl(p: parser) -> {lo: uint, ident: ast::ident} {
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_last_lo_pos();
|
|
|
|
let id = parse_ident(p);
|
|
|
|
ret {lo: lo, ident: id};
|
2011-02-04 14:46:10 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_item_type(p: parser, attrs: [ast::attribute]) -> @ast::item {
|
2011-07-27 12:19:39 +00:00
|
|
|
let t = parse_type_decl(p);
|
2011-11-18 11:39:20 +00:00
|
|
|
let tps = parse_ty_params(p);
|
2011-05-12 15:24:54 +00:00
|
|
|
expect(p, token::EQ);
|
2011-08-15 19:06:10 +00:00
|
|
|
let ty = parse_ty(p, false);
|
2011-07-27 12:19:39 +00:00
|
|
|
let hi = p.get_hi_pos();
|
2011-05-12 15:24:54 +00:00
|
|
|
expect(p, token::SEMI);
|
2011-07-26 12:06:02 +00:00
|
|
|
ret mk_item(p, t.lo, hi, t.ident, ast::item_ty(ty, tps), attrs);
|
2010-11-20 00:34:47 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_item_tag(p: parser, attrs: [ast::attribute]) -> @ast::item {
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_last_lo_pos();
|
|
|
|
let id = parse_ident(p);
|
2011-11-18 11:39:20 +00:00
|
|
|
let ty_params = parse_ty_params(p);
|
2011-08-19 22:16:48 +00:00
|
|
|
let variants: [ast::variant] = [];
|
2011-07-01 11:03:05 +00:00
|
|
|
// Newtype syntax
|
2011-07-27 12:19:39 +00:00
|
|
|
if p.peek() == token::EQ {
|
2011-08-26 00:00:12 +00:00
|
|
|
if p.get_bad_expr_words().contains_key(id) {
|
2011-09-02 22:34:58 +00:00
|
|
|
p.fatal("found " + id + " in tag constructor position");
|
2011-07-01 11:03:05 +00:00
|
|
|
}
|
|
|
|
p.bump();
|
2011-08-15 19:06:10 +00:00
|
|
|
let ty = parse_ty(p, false);
|
2011-07-01 11:03:05 +00:00
|
|
|
expect(p, token::SEMI);
|
2011-07-27 12:19:39 +00:00
|
|
|
let variant =
|
|
|
|
spanned(ty.span.lo, ty.span.hi,
|
|
|
|
{name: id,
|
2011-08-19 22:16:48 +00:00
|
|
|
args: [{ty: ty, id: p.get_id()}],
|
2012-01-10 21:50:40 +00:00
|
|
|
id: p.get_id(),
|
|
|
|
disr_val: 0,
|
|
|
|
disr_expr: none});
|
2011-07-01 11:03:05 +00:00
|
|
|
ret mk_item(p, lo, ty.span.hi, id,
|
2011-08-19 22:16:48 +00:00
|
|
|
ast::item_tag([variant], ty_params), attrs);
|
2011-07-01 11:03:05 +00:00
|
|
|
}
|
2011-05-12 15:24:54 +00:00
|
|
|
expect(p, token::LBRACE);
|
2012-01-10 21:50:40 +00:00
|
|
|
let all_nullary = true;
|
|
|
|
let have_disr = false;
|
|
|
|
let disr_val = 0;
|
2011-07-27 12:19:39 +00:00
|
|
|
while p.peek() != token::RBRACE {
|
|
|
|
let tok = p.peek();
|
|
|
|
alt tok {
|
|
|
|
token::IDENT(name, _) {
|
|
|
|
check_bad_word(p);
|
|
|
|
let vlo = p.get_lo_pos();
|
|
|
|
p.bump();
|
2011-08-19 22:16:48 +00:00
|
|
|
let args: [ast::variant_arg] = [];
|
2011-07-27 12:19:39 +00:00
|
|
|
let vhi = p.get_hi_pos();
|
2012-01-10 21:50:40 +00:00
|
|
|
let disr_expr = none;
|
2011-07-27 12:19:39 +00:00
|
|
|
alt p.peek() {
|
|
|
|
token::LPAREN. {
|
2012-01-10 21:50:40 +00:00
|
|
|
all_nullary = false;
|
2011-10-28 14:41:56 +00:00
|
|
|
let arg_tys = parse_seq(token::LPAREN, token::RPAREN,
|
2011-11-21 15:36:11 +00:00
|
|
|
seq_sep(token::COMMA),
|
2011-10-28 14:41:56 +00:00
|
|
|
{|p| parse_ty(p, false)}, p);
|
2011-08-16 04:54:52 +00:00
|
|
|
for ty: @ast::ty in arg_tys.node {
|
2011-08-19 22:16:48 +00:00
|
|
|
args += [{ty: ty, id: p.get_id()}];
|
2010-11-24 19:36:35 +00:00
|
|
|
}
|
2011-07-27 12:19:39 +00:00
|
|
|
vhi = arg_tys.span.hi;
|
|
|
|
}
|
2012-01-10 21:50:40 +00:00
|
|
|
token::EQ. {
|
|
|
|
have_disr = true;
|
|
|
|
p.bump();
|
|
|
|
let e = parse_expr(p);
|
|
|
|
// FIXME: eval_const_expr does no error checking, nor do I.
|
|
|
|
// Also, the parser is not the right place to do this; likely
|
|
|
|
// somewhere in the middle end so that constants can be
|
|
|
|
// refereed to, even if they are after the declaration for the
|
|
|
|
// type. Finally, eval_const_expr probably shouldn't exist as
|
|
|
|
// it Graydon puts it: "[I] am a little worried at its
|
|
|
|
// presence since it quasi-duplicates stuff that trans should
|
|
|
|
// probably be doing." (See issue #1417)
|
|
|
|
alt syntax::ast_util::eval_const_expr(e) {
|
|
|
|
syntax::ast_util::const_int(val) {
|
2012-01-11 01:33:26 +00:00
|
|
|
// FIXME: check that value is in range
|
2012-01-10 21:50:40 +00:00
|
|
|
disr_val = val as int;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if option::is_some
|
|
|
|
(vec::find
|
|
|
|
(variants, {|v| v.node.disr_val == disr_val}))
|
|
|
|
{
|
|
|
|
p.fatal("discriminator value " + /* str(disr_val) + */
|
|
|
|
"already exists.");
|
|
|
|
}
|
|
|
|
disr_expr = some(e);
|
|
|
|
}
|
2011-07-27 12:19:39 +00:00
|
|
|
_ {/* empty */ }
|
2010-11-24 19:36:35 +00:00
|
|
|
}
|
2011-07-27 12:19:39 +00:00
|
|
|
expect(p, token::SEMI);
|
|
|
|
p.get_id();
|
2012-01-10 21:50:40 +00:00
|
|
|
let vr = {name: p.get_str(name), args: args, id: p.get_id(),
|
|
|
|
disr_val: disr_val, disr_expr: disr_expr};
|
2011-08-19 22:16:48 +00:00
|
|
|
variants += [spanned(vlo, vhi, vr)];
|
2012-01-10 21:50:40 +00:00
|
|
|
disr_val += 1;
|
2011-07-27 12:19:39 +00:00
|
|
|
}
|
|
|
|
token::RBRACE. {/* empty */ }
|
|
|
|
_ {
|
2011-12-19 08:48:28 +00:00
|
|
|
p.fatal("expected name of variant or '}' but found '" +
|
|
|
|
token::to_str(p.get_reader(), tok) + "'");
|
2011-07-27 12:19:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
let hi = p.get_hi_pos();
|
2012-01-10 21:50:40 +00:00
|
|
|
if (have_disr && !all_nullary) {
|
2012-01-11 01:33:26 +00:00
|
|
|
p.fatal("discriminator values can only be used with a c-like enum");
|
2012-01-10 21:50:40 +00:00
|
|
|
}
|
2010-11-24 19:36:35 +00:00
|
|
|
p.bump();
|
2011-06-16 09:53:06 +00:00
|
|
|
ret mk_item(p, lo, hi, id, ast::item_tag(variants, ty_params), attrs);
|
2010-11-24 19:36:35 +00:00
|
|
|
}
|
|
|
|
|
2011-10-14 21:57:21 +00:00
|
|
|
fn parse_fn_ty_proto(p: parser) -> ast::proto {
|
2011-10-21 03:34:04 +00:00
|
|
|
if p.peek() == token::AT {
|
2011-10-14 21:57:21 +00:00
|
|
|
p.bump();
|
2012-01-10 14:49:15 +00:00
|
|
|
ast::proto_box
|
|
|
|
} else if p.peek() == token::TILDE {
|
|
|
|
p.bump();
|
|
|
|
ast::proto_uniq
|
2011-10-14 21:57:21 +00:00
|
|
|
} else {
|
2011-10-18 22:07:40 +00:00
|
|
|
ast::proto_bare
|
2011-10-14 21:57:21 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-01-12 20:31:45 +00:00
|
|
|
fn fn_expr_lookahead(tok: token::token) -> bool {
|
|
|
|
alt tok {
|
|
|
|
token::LPAREN. | token::AT. | token::TILDE. | token::BINOP(_) {
|
|
|
|
true
|
|
|
|
}
|
|
|
|
_ {
|
|
|
|
false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_item(p: parser, attrs: [ast::attribute]) -> option::t<@ast::item> {
|
2011-09-02 22:34:58 +00:00
|
|
|
if eat_word(p, "const") {
|
2011-07-29 13:01:06 +00:00
|
|
|
ret some(parse_item_const(p, attrs));
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "inline") {
|
|
|
|
expect_word(p, "fn");
|
2011-12-30 04:07:55 +00:00
|
|
|
ret some(parse_item_fn(p, ast::impure_fn, attrs));
|
2012-01-12 20:31:45 +00:00
|
|
|
} else if is_word(p, "fn") && !fn_expr_lookahead(p.look_ahead(1u)) {
|
2011-07-29 13:01:06 +00:00
|
|
|
p.bump();
|
2011-12-30 04:07:55 +00:00
|
|
|
ret some(parse_item_fn(p, ast::impure_fn, attrs));
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "pure") {
|
|
|
|
expect_word(p, "fn");
|
2011-12-30 04:07:55 +00:00
|
|
|
ret some(parse_item_fn(p, ast::pure_fn, attrs));
|
2011-10-07 04:33:04 +00:00
|
|
|
} else if is_word(p, "unsafe") && p.look_ahead(1u) != token::LBRACE {
|
|
|
|
p.bump();
|
2011-10-06 22:29:54 +00:00
|
|
|
expect_word(p, "fn");
|
2011-12-30 04:07:55 +00:00
|
|
|
ret some(parse_item_fn(p, ast::unsafe_fn, attrs));
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "mod") {
|
2011-07-29 13:01:06 +00:00
|
|
|
ret some(parse_item_mod(p, attrs));
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "native") {
|
2011-07-29 13:01:06 +00:00
|
|
|
ret some(parse_item_native_mod(p, attrs));
|
2011-12-13 12:19:56 +00:00
|
|
|
} if eat_word(p, "type") {
|
2011-07-29 13:01:06 +00:00
|
|
|
ret some(parse_item_type(p, attrs));
|
2012-01-10 23:29:50 +00:00
|
|
|
} else if eat_word(p, "tag") || eat_word(p, "enum") {
|
2011-07-29 13:01:06 +00:00
|
|
|
ret some(parse_item_tag(p, attrs));
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if is_word(p, "obj") && p.look_ahead(1u) != token::LPAREN {
|
2011-07-29 13:01:06 +00:00
|
|
|
p.bump();
|
|
|
|
ret some(parse_item_obj(p, attrs));
|
2011-12-20 15:33:55 +00:00
|
|
|
} else if eat_word(p, "iface") {
|
|
|
|
ret some(parse_item_iface(p, attrs));
|
2011-12-13 12:19:56 +00:00
|
|
|
} else if eat_word(p, "impl") {
|
|
|
|
ret some(parse_item_impl(p, attrs));
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "resource") {
|
2011-07-29 13:01:06 +00:00
|
|
|
ret some(parse_item_res(p, attrs));
|
|
|
|
} else { ret none; }
|
2010-09-21 23:22:32 +00:00
|
|
|
}
|
|
|
|
|
2011-06-15 20:27:39 +00:00
|
|
|
// A type to distingush between the parsing of item attributes or syntax
|
|
|
|
// extensions, which both begin with token.POUND
|
2011-08-12 14:15:18 +00:00
|
|
|
type attr_or_ext = option::t<either::t<[ast::attribute], @ast::expr>>;
|
2011-06-15 20:27:39 +00:00
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_outer_attrs_or_ext(p: parser) -> attr_or_ext {
|
2011-07-27 12:19:39 +00:00
|
|
|
if p.peek() == token::POUND {
|
|
|
|
let lo = p.get_lo_pos();
|
2011-06-15 20:27:39 +00:00
|
|
|
p.bump();
|
2011-07-27 12:19:39 +00:00
|
|
|
if p.peek() == token::LBRACKET {
|
|
|
|
let first_attr = parse_attribute_naked(p, ast::attr_outer, lo);
|
2011-08-19 22:16:48 +00:00
|
|
|
ret some(left([first_attr] + parse_outer_attributes(p)));
|
|
|
|
} else if !(p.peek() == token::LT || p.peek() == token::LBRACKET) {
|
2011-06-16 19:19:27 +00:00
|
|
|
ret some(right(parse_syntax_ext_naked(p, lo)));
|
2011-07-27 12:19:39 +00:00
|
|
|
} else { ret none; }
|
|
|
|
} else { ret none; }
|
2011-06-15 20:27:39 +00:00
|
|
|
}
|
|
|
|
|
2011-06-16 19:19:27 +00:00
|
|
|
// Parse attributes that appear before an item
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_outer_attributes(p: parser) -> [ast::attribute] {
|
2011-08-19 22:16:48 +00:00
|
|
|
let attrs: [ast::attribute] = [];
|
2011-07-27 12:19:39 +00:00
|
|
|
while p.peek() == token::POUND {
|
2011-08-19 22:16:48 +00:00
|
|
|
attrs += [parse_attribute(p, ast::attr_outer)];
|
2011-06-16 19:19:27 +00:00
|
|
|
}
|
2011-06-14 22:39:23 +00:00
|
|
|
ret attrs;
|
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_attribute(p: parser, style: ast::attr_style) -> ast::attribute {
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_lo_pos();
|
2011-06-14 23:13:19 +00:00
|
|
|
expect(p, token::POUND);
|
2011-06-16 19:52:43 +00:00
|
|
|
ret parse_attribute_naked(p, style, lo);
|
2011-06-15 20:27:39 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_attribute_naked(p: parser, style: ast::attr_style, lo: uint) ->
|
2011-07-27 12:19:39 +00:00
|
|
|
ast::attribute {
|
2011-06-14 23:13:19 +00:00
|
|
|
expect(p, token::LBRACKET);
|
2011-07-27 12:19:39 +00:00
|
|
|
let meta_item = parse_meta_item(p);
|
2011-06-14 23:13:19 +00:00
|
|
|
expect(p, token::RBRACKET);
|
2011-07-27 12:19:39 +00:00
|
|
|
let hi = p.get_hi_pos();
|
|
|
|
ret spanned(lo, hi, {style: style, value: *meta_item});
|
2011-06-14 23:13:19 +00:00
|
|
|
}
|
|
|
|
|
2011-06-16 19:19:27 +00:00
|
|
|
// Parse attributes that appear after the opening of an item, each terminated
|
|
|
|
// by a semicolon. In addition to a vector of inner attributes, this function
|
|
|
|
// also returns a vector that may contain the first outer attribute of the
|
|
|
|
// next item (since we can't know whether the attribute is an inner attribute
|
|
|
|
// of the containing item or an outer attribute of the first contained item
|
|
|
|
// until we see the semi).
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_inner_attrs_and_next(p: parser) ->
|
2011-08-04 23:20:09 +00:00
|
|
|
{inner: [ast::attribute], next: [ast::attribute]} {
|
2011-08-19 22:16:48 +00:00
|
|
|
let inner_attrs: [ast::attribute] = [];
|
|
|
|
let next_outer_attrs: [ast::attribute] = [];
|
2011-07-27 12:19:39 +00:00
|
|
|
while p.peek() == token::POUND {
|
|
|
|
let attr = parse_attribute(p, ast::attr_inner);
|
|
|
|
if p.peek() == token::SEMI {
|
2011-06-16 19:19:27 +00:00
|
|
|
p.bump();
|
2011-08-19 22:16:48 +00:00
|
|
|
inner_attrs += [attr];
|
2011-06-16 19:19:27 +00:00
|
|
|
} else {
|
2011-06-16 19:52:43 +00:00
|
|
|
// It's not really an inner attribute
|
2011-07-27 12:19:39 +00:00
|
|
|
let outer_attr =
|
|
|
|
spanned(attr.span.lo, attr.span.hi,
|
|
|
|
{style: ast::attr_outer, value: attr.node.value});
|
2011-08-19 22:16:48 +00:00
|
|
|
next_outer_attrs += [outer_attr];
|
2011-06-16 19:19:27 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2011-07-27 12:19:39 +00:00
|
|
|
ret {inner: inner_attrs, next: next_outer_attrs};
|
2010-12-25 01:03:46 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_meta_item(p: parser) -> @ast::meta_item {
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_lo_pos();
|
|
|
|
let ident = parse_ident(p);
|
|
|
|
alt p.peek() {
|
|
|
|
token::EQ. {
|
|
|
|
p.bump();
|
|
|
|
let lit = parse_lit(p);
|
|
|
|
let hi = p.get_hi_pos();
|
|
|
|
ret @spanned(lo, hi, ast::meta_name_value(ident, lit));
|
|
|
|
}
|
|
|
|
token::LPAREN. {
|
|
|
|
let inner_items = parse_meta_seq(p);
|
|
|
|
let hi = p.get_hi_pos();
|
|
|
|
ret @spanned(lo, hi, ast::meta_list(ident, inner_items));
|
|
|
|
}
|
|
|
|
_ {
|
|
|
|
let hi = p.get_hi_pos();
|
|
|
|
ret @spanned(lo, hi, ast::meta_word(ident));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_meta_seq(p: parser) -> [@ast::meta_item] {
|
2011-11-21 15:36:11 +00:00
|
|
|
ret parse_seq(token::LPAREN, token::RPAREN, seq_sep(token::COMMA),
|
2011-07-15 00:26:10 +00:00
|
|
|
parse_meta_item, p).node;
|
2010-12-25 01:03:46 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_optional_meta(p: parser) -> [@ast::meta_item] {
|
2011-08-19 22:16:48 +00:00
|
|
|
alt p.peek() { token::LPAREN. { ret parse_meta_seq(p); } _ { ret []; } }
|
2010-12-25 01:03:46 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_use(p: parser) -> ast::view_item_ {
|
2011-07-27 12:19:39 +00:00
|
|
|
let ident = parse_ident(p);
|
|
|
|
let metadata = parse_optional_meta(p);
|
2011-07-12 02:00:56 +00:00
|
|
|
ret ast::view_item_use(ident, metadata, p.get_id());
|
2010-12-30 16:21:37 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_rest_import_name(p: parser, first: ast::ident,
|
2011-08-12 14:15:18 +00:00
|
|
|
def_ident: option::t<ast::ident>) ->
|
2011-07-12 02:00:56 +00:00
|
|
|
ast::view_item_ {
|
2011-08-19 22:16:48 +00:00
|
|
|
let identifiers: [ast::ident] = [first];
|
2011-07-27 12:19:39 +00:00
|
|
|
let glob: bool = false;
|
2011-08-16 22:21:30 +00:00
|
|
|
let from_idents = option::none::<[ast::import_ident]>;
|
2011-07-27 12:19:39 +00:00
|
|
|
while true {
|
|
|
|
alt p.peek() {
|
|
|
|
token::SEMI. { break; }
|
|
|
|
token::MOD_SEP. {
|
2011-09-02 22:34:58 +00:00
|
|
|
if glob { p.fatal("cannot path into a glob"); }
|
2011-08-16 22:21:30 +00:00
|
|
|
if option::is_some(from_idents) {
|
2011-09-02 22:34:58 +00:00
|
|
|
p.fatal("cannot path into import list");
|
2011-08-16 22:21:30 +00:00
|
|
|
}
|
2011-07-27 12:19:39 +00:00
|
|
|
p.bump();
|
|
|
|
}
|
2011-09-02 22:34:58 +00:00
|
|
|
_ { p.fatal("expecting '::' or ';'"); }
|
2011-01-28 16:54:59 +00:00
|
|
|
}
|
2011-07-27 12:19:39 +00:00
|
|
|
alt p.peek() {
|
2011-08-19 22:16:48 +00:00
|
|
|
token::IDENT(_, _) { identifiers += [parse_ident(p)]; }
|
|
|
|
|
2011-07-27 12:19:39 +00:00
|
|
|
|
2011-09-02 22:34:58 +00:00
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
|
2011-09-12 10:39:38 +00:00
|
|
|
|
2011-07-27 12:19:39 +00:00
|
|
|
//the lexer can't tell the different kinds of stars apart ) :
|
|
|
|
token::BINOP(token::STAR.) {
|
|
|
|
glob = true;
|
|
|
|
p.bump();
|
|
|
|
}
|
2011-08-16 22:21:30 +00:00
|
|
|
|
2011-08-19 22:16:48 +00:00
|
|
|
|
2011-09-02 22:34:58 +00:00
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
|
2011-09-12 10:39:38 +00:00
|
|
|
|
2011-08-16 22:21:30 +00:00
|
|
|
token::LBRACE. {
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_import_ident(p: parser) -> ast::import_ident {
|
2011-08-16 22:21:30 +00:00
|
|
|
let lo = p.get_lo_pos();
|
|
|
|
let ident = parse_ident(p);
|
|
|
|
let hi = p.get_hi_pos();
|
2011-08-19 22:16:48 +00:00
|
|
|
ret spanned(lo, hi, {name: ident, id: p.get_id()});
|
2011-08-16 22:21:30 +00:00
|
|
|
}
|
2011-08-19 22:16:48 +00:00
|
|
|
let from_idents_ =
|
2011-11-21 15:36:11 +00:00
|
|
|
parse_seq(token::LBRACE, token::RBRACE, seq_sep(token::COMMA),
|
2011-08-19 22:16:48 +00:00
|
|
|
parse_import_ident, p).node;
|
2011-08-16 22:21:30 +00:00
|
|
|
if vec::is_empty(from_idents_) {
|
2011-09-02 22:34:58 +00:00
|
|
|
p.fatal("at least one import is required");
|
2011-08-16 22:21:30 +00:00
|
|
|
}
|
|
|
|
from_idents = some(from_idents_);
|
|
|
|
}
|
|
|
|
|
2011-08-19 22:16:48 +00:00
|
|
|
|
2011-09-02 22:34:58 +00:00
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
|
2011-09-12 10:39:38 +00:00
|
|
|
|
2011-08-19 22:16:48 +00:00
|
|
|
_ {
|
2011-09-02 22:34:58 +00:00
|
|
|
p.fatal("expecting an identifier, or '*'");
|
2011-08-19 22:16:48 +00:00
|
|
|
}
|
2011-07-27 12:19:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
alt def_ident {
|
|
|
|
some(i) {
|
2011-09-02 22:34:58 +00:00
|
|
|
if glob { p.fatal("globbed imports can't be renamed"); }
|
2011-08-16 22:21:30 +00:00
|
|
|
if option::is_some(from_idents) {
|
2011-09-02 22:34:58 +00:00
|
|
|
p.fatal("can't rename import list");
|
2011-08-16 22:21:30 +00:00
|
|
|
}
|
2011-11-30 12:38:38 +00:00
|
|
|
ret ast::view_item_import(i, @identifiers, p.get_id());
|
2011-07-27 12:19:39 +00:00
|
|
|
}
|
|
|
|
_ {
|
|
|
|
if glob {
|
2011-11-30 12:38:38 +00:00
|
|
|
ret ast::view_item_import_glob(@identifiers, p.get_id());
|
2011-08-16 22:21:30 +00:00
|
|
|
} else if option::is_some(from_idents) {
|
2011-11-30 12:38:38 +00:00
|
|
|
ret ast::view_item_import_from(@identifiers,
|
2011-08-16 22:21:30 +00:00
|
|
|
option::get(from_idents),
|
|
|
|
p.get_id());
|
2011-07-27 12:19:39 +00:00
|
|
|
} else {
|
2011-08-15 23:38:23 +00:00
|
|
|
let len = vec::len(identifiers);
|
2011-11-30 12:38:38 +00:00
|
|
|
ret ast::view_item_import(identifiers[len - 1u], @identifiers,
|
2011-07-27 12:19:39 +00:00
|
|
|
p.get_id());
|
2011-01-28 16:54:59 +00:00
|
|
|
}
|
2011-07-27 12:19:39 +00:00
|
|
|
}
|
2011-01-28 16:54:59 +00:00
|
|
|
}
|
2010-12-25 04:25:02 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_full_import_name(p: parser, def_ident: ast::ident) ->
|
2011-07-12 02:00:56 +00:00
|
|
|
ast::view_item_ {
|
2011-07-27 12:19:39 +00:00
|
|
|
alt p.peek() {
|
|
|
|
token::IDENT(i, _) {
|
|
|
|
p.bump();
|
2011-09-02 22:34:58 +00:00
|
|
|
ret parse_rest_import_name(p, p.get_str(i), some(def_ident));
|
2011-07-27 12:19:39 +00:00
|
|
|
}
|
2011-09-02 22:34:58 +00:00
|
|
|
_ { p.fatal("expecting an identifier"); }
|
2010-12-25 04:25:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_import(p: parser) -> ast::view_item_ {
|
2011-07-27 12:19:39 +00:00
|
|
|
alt p.peek() {
|
|
|
|
token::IDENT(i, _) {
|
|
|
|
p.bump();
|
|
|
|
alt p.peek() {
|
|
|
|
token::EQ. {
|
2010-12-25 04:25:02 +00:00
|
|
|
p.bump();
|
2011-08-27 07:43:22 +00:00
|
|
|
ret parse_full_import_name(p, p.get_str(i));
|
2011-08-26 00:00:12 +00:00
|
|
|
}
|
2011-09-02 22:34:58 +00:00
|
|
|
_ { ret parse_rest_import_name(p, p.get_str(i), none); }
|
2010-12-25 04:25:02 +00:00
|
|
|
}
|
2011-07-27 12:19:39 +00:00
|
|
|
}
|
2011-09-02 22:34:58 +00:00
|
|
|
_ { p.fatal("expecting an identifier"); }
|
2010-12-25 04:25:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_export(p: parser) -> ast::view_item_ {
|
2011-08-19 22:16:48 +00:00
|
|
|
let ids =
|
2011-11-21 15:36:11 +00:00
|
|
|
parse_seq_to_before_end(token::SEMI, seq_sep(token::COMMA),
|
2011-08-19 22:16:48 +00:00
|
|
|
parse_ident, p);
|
2011-08-16 19:09:47 +00:00
|
|
|
ret ast::view_item_export(ids, p.get_id());
|
2011-03-02 21:50:42 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_view_item(p: parser) -> @ast::view_item {
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_lo_pos();
|
|
|
|
let the_item =
|
2011-09-02 22:34:58 +00:00
|
|
|
if eat_word(p, "use") {
|
2011-07-27 12:19:39 +00:00
|
|
|
parse_use(p)
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "import") {
|
2011-07-27 12:19:39 +00:00
|
|
|
parse_import(p)
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if eat_word(p, "export") { parse_export(p) } else { fail };
|
2011-07-27 12:19:39 +00:00
|
|
|
let hi = p.get_lo_pos();
|
2011-07-12 02:00:56 +00:00
|
|
|
expect(p, token::SEMI);
|
|
|
|
ret @spanned(lo, hi, the_item);
|
2011-01-01 17:55:18 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn is_view_item(p: parser) -> bool {
|
2011-07-27 12:19:39 +00:00
|
|
|
alt p.peek() {
|
|
|
|
token::IDENT(sid, false) {
|
|
|
|
let st = p.get_str(sid);
|
2011-09-02 22:34:58 +00:00
|
|
|
ret str::eq(st, "use") || str::eq(st, "import") ||
|
|
|
|
str::eq(st, "export");
|
2011-07-27 12:19:39 +00:00
|
|
|
}
|
|
|
|
_ { ret false; }
|
2011-01-01 17:55:18 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_view(p: parser) -> [@ast::view_item] {
|
2011-08-19 22:16:48 +00:00
|
|
|
let items: [@ast::view_item] = [];
|
|
|
|
while is_view_item(p) { items += [parse_view_item(p)]; }
|
2011-01-01 17:55:18 +00:00
|
|
|
ret items;
|
2010-12-25 01:03:46 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_native_view(p: parser) -> [@ast::view_item] {
|
2011-08-19 22:16:48 +00:00
|
|
|
let items: [@ast::view_item] = [];
|
|
|
|
while is_view_item(p) { items += [parse_view_item(p)]; }
|
2011-03-07 19:48:43 +00:00
|
|
|
ret items;
|
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_crate_from_source_file(input: str, cfg: ast::crate_cfg,
|
|
|
|
sess: parse_sess) -> @ast::crate {
|
2011-07-30 22:50:16 +00:00
|
|
|
let p = new_parser_from_file(sess, cfg, input, 0u, 0u, SOURCE_FILE);
|
2011-08-18 08:01:39 +00:00
|
|
|
ret parse_crate_mod(p, cfg);
|
2011-07-12 00:21:02 +00:00
|
|
|
}
|
|
|
|
|
2011-12-20 21:38:10 +00:00
|
|
|
|
|
|
|
fn parse_expr_from_source_str(name: str, source: str, cfg: ast::crate_cfg,
|
|
|
|
sess: parse_sess) -> @ast::expr {
|
|
|
|
let p = new_parser_from_source_str(sess, cfg, name, source);
|
|
|
|
ret parse_expr(p);
|
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_crate_from_source_str(name: str, source: str, cfg: ast::crate_cfg,
|
|
|
|
sess: parse_sess) -> @ast::crate {
|
2011-12-20 21:38:10 +00:00
|
|
|
let p = new_parser_from_source_str(sess, cfg, name, source);
|
2011-08-18 08:01:39 +00:00
|
|
|
ret parse_crate_mod(p, cfg);
|
2011-07-12 00:21:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Parses a source module as a crate
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_crate_mod(p: parser, _cfg: ast::crate_cfg) -> @ast::crate {
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_lo_pos();
|
|
|
|
let crate_attrs = parse_inner_attrs_and_next(p);
|
|
|
|
let first_item_outer_attrs = crate_attrs.next;
|
|
|
|
let m = parse_mod_items(p, token::EOF, first_item_outer_attrs);
|
|
|
|
ret @spanned(lo, p.get_lo_pos(),
|
2011-08-19 22:16:48 +00:00
|
|
|
{directives: [],
|
2011-07-27 12:19:39 +00:00
|
|
|
module: m,
|
|
|
|
attrs: crate_attrs.inner,
|
|
|
|
config: p.get_cfg()});
|
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_str(p: parser) -> str {
|
2011-07-27 12:19:39 +00:00
|
|
|
alt p.peek() {
|
2011-10-20 01:04:44 +00:00
|
|
|
token::LIT_STR(s) { p.bump(); p.get_str(s) }
|
|
|
|
_ {
|
|
|
|
p.fatal("expected string literal")
|
|
|
|
}
|
2011-06-23 22:42:55 +00:00
|
|
|
}
|
|
|
|
}
|
2011-06-15 18:19:50 +00:00
|
|
|
|
2011-01-11 02:18:16 +00:00
|
|
|
// Logic for parsing crate files (.rc)
|
|
|
|
//
|
|
|
|
// Each crate file is a sequence of directives.
|
|
|
|
//
|
|
|
|
// Each directive imperatively extends its environment with 0 or more items.
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_crate_directive(p: parser, first_outer_attr: [ast::attribute]) ->
|
2011-07-27 12:19:39 +00:00
|
|
|
ast::crate_directive {
|
2011-06-29 01:15:39 +00:00
|
|
|
|
|
|
|
// Collect the next attributes
|
2011-07-27 12:19:39 +00:00
|
|
|
let outer_attrs = first_outer_attr + parse_outer_attributes(p);
|
2011-06-29 01:15:39 +00:00
|
|
|
// In a crate file outer attributes are only going to apply to mods
|
2011-08-15 23:38:23 +00:00
|
|
|
let expect_mod = vec::len(outer_attrs) > 0u;
|
2011-06-29 01:15:39 +00:00
|
|
|
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_lo_pos();
|
2011-09-02 22:34:58 +00:00
|
|
|
if expect_mod || is_word(p, "mod") {
|
|
|
|
expect_word(p, "mod");
|
2011-07-27 12:19:39 +00:00
|
|
|
let id = parse_ident(p);
|
|
|
|
alt p.peek() {
|
|
|
|
// mod x = "foo.rs";
|
|
|
|
token::SEMI. {
|
|
|
|
let hi = p.get_hi_pos();
|
|
|
|
p.bump();
|
2011-11-22 04:31:09 +00:00
|
|
|
ret spanned(lo, hi, ast::cdir_src_mod(id, outer_attrs));
|
2011-07-27 12:19:39 +00:00
|
|
|
}
|
|
|
|
// mod x = "foo_dir" { ...directives... }
|
|
|
|
token::LBRACE. {
|
|
|
|
p.bump();
|
|
|
|
let inner_attrs = parse_inner_attrs_and_next(p);
|
|
|
|
let mod_attrs = outer_attrs + inner_attrs.inner;
|
|
|
|
let next_outer_attr = inner_attrs.next;
|
|
|
|
let cdirs =
|
|
|
|
parse_crate_directives(p, token::RBRACE, next_outer_attr);
|
|
|
|
let hi = p.get_hi_pos();
|
|
|
|
expect(p, token::RBRACE);
|
|
|
|
ret spanned(lo, hi,
|
2011-11-22 04:31:09 +00:00
|
|
|
ast::cdir_dir_mod(id, cdirs, mod_attrs));
|
2011-07-27 12:19:39 +00:00
|
|
|
}
|
|
|
|
t { unexpected(p, t); }
|
2011-02-25 01:00:24 +00:00
|
|
|
}
|
2011-08-19 22:16:48 +00:00
|
|
|
} else if is_view_item(p) {
|
2011-07-27 12:19:39 +00:00
|
|
|
let vi = parse_view_item(p);
|
2011-05-13 19:30:08 +00:00
|
|
|
ret spanned(lo, vi.span.hi, ast::cdir_view_item(vi));
|
2011-09-02 22:34:58 +00:00
|
|
|
} else { ret p.fatal("expected crate directive"); }
|
2011-01-11 02:18:16 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_crate_directives(p: parser, term: token::token,
|
|
|
|
first_outer_attr: [ast::attribute]) ->
|
2011-08-04 23:20:09 +00:00
|
|
|
[@ast::crate_directive] {
|
2011-06-29 01:15:39 +00:00
|
|
|
|
|
|
|
// This is pretty ugly. If we have an outer attribute then we can't accept
|
|
|
|
// seeing the terminator next, so if we do see it then fail the same way
|
|
|
|
// parse_crate_directive would
|
2011-08-15 23:38:23 +00:00
|
|
|
if vec::len(first_outer_attr) > 0u && p.peek() == term {
|
2011-09-02 22:34:58 +00:00
|
|
|
expect_word(p, "mod");
|
2011-06-29 01:15:39 +00:00
|
|
|
}
|
|
|
|
|
2011-08-19 22:16:48 +00:00
|
|
|
let cdirs: [@ast::crate_directive] = [];
|
2011-11-25 00:17:26 +00:00
|
|
|
let first_outer_attr = first_outer_attr;
|
2011-07-27 12:19:39 +00:00
|
|
|
while p.peek() != term {
|
|
|
|
let cdir = @parse_crate_directive(p, first_outer_attr);
|
2011-08-19 22:16:48 +00:00
|
|
|
cdirs += [cdir];
|
2011-11-25 00:17:26 +00:00
|
|
|
first_outer_attr = [];
|
2011-01-11 02:18:16 +00:00
|
|
|
}
|
2011-02-24 20:14:05 +00:00
|
|
|
ret cdirs;
|
2011-01-11 02:18:16 +00:00
|
|
|
}
|
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_crate_from_crate_file(input: str, cfg: ast::crate_cfg,
|
|
|
|
sess: parse_sess) -> @ast::crate {
|
2011-07-30 22:50:16 +00:00
|
|
|
let p = new_parser_from_file(sess, cfg, input, 0u, 0u, CRATE_FILE);
|
2011-07-27 12:19:39 +00:00
|
|
|
let lo = p.get_lo_pos();
|
2011-09-02 22:34:58 +00:00
|
|
|
let prefix = std::fs::dirname(p.get_filemap().name);
|
2011-07-27 12:19:39 +00:00
|
|
|
let leading_attrs = parse_inner_attrs_and_next(p);
|
|
|
|
let crate_attrs = leading_attrs.inner;
|
|
|
|
let first_cdir_attr = leading_attrs.next;
|
|
|
|
let cdirs = parse_crate_directives(p, token::EOF, first_cdir_attr);
|
|
|
|
let cx =
|
|
|
|
@{p: p,
|
|
|
|
sess: sess,
|
|
|
|
mutable chpos: p.get_chpos(),
|
|
|
|
mutable byte_pos: p.get_byte_pos(),
|
|
|
|
cfg: p.get_cfg()};
|
2011-10-29 08:21:43 +00:00
|
|
|
let (companionmod, _) = fs::splitext(fs::basename(input));
|
|
|
|
let (m, attrs) = eval::eval_crate_directives_to_mod(
|
|
|
|
cx, cdirs, prefix, option::some(companionmod));
|
2011-07-27 12:19:39 +00:00
|
|
|
let hi = p.get_hi_pos();
|
2011-05-12 15:24:54 +00:00
|
|
|
expect(p, token::EOF);
|
2011-07-27 12:19:39 +00:00
|
|
|
ret @spanned(lo, hi,
|
|
|
|
{directives: cdirs,
|
|
|
|
module: m,
|
2011-10-29 08:21:43 +00:00
|
|
|
attrs: crate_attrs + attrs,
|
2011-07-27 12:19:39 +00:00
|
|
|
config: p.get_cfg()});
|
2010-09-21 23:22:32 +00:00
|
|
|
}
|
2011-07-30 22:50:16 +00:00
|
|
|
|
2011-09-12 09:27:30 +00:00
|
|
|
fn parse_crate_from_file(input: str, cfg: ast::crate_cfg, sess: parse_sess) ->
|
|
|
|
@ast::crate {
|
2011-09-02 22:34:58 +00:00
|
|
|
if str::ends_with(input, ".rc") {
|
2011-07-30 22:50:16 +00:00
|
|
|
parse_crate_from_crate_file(input, cfg, sess)
|
2011-09-02 22:34:58 +00:00
|
|
|
} else if str::ends_with(input, ".rs") {
|
2011-07-30 22:50:16 +00:00
|
|
|
parse_crate_from_source_file(input, cfg, sess)
|
|
|
|
} else {
|
2011-09-02 22:34:58 +00:00
|
|
|
codemap::emit_error(none, "unknown input file type: " + input,
|
2011-07-30 22:50:16 +00:00
|
|
|
sess.cm);
|
|
|
|
fail
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-12 17:29:23 +00:00
|
|
|
//
|
|
|
|
// Local Variables:
|
|
|
|
// mode: rust
|
|
|
|
// fill-column: 78;
|
|
|
|
// indent-tabs-mode: nil
|
|
|
|
// c-basic-offset: 4
|
|
|
|
// buffer-file-coding-system: utf-8-unix
|
|
|
|
// End:
|
|
|
|
//
|