rust/src/comp/syntax/ast.rs

526 lines
14 KiB
Rust
Raw Normal View History

// The Rust abstract syntax tree.
import option;
2011-09-12 16:13:28 -07:00
import codemap::{span, filename};
type spanned<T> = {node: T, span: span};
2011-09-02 15:34:58 -07:00
type ident = str;
// Functions may or may not have names.
type fn_ident = option<ident>;
// FIXME: with typestate constraint, could say
// idents and types are the same length, and are
// non-empty
type path_ = {global: bool, idents: [ident], types: [@ty]};
type path = spanned<path_>;
type crate_num = int;
type node_id = int;
2011-07-27 14:19:39 +02:00
type def_id = {crate: crate_num, node: node_id};
2011-07-27 14:19:39 +02:00
const local_crate: crate_num = 0;
const crate_node_id: node_id = 0;
2012-01-19 14:24:03 -08:00
enum ty_param_bound {
2012-01-19 17:56:05 -08:00
bound_copy,
bound_send,
bound_iface(@ty),
}
type ty_param = {ident: ident, id: node_id, bounds: @[ty_param_bound]};
2012-01-19 14:24:03 -08:00
enum def {
2012-01-19 17:56:05 -08:00
def_fn(def_id, purity),
def_self(def_id),
def_mod(def_id),
def_native_mod(def_id),
def_const(def_id),
def_arg(def_id, mode),
def_local(def_id, let_style),
def_variant(def_id /* enum */, def_id /* variant */),
def_ty(def_id),
def_prim_ty(prim_ty),
2012-01-19 17:56:05 -08:00
def_ty_param(def_id, uint),
def_binding(def_id),
def_use(def_id),
def_upvar(def_id, @def, node_id), // node_id == expr_fn or expr_fn_block
}
// The set of meta_items that define the compilation environment of the crate,
// used to drive conditional compilation
type crate_cfg = [@meta_item];
type crate = spanned<crate_>;
2011-07-27 14:19:39 +02:00
type crate_ =
{directives: [@crate_directive],
2011-07-27 14:19:39 +02:00
module: _mod,
attrs: [attribute],
2011-07-27 14:19:39 +02:00
config: crate_cfg};
2012-01-19 14:24:03 -08:00
enum crate_directive_ {
2012-01-19 17:56:05 -08:00
cdir_src_mod(ident, [attribute]),
cdir_dir_mod(ident, [@crate_directive], [attribute]),
2012-01-19 17:56:05 -08:00
// NB: cdir_view_item is *not* processed by the rest of the compiler, the
// attached view_items are sunk into the crate's module during parsing,
2012-01-19 14:34:23 -08:00
// and processed (resolved, imported, etc.) there. This enum-variant
// exists only to preserve the view items in order in case we decide to
// pretty-print crates in the future.
2012-01-19 17:56:05 -08:00
cdir_view_item(@view_item),
2012-01-19 17:56:05 -08:00
cdir_syntax(@path),
}
type crate_directive = spanned<crate_directive_>;
type meta_item = spanned<meta_item_>;
2012-01-19 14:24:03 -08:00
enum meta_item_ {
2012-01-19 17:56:05 -08:00
meta_word(ident),
meta_list(ident, [@meta_item]),
meta_name_value(ident, lit),
}
2010-12-30 11:21:37 -05:00
type blk = spanned<blk_>;
type blk_ = {view_items: [@view_item], stmts: [@stmt], expr: option<@expr>,
id: node_id, rules: blk_check_mode};
2011-07-27 14:19:39 +02:00
type pat = {id: node_id, node: pat_, span: span};
2011-07-27 14:19:39 +02:00
type field_pat = {ident: ident, pat: @pat};
2011-07-11 14:13:20 +02:00
2012-01-19 14:24:03 -08:00
enum pat_ {
2012-01-19 17:56:05 -08:00
pat_wild,
// A pat_ident may either be a new bound variable,
2012-01-19 14:24:03 -08:00
// or a nullary enum (in which case the second field
// is none).
2012-01-19 14:24:03 -08:00
// In the nullary enum case, the parser can't determine
// which it is. The resolver determines this, and
// records this pattern's node_id in an auxiliary
// set (of "pat_idents that refer to nullary enums")
// After the resolution phase, code should never pattern-
// match on a pat directly! Always call pat_util::normalize_pat --
// it turns any pat_idents that refer to nullary enums into pat_enums.
pat_ident(@path, option<@pat>),
pat_enum(@path, [@pat]),
2012-01-19 17:56:05 -08:00
pat_rec([field_pat], bool),
pat_tup([@pat]),
pat_box(@pat),
pat_uniq(@pat),
pat_lit(@expr),
pat_range(@expr, @expr),
2010-11-24 14:42:01 -08:00
}
2012-01-19 17:56:05 -08:00
enum mutability { mut, imm, maybe_mut, }
2012-01-19 14:24:03 -08:00
enum proto {
2012-01-19 17:56:05 -08:00
proto_bare, // native fn
proto_any, // fn
proto_uniq, // fn~
proto_box, // fn@
proto_block, // fn&
2011-10-10 13:15:47 -07:00
}
pure fn is_blockish(p: ast::proto) -> bool {
alt p {
proto_any | proto_block { true }
proto_bare | proto_uniq | proto_box { false }
}
}
2012-01-19 14:24:03 -08:00
enum binop {
2012-01-19 17:56:05 -08:00
add,
subtract,
mul,
div,
rem,
and,
or,
bitxor,
bitand,
bitor,
lsl,
lsr,
asr,
eq,
lt,
le,
ne,
ge,
gt,
}
2012-01-19 14:24:03 -08:00
enum unop {
2012-01-19 17:56:05 -08:00
box(mutability),
uniq(mutability),
deref, not, neg,
}
// Generally, after typeck you can get the inferred value
// using ty::resolved_T(...).
enum inferable<T> {
expl(T), infer(node_id)
}
// "resolved" mode: the real modes.
enum rmode { by_ref, by_val, by_mut_ref, by_move, by_copy }
// inferable mode.
type mode = inferable<rmode>;
2010-12-03 18:03:28 -08:00
type stmt = spanned<stmt_>;
2012-01-19 14:24:03 -08:00
enum stmt_ {
2012-01-19 17:56:05 -08:00
stmt_decl(@decl, node_id),
// expr without trailing semi-colon (must have unit type):
2012-01-19 17:56:05 -08:00
stmt_expr(@expr, node_id),
// expr with trailing semi-colon (may have any type):
2012-01-19 17:56:05 -08:00
stmt_semi(@expr, node_id),
}
2012-01-19 17:56:05 -08:00
enum init_op { init_assign, init_move, }
2011-07-27 14:19:39 +02:00
type initializer = {op: init_op, expr: @expr};
type local_ = // FIXME: should really be a refinement on pat
{ty: @ty, pat: @pat, init: option<initializer>, id: node_id};
type local = spanned<local_>;
type decl = spanned<decl_>;
2012-01-19 17:56:05 -08:00
enum let_style { let_copy, let_ref, }
2012-01-19 17:56:05 -08:00
enum decl_ { decl_local([(let_style, @local)]), decl_item(@item), }
type arm = {pats: [@pat], guard: option<@expr>, body: blk};
2010-11-24 15:45:59 -08:00
2011-07-27 14:19:39 +02:00
type field_ = {mut: mutability, ident: ident, expr: @expr};
type field = spanned<field_>;
2012-01-19 17:56:05 -08:00
enum blk_check_mode { default_blk, unchecked_blk, unsafe_blk, }
2012-01-19 17:56:05 -08:00
enum expr_check_mode { claimed_expr, checked_expr, }
2011-07-27 14:19:39 +02:00
type expr = {id: node_id, node: expr_, span: span};
2012-01-19 14:24:03 -08:00
enum expr_ {
2012-01-19 17:56:05 -08:00
expr_vec([@expr], mutability),
expr_rec([field], option<@expr>),
2012-01-19 17:56:05 -08:00
expr_call(@expr, [@expr], bool),
expr_tup([@expr]),
expr_bind(@expr, [option<@expr>]),
2012-01-19 17:56:05 -08:00
expr_binary(binop, @expr, @expr),
expr_unary(unop, @expr),
expr_lit(@lit),
expr_cast(@expr, @ty),
expr_if(@expr, blk, option<@expr>),
2012-01-19 17:56:05 -08:00
expr_while(@expr, blk),
expr_for(@local, @expr, blk),
expr_do_while(blk, @expr),
expr_alt(@expr, [arm]),
expr_fn(proto, fn_decl, blk, @capture_clause),
expr_fn_block(fn_decl, blk),
expr_block(blk),
2011-06-16 16:55:46 -07:00
/*
* FIXME: many of these @exprs should be constrained with
* is_lval once we have constrained types working.
*/
2012-01-19 17:56:05 -08:00
expr_copy(@expr),
expr_move(@expr, @expr),
expr_assign(@expr, @expr),
expr_swap(@expr, @expr),
expr_assign_op(binop, @expr, @expr),
expr_field(@expr, ident, [@ty]),
expr_index(@expr, @expr),
expr_path(@path),
expr_fail(option<@expr>),
2012-01-19 17:56:05 -08:00
expr_break,
expr_cont,
expr_ret(option<@expr>),
2012-01-19 17:56:05 -08:00
expr_be(@expr),
expr_log(int, @expr, @expr),
/* just an assert, no significance to typestate */
2012-01-19 17:56:05 -08:00
expr_assert(@expr),
/* preds that typestate is aware of */
2012-01-19 17:56:05 -08:00
expr_check(expr_check_mode, @expr),
2011-07-27 14:19:39 +02:00
/* FIXME Would be nice if expr_check desugared
to expr_if_check. */
expr_if_check(@expr, blk, option<@expr>),
2012-01-19 17:56:05 -08:00
expr_mac(mac),
}
2011-12-17 21:12:30 -08:00
type capture_item = {
id: int,
name: ident, // Currently, can only capture a local var.
span: span
};
type capture_clause = {
copies: [@capture_item],
moves: [@capture_item]
};
/*
// Says whether this is a block the user marked as
// "unchecked"
2012-01-19 14:24:03 -08:00
enum blk_sort {
2012-01-19 17:56:05 -08:00
blk_unchecked, // declared as "exception to effect-checking rules"
blk_checked, // all typing rules apply
}
*/
type mac = spanned<mac_>;
type mac_arg = option::t<@expr>;
type mac_body_ = {span: span};
type mac_body = option::t<mac_body_>;
2012-01-19 14:24:03 -08:00
enum mac_ {
mac_invoc(@path, mac_arg, mac_body),
2012-01-19 17:56:05 -08:00
mac_embed_type(@ty),
mac_embed_block(blk),
mac_ellipsis,
// the span is used by the quoter/anti-quoter ...
mac_aq(span /* span of quote */, @expr), // anti-quote
mac_var(uint)
}
type lit = spanned<lit_>;
2012-01-19 14:24:03 -08:00
enum lit_ {
2012-01-19 17:56:05 -08:00
lit_str(str),
lit_int(i64, int_ty),
lit_uint(u64, uint_ty),
lit_float(str, float_ty),
lit_nil,
lit_bool(bool),
}
2010-11-03 16:43:12 -07:00
// NB: If you change this, you'll probably want to change the corresponding
// type structure in middle/ty.rs as well.
2011-07-27 14:19:39 +02:00
type mt = {ty: @ty, mut: mutability};
2011-07-27 14:19:39 +02:00
type ty_field_ = {ident: ident, mt: mt};
type ty_field = spanned<ty_field_>;
type ty_method = {ident: ident, attrs: [attribute],
decl: fn_decl, tps: [ty_param], span: span};
2012-01-19 17:56:05 -08:00
enum int_ty { ty_i, ty_char, ty_i8, ty_i16, ty_i32, ty_i64, }
2012-01-19 17:56:05 -08:00
enum uint_ty { ty_u, ty_u8, ty_u16, ty_u32, ty_u64, }
2012-01-19 17:56:05 -08:00
enum float_ty { ty_f, ty_f32, ty_f64, }
type ty = spanned<ty_>;
// Not represented directly in the AST, referred to by name through a ty_path.
enum prim_ty {
2012-01-19 17:56:05 -08:00
ty_int(int_ty),
ty_uint(uint_ty),
ty_float(float_ty),
ty_str,
ty_bool,
}
enum ty_ {
ty_nil,
ty_bot, /* bottom type */
2012-01-19 17:56:05 -08:00
ty_box(mt),
ty_uniq(mt),
ty_vec(mt),
ty_ptr(mt),
ty_rec([ty_field]),
ty_fn(proto, fn_decl),
ty_tup([@ty]),
ty_path(@path, node_id),
ty_constr(@ty, [@ty_constr]),
ty_mac(mac),
// ty_infer means the type should be inferred instead of it having been
// specified. This should only appear at the "top level" of a type and not
// nested in one.
2012-01-19 17:56:05 -08:00
ty_infer,
}
/*
A constraint arg that's a function argument is referred to by its position
rather than name. This is so we could have higher-order functions that have
constraints (potentially -- right now there's no way to write that), and also
so that the typestate pass doesn't have to map a function name onto its decl.
So, the constr_arg type is parameterized: it's instantiated with uint for
declarations, and ident for uses.
*/
2012-01-19 17:56:05 -08:00
enum constr_arg_general_<T> { carg_base, carg_ident(T), carg_lit(@lit), }
type fn_constr_arg = constr_arg_general_<uint>;
type sp_constr_arg<T> = spanned<constr_arg_general_<T>>;
type ty_constr_arg = sp_constr_arg<@path>;
type constr_arg = spanned<fn_constr_arg>;
// Constrained types' args are parameterized by paths, since
// we refer to paths directly and not by indices.
// The implicit root of such path, in the constraint-list for a
// constrained type, is * (referring to the base record)
type constr_general_<ARG, ID> =
{path: @path, args: [@spanned<constr_arg_general_<ARG>>], id: ID};
// In the front end, constraints have a node ID attached.
// Typeck turns this to a def_id, using the output of resolve.
type constr_general<ARG> = spanned<constr_general_<ARG, node_id>>;
type constr_ = constr_general_<uint, node_id>;
type constr = spanned<constr_general_<uint, node_id>>;
type ty_constr_ = constr_general_<@path, node_id>;
type ty_constr = spanned<ty_constr_>;
2011-06-16 16:55:46 -07:00
/* The parser generates ast::constrs; resolve generates
a mapping from each function to a list of ty::constr_defs,
corresponding to these. */
2011-07-27 14:19:39 +02:00
type arg = {mode: mode, ty: @ty, ident: ident, id: node_id};
type fn_decl =
{inputs: [arg],
2011-07-27 14:19:39 +02:00
output: @ty,
purity: purity,
cf: ret_style,
constraints: [@constr]};
2012-01-19 14:24:03 -08:00
enum purity {
2012-01-19 17:56:05 -08:00
pure_fn, // declared with "pure fn"
unsafe_fn, // declared with "unsafe fn"
impure_fn, // declared with "fn"
}
2012-01-19 14:24:03 -08:00
enum ret_style {
2012-01-19 17:56:05 -08:00
noreturn, // functions with return type _|_ that always
// raise an error or exit (i.e. never return to the caller)
2012-01-19 17:56:05 -08:00
return_val, // everything else
}
type method = {ident: ident, attrs: [attribute],
tps: [ty_param], decl: fn_decl, body: blk,
id: node_id, span: span};
2010-12-14 15:32:13 -08:00
type _mod = {view_items: [@view_item], items: [@item]};
2012-01-19 14:24:03 -08:00
enum native_abi {
2012-01-19 17:56:05 -08:00
native_abi_rust_intrinsic,
native_abi_cdecl,
native_abi_stdcall,
2011-02-23 14:06:37 -05:00
}
type native_mod =
2011-11-21 02:15:40 +08:00
{view_items: [@view_item],
items: [@native_item]};
2011-07-27 14:19:39 +02:00
type variant_arg = {ty: @ty, id: node_id};
type variant_ = {name: ident, attrs: [attribute], args: [variant_arg],
id: node_id, disr_expr: option<@expr>};
type variant = spanned<variant_>;
type view_item = spanned<view_item_>;
// FIXME: May want to just use path here, which would allow things like
// 'import ::foo'
type simple_path = [ident];
type import_ident_ = {name: ident, id: node_id};
type import_ident = spanned<import_ident_>;
2012-01-19 14:24:03 -08:00
enum view_item_ {
2012-01-19 17:56:05 -08:00
view_item_use(ident, [@meta_item], node_id),
view_item_import(ident, @simple_path, node_id),
view_item_import_glob(@simple_path, node_id),
view_item_import_from(@simple_path, [import_ident], node_id),
view_item_export([ident], node_id),
// export foo::{}
view_item_export_enum_none(ident, node_id),
// export foo::{bar, baz, blat}
view_item_export_enum_some(ident, [import_ident], node_id)
}
// Meta-data associated with an item
type attribute = spanned<attribute_>;
// Distinguishes between attributes that decorate items and attributes that
// are contained as statements within items. These two cases need to be
// distinguished for pretty-printing.
2012-01-19 17:56:05 -08:00
enum attr_style { attr_outer, attr_inner, }
2011-07-27 14:19:39 +02:00
type attribute_ = {style: attr_style, value: meta_item};
type item = {ident: ident, attrs: [attribute],
id: node_id, node: item_, span: span};
2012-01-19 14:24:03 -08:00
enum item_ {
2012-01-19 17:56:05 -08:00
item_const(@ty, @expr),
item_fn(fn_decl, [ty_param], blk),
item_mod(_mod),
item_native_mod(native_mod),
item_ty(@ty, [ty_param]),
item_enum([variant], [ty_param]),
item_res(fn_decl /* dtor */, [ty_param], blk,
2012-01-19 17:56:05 -08:00
node_id /* dtor id */, node_id /* ctor id */),
item_class([ty_param], /* ty params for class */
[@class_item], /* methods, etc. */
/* (not including ctor) */
fn_decl, /* ctor decl */
blk /* ctor body */
),
2012-01-19 17:56:05 -08:00
item_iface([ty_param], [ty_method]),
item_impl([ty_param], option<@ty> /* iface */,
2012-01-19 17:56:05 -08:00
@ty /* self */, [@method]),
2011-05-11 15:10:24 +02:00
}
type class_item_ = {privacy: privacy, decl: @class_member};
type class_item = spanned<class_item_>;
enum class_member {
instance_var(ident, @ty, class_mutability, node_id),
class_method(@item)
}
enum class_mutability { class_mutable, class_immutable }
enum privacy { priv, pub }
2011-07-27 14:19:39 +02:00
type native_item =
{ident: ident,
attrs: [attribute],
2011-07-27 14:19:39 +02:00
node: native_item_,
id: node_id,
span: span};
2012-01-19 14:24:03 -08:00
enum native_item_ {
2012-01-19 17:56:05 -08:00
native_item_fn(fn_decl, [ty_param]),
}
2010-08-12 10:27:50 -07:00
//
// Local Variables:
// mode: rust
// fill-column: 78;
// indent-tabs-mode: nil
// c-basic-offset: 4
// buffer-file-coding-system: utf-8-unix
// End:
//