about summary refs log tree commit diff
path: root/src/libsyntax/ast.rs
diff options
context:
space:
mode:
Diffstat (limited to 'src/libsyntax/ast.rs')
-rw-r--r--src/libsyntax/ast.rs743
1 files changed, 743 insertions, 0 deletions
diff --git a/src/libsyntax/ast.rs b/src/libsyntax/ast.rs
new file mode 100644
index 00000000000..f0a3006b8c6
--- /dev/null
+++ b/src/libsyntax/ast.rs
@@ -0,0 +1,743 @@
+// The Rust abstract syntax tree.
+
+import codemap::{span, filename};
+import std::serialization::{serializer,
+                            deserializer,
+                            serialize_option,
+                            deserialize_option,
+                            serialize_uint,
+                            deserialize_uint,
+                            serialize_int,
+                            deserialize_int,
+                            serialize_i64,
+                            deserialize_i64,
+                            serialize_u64,
+                            deserialize_u64,
+                            serialize_str,
+                            deserialize_str,
+                            serialize_bool,
+                            deserialize_bool};
+
+/* Note #1972 -- spans are serialized but not deserialized */
+fn serialize_span<S>(_s: S, _v: span) {
+}
+
+fn deserialize_span<D>(_d: D) -> span {
+    ast_util::dummy_sp()
+}
+
+#[auto_serialize]
+type spanned<T> = {node: T, span: span};
+
+#[auto_serialize]
+type ident = str;
+
+// Functions may or may not have names.
+#[auto_serialize]
+type fn_ident = option<ident>;
+
+#[auto_serialize]
+type path = {span: span,
+             global: bool,
+             idents: [ident],
+             rp: option<@region>,
+             types: [@ty]};
+
+#[auto_serialize]
+type crate_num = int;
+
+#[auto_serialize]
+type node_id = int;
+
+#[auto_serialize]
+type def_id = {crate: crate_num, node: node_id};
+
+const local_crate: crate_num = 0;
+const crate_node_id: node_id = 0;
+
+#[auto_serialize]
+enum ty_param_bound {
+    bound_copy,
+    bound_send,
+    bound_const,
+    bound_iface(@ty),
+}
+
+#[auto_serialize]
+type ty_param = {ident: ident, id: node_id, bounds: @[ty_param_bound]};
+
+#[auto_serialize]
+enum def {
+    def_fn(def_id, purity),
+    def_self(node_id),
+    def_mod(def_id),
+    def_native_mod(def_id),
+    def_const(def_id),
+    def_arg(node_id, mode),
+    def_local(node_id, bool /* is_mutbl */),
+    def_variant(def_id /* enum */, def_id /* variant */),
+    def_ty(def_id),
+    def_prim_ty(prim_ty),
+    def_ty_param(def_id, uint),
+    def_binding(node_id),
+    def_use(def_id),
+    def_upvar(node_id /* local id of closed over var */,
+              @def    /* closed over def */,
+              node_id /* expr node that creates the closure */),
+    def_class(def_id),
+    def_region(node_id)
+}
+
+// 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_>;
+
+type crate_ =
+    {directives: [@crate_directive],
+     module: _mod,
+     attrs: [attribute],
+     config: crate_cfg};
+
+enum crate_directive_ {
+    cdir_src_mod(ident, [attribute]),
+    cdir_dir_mod(ident, [@crate_directive], [attribute]),
+
+    // 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,
+    // 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.
+    cdir_view_item(@view_item),
+
+    cdir_syntax(@path),
+}
+
+type crate_directive = spanned<crate_directive_>;
+
+#[auto_serialize]
+type meta_item = spanned<meta_item_>;
+
+#[auto_serialize]
+enum meta_item_ {
+    meta_word(ident),
+    meta_list(ident, [@meta_item]),
+    meta_name_value(ident, lit),
+}
+
+#[auto_serialize]
+type blk = spanned<blk_>;
+
+#[auto_serialize]
+type blk_ = {view_items: [@view_item], stmts: [@stmt], expr: option<@expr>,
+             id: node_id, rules: blk_check_mode};
+
+#[auto_serialize]
+type pat = {id: node_id, node: pat_, span: span};
+
+#[auto_serialize]
+type field_pat = {ident: ident, pat: @pat};
+
+#[auto_serialize]
+enum pat_ {
+    pat_wild,
+    // A pat_ident may either be a new bound variable,
+    // or a nullary enum (in which case the second field
+    // is none).
+    // 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")
+    pat_ident(@path, option<@pat>),
+    pat_enum(@path, option<[@pat]>), // "none" means a * pattern where
+                                  // we don't bind the fields to names
+    pat_rec([field_pat], bool),
+    pat_tup([@pat]),
+    pat_box(@pat),
+    pat_uniq(@pat),
+    pat_lit(@expr),
+    pat_range(@expr, @expr),
+}
+
+#[auto_serialize]
+enum mutability { m_mutbl, m_imm, m_const, }
+
+#[auto_serialize]
+enum proto {
+    proto_bare,    // native fn
+    proto_any,     // fn
+    proto_uniq,    // fn~
+    proto_box,     // fn@
+    proto_block,   // fn&
+}
+
+#[auto_serialize]
+enum vstore {
+    /* FIXME: Change uint to @expr (actually only constant exprs,
+       as per #2112)
+     */
+    vstore_fixed(option<uint>),   // [1,2,3,4]/_ or 4
+    vstore_uniq,                  // [1,2,3,4]/~
+    vstore_box,                   // [1,2,3,4]/@
+    vstore_slice(@region)         // [1,2,3,4]/&(foo)?
+}
+
+pure fn is_blockish(p: ast::proto) -> bool {
+    alt p {
+      proto_any | proto_block { true }
+      proto_bare | proto_uniq | proto_box { false }
+    }
+}
+
+#[auto_serialize]
+enum binop {
+    add,
+    subtract,
+    mul,
+    div,
+    rem,
+    and,
+    or,
+    bitxor,
+    bitand,
+    bitor,
+    shl,
+    shr,
+    eq,
+    lt,
+    le,
+    ne,
+    ge,
+    gt,
+}
+
+#[auto_serialize]
+enum unop {
+    box(mutability),
+    uniq(mutability),
+    deref, not, neg
+}
+
+// Generally, after typeck you can get the inferred value
+// using ty::resolved_T(...).
+#[auto_serialize]
+enum inferable<T> {
+    expl(T), infer(node_id)
+}
+
+// "resolved" mode: the real modes.
+#[auto_serialize]
+enum rmode { by_ref, by_val, by_mutbl_ref, by_move, by_copy }
+
+// inferable mode.
+#[auto_serialize]
+type mode = inferable<rmode>;
+
+#[auto_serialize]
+type stmt = spanned<stmt_>;
+
+#[auto_serialize]
+enum stmt_ {
+    stmt_decl(@decl, node_id),
+
+    // expr without trailing semi-colon (must have unit type):
+    stmt_expr(@expr, node_id),
+
+    // expr with trailing semi-colon (may have any type):
+    stmt_semi(@expr, node_id),
+}
+
+#[auto_serialize]
+enum init_op { init_assign, init_move, }
+
+#[auto_serialize]
+type initializer = {op: init_op, expr: @expr};
+
+#[auto_serialize]
+type local_ =  /* FIXME: should really be a refinement on pat
+                  (pending discussion of #1697, #2178...)
+                */
+    {is_mutbl: bool, ty: @ty, pat: @pat,
+     init: option<initializer>, id: node_id};
+
+#[auto_serialize]
+type local = spanned<local_>;
+
+#[auto_serialize]
+type decl = spanned<decl_>;
+
+#[auto_serialize]
+enum decl_ { decl_local([@local]), decl_item(@item), }
+
+#[auto_serialize]
+type arm = {pats: [@pat], guard: option<@expr>, body: blk};
+
+#[auto_serialize]
+type field_ = {mutbl: mutability, ident: ident, expr: @expr};
+
+#[auto_serialize]
+type field = spanned<field_>;
+
+#[auto_serialize]
+enum blk_check_mode { default_blk, unchecked_blk, unsafe_blk, }
+
+#[auto_serialize]
+enum expr_check_mode { claimed_expr, checked_expr, }
+
+#[auto_serialize]
+type expr = {id: node_id, node: expr_, span: span};
+
+#[auto_serialize]
+enum alt_mode { alt_check, alt_exhaustive, }
+
+#[auto_serialize]
+enum expr_ {
+    expr_vstore(@expr, vstore),
+    expr_vec([@expr], mutability),
+    expr_rec([field], option<@expr>),
+    expr_call(@expr, [@expr], bool), // True iff last argument is a block
+    expr_tup([@expr]),
+    expr_bind(@expr, [option<@expr>]),
+    expr_binary(binop, @expr, @expr),
+    expr_unary(unop, @expr),
+    expr_lit(@lit),
+    expr_cast(@expr, @ty),
+    expr_if(@expr, blk, option<@expr>),
+    expr_while(@expr, blk),
+    /* Conditionless loop (can be exited with break, cont, ret, or fail)
+       Same semantics as while(true) { body }, but typestate knows that the
+       (implicit) condition is always true. */
+    expr_loop(blk),
+    expr_alt(@expr, [arm], alt_mode),
+    expr_fn(proto, fn_decl, blk, capture_clause),
+    expr_fn_block(fn_decl, blk, capture_clause),
+    // Inner expr is always an expr_fn_block. We need the wrapping node to
+    // sanely type this (a function returning nil on the inside but bool on
+    // the outside).
+    expr_loop_body(@expr),
+    expr_block(blk),
+
+    /*
+     * FIXME: many of these @exprs should be constrained with
+     * is_lval once we have constrained types working.
+     * (See #34)
+     */
+    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_addr_of(mutability, @expr),
+    expr_fail(option<@expr>),
+    expr_break,
+    expr_cont,
+    expr_ret(option<@expr>),
+    expr_log(int, @expr, @expr),
+
+    expr_new(/* arena */ @expr,
+             /* id for the alloc() call */ node_id,
+             /* value */ @expr),
+
+    /* just an assert, no significance to typestate */
+    expr_assert(@expr),
+
+    /* preds that typestate is aware of */
+    expr_check(expr_check_mode, @expr),
+    expr_if_check(@expr, blk, option<@expr>),
+    expr_mac(mac),
+}
+
+#[auto_serialize]
+type capture_item = @{
+    id: int,
+    is_move: bool,
+    name: ident, // Currently, can only capture a local var.
+    span: span
+};
+
+#[auto_serialize]
+type capture_clause = @[capture_item];
+
+/*
+// Says whether this is a block the user marked as
+// "unchecked"
+enum blk_sort {
+    blk_unchecked, // declared as "exception to effect-checking rules"
+    blk_checked, // all typing rules apply
+}
+*/
+
+#[auto_serialize]
+type mac = spanned<mac_>;
+
+#[auto_serialize]
+type mac_arg = option<@expr>;
+
+#[auto_serialize]
+type mac_body_ = {span: span};
+
+#[auto_serialize]
+type mac_body = option<mac_body_>;
+
+#[auto_serialize]
+enum mac_ {
+    mac_invoc(@path, mac_arg, mac_body),
+    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)
+}
+
+#[auto_serialize]
+type lit = spanned<lit_>;
+
+#[auto_serialize]
+enum lit_ {
+    lit_str(str),
+    lit_int(i64, int_ty),
+    lit_uint(u64, uint_ty),
+    lit_float(str, float_ty),
+    lit_nil,
+    lit_bool(bool),
+}
+
+// NB: If you change this, you'll probably want to change the corresponding
+// type structure in middle/ty.rs as well.
+#[auto_serialize]
+type mt = {ty: @ty, mutbl: mutability};
+
+#[auto_serialize]
+type ty_field_ = {ident: ident, mt: mt};
+
+#[auto_serialize]
+type ty_field = spanned<ty_field_>;
+
+#[auto_serialize]
+type ty_method = {ident: ident, attrs: [attribute],
+                  decl: fn_decl, tps: [ty_param], span: span};
+
+#[auto_serialize]
+enum int_ty { ty_i, ty_char, ty_i8, ty_i16, ty_i32, ty_i64, }
+
+#[auto_serialize]
+enum uint_ty { ty_u, ty_u8, ty_u16, ty_u32, ty_u64, }
+
+#[auto_serialize]
+enum float_ty { ty_f, ty_f32, ty_f64, }
+
+#[auto_serialize]
+type ty = {id: node_id, node: ty_, span: span};
+
+// Not represented directly in the AST, referred to by name through a ty_path.
+#[auto_serialize]
+enum prim_ty {
+    ty_int(int_ty),
+    ty_uint(uint_ty),
+    ty_float(float_ty),
+    ty_str,
+    ty_bool,
+}
+
+#[auto_serialize]
+type region = {id: node_id, node: region_};
+
+#[auto_serialize]
+enum region_ { re_anon, re_named(ident) }
+
+#[auto_serialize]
+enum ty_ {
+    ty_nil,
+    ty_bot, /* bottom type */
+    ty_box(mt),
+    ty_uniq(mt),
+    ty_vec(mt),
+    ty_ptr(mt),
+    ty_rptr(@region, mt),
+    ty_rec([ty_field]),
+    ty_fn(proto, fn_decl),
+    ty_tup([@ty]),
+    ty_path(@path, node_id),
+    ty_constr(@ty, [@ty_constr]),
+    ty_vstore(@ty, vstore),
+    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.
+    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.
+*/
+#[auto_serialize]
+enum constr_arg_general_<T> { carg_base, carg_ident(T), carg_lit(@lit), }
+
+#[auto_serialize]
+type fn_constr_arg = constr_arg_general_<uint>;
+
+#[auto_serialize]
+type sp_constr_arg<T> = spanned<constr_arg_general_<T>>;
+
+#[auto_serialize]
+type ty_constr_arg = sp_constr_arg<@path>;
+
+#[auto_serialize]
+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)
+
+#[auto_serialize]
+type constr_general_<ARG, ID> =
+    {path: @path, args: [@sp_constr_arg<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.
+#[auto_serialize]
+type constr_general<ARG> = spanned<constr_general_<ARG, node_id>>;
+
+#[auto_serialize]
+type constr_ = constr_general_<uint, node_id>;
+
+#[auto_serialize]
+type constr = spanned<constr_general_<uint, node_id>>;
+
+#[auto_serialize]
+type ty_constr_ = constr_general_<@path, node_id>;
+
+#[auto_serialize]
+type ty_constr = spanned<ty_constr_>;
+
+/* The parser generates ast::constrs; resolve generates
+ a mapping from each function to a list of ty::constr_defs,
+ corresponding to these. */
+#[auto_serialize]
+type arg = {mode: mode, ty: @ty, ident: ident, id: node_id};
+
+#[auto_serialize]
+type fn_decl =
+    {inputs: [arg],
+     output: @ty,
+     purity: purity,
+     cf: ret_style,
+     constraints: [@constr]};
+
+#[auto_serialize]
+enum purity {
+    pure_fn, // declared with "pure fn"
+    unsafe_fn, // declared with "unsafe fn"
+    impure_fn, // declared with "fn"
+    crust_fn, // declared with "crust fn"
+}
+
+#[auto_serialize]
+enum ret_style {
+    noreturn, // functions with return type _|_ that always
+              // raise an error or exit (i.e. never return to the caller)
+    return_val, // everything else
+}
+
+#[auto_serialize]
+type method = {ident: ident, attrs: [attribute],
+               tps: [ty_param], decl: fn_decl, body: blk,
+               id: node_id, span: span, self_id: node_id,
+               vis: visibility};  // always public, unless it's a
+                                  // class method
+
+#[auto_serialize]
+type _mod = {view_items: [@view_item], items: [@item]};
+
+#[auto_serialize]
+enum native_abi {
+    native_abi_rust_intrinsic,
+    native_abi_cdecl,
+    native_abi_stdcall,
+}
+
+#[auto_serialize]
+type native_mod =
+    {view_items: [@view_item],
+     items: [@native_item]};
+
+#[auto_serialize]
+type variant_arg = {ty: @ty, id: node_id};
+
+#[auto_serialize]
+type variant_ = {name: ident, attrs: [attribute], args: [variant_arg],
+                 id: node_id, disr_expr: option<@expr>, vis: visibility};
+
+#[auto_serialize]
+type variant = spanned<variant_>;
+
+#[auto_serialize]
+type path_list_ident_ = {name: ident, id: node_id};
+
+#[auto_serialize]
+type path_list_ident = spanned<path_list_ident_>;
+
+#[auto_serialize]
+type view_path = spanned<view_path_>;
+
+#[auto_serialize]
+enum view_path_ {
+
+    // quux = foo::bar::baz
+    //
+    // or just
+    //
+    // foo::bar::baz  (with 'baz =' implicitly on the left)
+    view_path_simple(ident, @path, node_id),
+
+    // foo::bar::*
+    view_path_glob(@path, node_id),
+
+    // foo::bar::{a,b,c}
+    view_path_list(@path, [path_list_ident], node_id)
+}
+
+#[auto_serialize]
+type view_item = {node: view_item_, attrs: [attribute],
+                  vis: visibility, span: span};
+
+#[auto_serialize]
+enum view_item_ {
+    view_item_use(ident, [@meta_item], node_id),
+    view_item_import([@view_path]),
+    view_item_export([@view_path])
+}
+
+// Meta-data associated with an item
+#[auto_serialize]
+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.
+#[auto_serialize]
+enum attr_style { attr_outer, attr_inner, }
+
+#[auto_serialize]
+type attribute_ = {style: attr_style, value: meta_item};
+
+/*
+  iface_refs appear in both impls and in classes that implement ifaces.
+  resolve maps each iface_ref's id to its defining iface.
+ */
+#[auto_serialize]
+type iface_ref = {path: @path, id: node_id};
+
+#[auto_serialize]
+enum visibility { public, private }
+
+#[auto_serialize]
+type item = {ident: ident, attrs: [attribute],
+             id: node_id, node: item_,
+             vis: visibility, span: span};
+
+#[auto_serialize]
+enum region_param {
+    rp_none,
+    rp_self
+}
+
+#[auto_serialize]
+enum item_ {
+    item_const(@ty, @expr),
+    item_fn(fn_decl, [ty_param], blk),
+    item_mod(_mod),
+    item_native_mod(native_mod),
+    item_ty(@ty, [ty_param], region_param),
+    item_enum([variant], [ty_param], region_param),
+    item_res(fn_decl /* dtor */, [ty_param], blk /* dtor body */,
+             node_id /* dtor id */, node_id /* ctor id */,
+             region_param),
+    item_class([ty_param], /* ty params for class */
+               [@iface_ref],   /* ifaces this class implements */
+               [@class_member], /* methods, etc. */
+                               /* (not including ctor or dtor) */
+               class_ctor,
+               /* dtor is optional */
+               option<class_dtor>,
+               region_param
+               ),
+    item_iface([ty_param], region_param, [ty_method]),
+    item_impl([ty_param], region_param, option<@iface_ref> /* iface */,
+              @ty /* self */, [@method]),
+}
+
+#[auto_serialize]
+type class_member = spanned<class_member_>;
+
+#[auto_serialize]
+enum class_member_ {
+    instance_var(ident, @ty, class_mutability, node_id, visibility),
+    class_method(@method)
+}
+
+#[auto_serialize]
+enum class_mutability { class_mutable, class_immutable }
+
+#[auto_serialize]
+type class_ctor = spanned<class_ctor_>;
+
+#[auto_serialize]
+type class_ctor_ = {id: node_id,
+                    self_id: node_id,
+                    dec: fn_decl,
+                    body: blk};
+
+#[auto_serialize]
+type class_dtor = spanned<class_dtor_>;
+
+#[auto_serialize]
+type class_dtor_ = {id: node_id,
+                    self_id: node_id,
+                    body: blk};
+
+#[auto_serialize]
+type native_item =
+    {ident: ident,
+     attrs: [attribute],
+     node: native_item_,
+     id: node_id,
+     span: span};
+
+#[auto_serialize]
+enum native_item_ {
+    native_item_fn(fn_decl, [ty_param]),
+}
+
+// The data we save and restore about an inlined item or method.  This is not
+// part of the AST that we parse from a file, but it becomes part of the tree
+// that we trans.
+#[auto_serialize]
+enum inlined_item {
+    ii_item(@item),
+    ii_method(def_id /* impl id */, @method),
+    ii_native(@native_item),
+    ii_ctor(class_ctor, ident, [ty_param], def_id /* parent id */)
+}
+
+//
+// Local Variables:
+// mode: rust
+// fill-column: 78;
+// indent-tabs-mode: nil
+// c-basic-offset: 4
+// buffer-file-coding-system: utf-8-unix
+// End:
+//