// 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, _v: span) { } fn deserialize_span(_d: D) -> span { ast_util::dummy_sp() } #[auto_serialize] type spanned = {node: T, span: span}; #[auto_serialize] type ident = str; // Functions may or may not have names. #[auto_serialize] type fn_ident = option; #[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_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; 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; #[auto_serialize] type meta_item = spanned; #[auto_serialize] enum meta_item_ { meta_word(ident), meta_list(ident, [@meta_item]), meta_name_value(ident, lit), } #[auto_serialize] type blk = spanned; #[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), // [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, lsl, lsr, asr, 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 { 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; #[auto_serialize] type stmt = spanned; #[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, id: node_id}; #[auto_serialize] type local = spanned; #[auto_serialize] type decl = spanned; #[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; #[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), expr_do_while(blk, @expr), /* 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), // 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_be(@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, name: ident, // Currently, can only capture a local var. span: span }; #[auto_serialize] type capture_clause = { copies: [@capture_item], moves: [@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; #[auto_serialize] type mac_arg = option<@expr>; #[auto_serialize] type mac_body_ = {span: span}; #[auto_serialize] type mac_body = option; #[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; #[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; #[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), re_static } #[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_ { carg_base, carg_ident(T), carg_lit(@lit), } #[auto_serialize] type fn_constr_arg = constr_arg_general_; #[auto_serialize] type sp_constr_arg = spanned>; #[auto_serialize] type ty_constr_arg = sp_constr_arg<@path>; #[auto_serialize] type constr_arg = spanned; // 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_ = {path: @path, args: [@spanned>], 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 = spanned>; #[auto_serialize] type constr_ = constr_general_; #[auto_serialize] type constr = spanned>; #[auto_serialize] type ty_constr_ = constr_general_<@path, node_id>; #[auto_serialize] type ty_constr = spanned; /* 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, privacy: privacy}; // privacy is 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>}; #[auto_serialize] type variant = spanned; #[auto_serialize] type path_list_ident_ = {name: ident, id: node_id}; #[auto_serialize] type path_list_ident = spanned; #[auto_serialize] type view_path = spanned; #[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 = spanned; #[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; // 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] type item = {ident: ident, attrs: [attribute], id: node_id, node: item_, 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) */ class_ctor, 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; #[auto_serialize] enum class_member_ { instance_var(ident, @ty, class_mutability, node_id, privacy), class_method(@method) } #[auto_serialize] enum class_mutability { class_mutable, class_immutable } #[auto_serialize] enum privacy { priv, pub } #[auto_serialize] type class_ctor = spanned; #[auto_serialize] type class_ctor_ = {id: node_id, self_id: node_id, dec: fn_decl, 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: //