2012-02-09 15:30:19 +01:00
|
|
|
import std::{ufind, map, smallintmap};
|
2012-03-16 15:14:37 -07:00
|
|
|
import result::result;
|
2011-05-12 17:24:54 +02:00
|
|
|
import std::map::hashmap;
|
|
|
|
import driver::session;
|
2012-01-12 17:59:49 +01:00
|
|
|
import session::session;
|
2011-07-05 11:48:19 +02:00
|
|
|
import syntax::ast;
|
2011-07-19 17:52:34 -07:00
|
|
|
import syntax::ast::*;
|
2011-08-21 21:44:41 -07:00
|
|
|
import syntax::ast_util;
|
2012-03-21 12:42:34 -07:00
|
|
|
import syntax::ast_util::{is_local, split_class_items};
|
2011-07-05 11:48:19 +02:00
|
|
|
import syntax::codemap::span;
|
2011-07-07 22:18:38 -07:00
|
|
|
import metadata::csearch;
|
2011-06-15 15:14:30 -07:00
|
|
|
import util::common::*;
|
2012-03-15 19:19:32 -07:00
|
|
|
import util::ppaux::region_to_str;
|
2011-07-05 11:48:19 +02:00
|
|
|
import util::ppaux::ty_to_str;
|
2011-07-19 17:52:34 -07:00
|
|
|
import util::ppaux::ty_constr_to_str;
|
|
|
|
import syntax::print::pprust::*;
|
2011-05-16 13:58:13 -07:00
|
|
|
|
2012-03-28 13:42:33 -07:00
|
|
|
export is_instantiable;
|
2011-06-19 22:41:21 +02:00
|
|
|
export node_id_to_type;
|
|
|
|
export node_id_to_type_params;
|
2011-06-19 02:34:20 -07:00
|
|
|
export arg;
|
|
|
|
export args_eq;
|
2011-07-21 15:59:41 -07:00
|
|
|
export ast_constr_to_constr;
|
2011-06-19 02:34:20 -07:00
|
|
|
export block_ty;
|
2012-03-03 17:49:23 -08:00
|
|
|
export class_items_as_fields;
|
2011-07-19 17:52:34 -07:00
|
|
|
export constr;
|
2011-06-19 02:34:20 -07:00
|
|
|
export constr_general;
|
|
|
|
export constr_table;
|
|
|
|
export ctxt;
|
|
|
|
export def_has_ty_params;
|
|
|
|
export expr_has_ty_params;
|
|
|
|
export expr_ty;
|
2011-07-29 16:40:23 -07:00
|
|
|
export expr_ty_params_and_ty;
|
2011-11-15 18:15:35 +01:00
|
|
|
export expr_is_lval;
|
2012-03-19 10:19:00 -07:00
|
|
|
export field_ty;
|
2011-06-19 02:34:20 -07:00
|
|
|
export fold_ty;
|
|
|
|
export field;
|
|
|
|
export field_idx;
|
2011-09-13 12:14:30 +02:00
|
|
|
export get_field;
|
2012-01-30 21:00:57 -08:00
|
|
|
export get_fields;
|
2012-03-22 20:06:01 -07:00
|
|
|
export fm_var, fm_general, fm_rptr;
|
2011-06-19 02:34:20 -07:00
|
|
|
export get_element_type;
|
2011-06-24 14:19:58 -07:00
|
|
|
export is_binopable;
|
2011-08-24 17:24:58 -07:00
|
|
|
export is_pred_ty;
|
2012-03-27 22:08:48 -07:00
|
|
|
export lookup_class_field, lookup_class_fields;
|
2012-03-21 12:42:34 -07:00
|
|
|
export lookup_class_method_by_name;
|
2012-03-19 10:19:00 -07:00
|
|
|
export lookup_field_type;
|
2011-06-19 02:34:20 -07:00
|
|
|
export lookup_item_type;
|
2012-03-26 09:59:59 -07:00
|
|
|
export lookup_public_fields;
|
2011-06-19 02:34:20 -07:00
|
|
|
export method;
|
|
|
|
export method_idx;
|
2012-02-09 14:16:12 -08:00
|
|
|
export mk_class;
|
2011-06-19 02:34:20 -07:00
|
|
|
export mk_ctxt;
|
2012-02-10 10:28:35 -08:00
|
|
|
export mk_with_id, type_def_id;
|
2011-06-19 02:34:20 -07:00
|
|
|
export mt;
|
|
|
|
export node_type_table;
|
|
|
|
export pat_ty;
|
|
|
|
export sequence_element_type;
|
|
|
|
export sort_methods;
|
2011-06-19 22:41:21 +02:00
|
|
|
export stmt_node_id;
|
2011-06-19 02:34:20 -07:00
|
|
|
export sty;
|
|
|
|
export substitute_type_params;
|
|
|
|
export t;
|
2011-12-22 14:24:36 +01:00
|
|
|
export new_ty_hash;
|
2012-02-07 21:19:53 -08:00
|
|
|
export enum_variants, substd_enum_variants;
|
2012-01-05 10:57:19 +01:00
|
|
|
export iface_methods, store_iface_methods, impl_iface;
|
2012-01-25 14:34:31 +01:00
|
|
|
export enum_variant_with_id;
|
2011-12-28 17:50:12 +01:00
|
|
|
export ty_param_bounds_and_ty;
|
2012-02-07 11:25:04 +01:00
|
|
|
export ty_bool, mk_bool, type_is_bool;
|
|
|
|
export ty_bot, mk_bot, type_is_bot;
|
|
|
|
export ty_box, mk_box, mk_imm_box, type_is_box, type_is_boxed;
|
|
|
|
export ty_constr, mk_constr;
|
|
|
|
export ty_opaque_closure_ptr, mk_opaque_closure_ptr;
|
|
|
|
export ty_opaque_box, mk_opaque_box;
|
2011-07-19 20:15:27 -07:00
|
|
|
export ty_constr_arg;
|
2012-02-07 11:25:04 +01:00
|
|
|
export ty_float, mk_float, mk_mach_float, type_is_fp;
|
|
|
|
export ty_fn, fn_ty, mk_fn;
|
|
|
|
export ty_fn_proto, ty_fn_ret, ty_fn_ret_style;
|
|
|
|
export ty_int, mk_int, mk_mach_int, mk_char;
|
|
|
|
export ty_str, mk_str, type_is_str;
|
|
|
|
export ty_vec, mk_vec, type_is_vec;
|
|
|
|
export ty_nil, mk_nil, type_is_nil;
|
|
|
|
export ty_iface, mk_iface;
|
|
|
|
export ty_res, mk_res;
|
|
|
|
export ty_param, mk_param;
|
2012-03-21 15:42:20 +01:00
|
|
|
export ty_ptr, mk_ptr, mk_mut_ptr, mk_imm_ptr, mk_nil_ptr, type_is_unsafe_ptr;
|
2012-03-08 14:05:16 -08:00
|
|
|
export ty_rptr, mk_rptr;
|
2012-02-07 11:25:04 +01:00
|
|
|
export ty_rec, mk_rec;
|
|
|
|
export ty_enum, mk_enum, type_is_enum;
|
|
|
|
export ty_tup, mk_tup;
|
|
|
|
export ty_type, mk_type;
|
|
|
|
export ty_uint, mk_uint, mk_mach_uint;
|
|
|
|
export ty_uniq, mk_uniq, mk_imm_uniq, type_is_unique_box;
|
|
|
|
export ty_var, mk_var;
|
|
|
|
export ty_self, mk_self;
|
2012-03-23 11:37:10 -07:00
|
|
|
export region, re_block, re_param, re_var;
|
2012-03-12 13:24:37 -07:00
|
|
|
export get, type_has_params, type_has_vars, type_has_rptrs, type_id;
|
2011-06-19 02:34:20 -07:00
|
|
|
export ty_var_id;
|
2012-03-08 16:10:25 +01:00
|
|
|
export ty_to_def_id;
|
2011-06-19 02:34:20 -07:00
|
|
|
export ty_fn_args;
|
2011-07-19 17:52:34 -07:00
|
|
|
export type_constr;
|
2011-12-28 18:11:33 +01:00
|
|
|
export kind, kind_sendable, kind_copyable, kind_noncopyable;
|
|
|
|
export kind_can_be_copied, kind_can_be_sent, proto_kind, kind_lteq, type_kind;
|
2011-06-19 02:34:20 -07:00
|
|
|
export type_err;
|
|
|
|
export type_err_to_str;
|
2011-07-29 12:53:58 +02:00
|
|
|
export type_needs_drop;
|
2011-09-07 15:13:19 +02:00
|
|
|
export type_allows_implicit_copy;
|
2011-06-19 02:34:20 -07:00
|
|
|
export type_is_integral;
|
2011-12-02 13:42:51 +01:00
|
|
|
export type_is_numeric;
|
2011-08-04 10:46:10 -07:00
|
|
|
export type_is_pod;
|
2011-06-19 02:34:20 -07:00
|
|
|
export type_is_scalar;
|
2011-10-06 12:26:12 +02:00
|
|
|
export type_is_immediate;
|
2011-06-19 02:34:20 -07:00
|
|
|
export type_is_sequence;
|
|
|
|
export type_is_signed;
|
|
|
|
export type_is_structural;
|
2011-06-25 12:22:50 +02:00
|
|
|
export type_is_copyable;
|
2011-06-19 02:34:20 -07:00
|
|
|
export type_is_tup_like;
|
2011-08-16 12:38:42 -07:00
|
|
|
export type_is_unique;
|
2012-01-10 18:33:26 -07:00
|
|
|
export type_is_c_like_enum;
|
2012-01-27 16:01:04 +01:00
|
|
|
export type_structurally_contains;
|
2011-09-07 15:13:19 +02:00
|
|
|
export type_structurally_contains_uniques;
|
2011-07-01 01:48:32 -07:00
|
|
|
export type_autoderef;
|
2011-06-19 02:34:20 -07:00
|
|
|
export type_param;
|
2012-03-23 17:52:20 -07:00
|
|
|
export type_needs_unwind_cleanup;
|
2012-02-03 17:10:29 -08:00
|
|
|
export canon_mode;
|
2012-02-02 16:50:17 -08:00
|
|
|
export resolved_mode;
|
|
|
|
export arg_mode;
|
|
|
|
export unify_mode;
|
|
|
|
export set_default_mode;
|
2011-06-19 02:34:20 -07:00
|
|
|
export unify;
|
|
|
|
export variant_info;
|
2012-03-12 16:31:22 +01:00
|
|
|
export walk_ty, maybe_walk_ty;
|
2012-03-10 20:35:41 -08:00
|
|
|
export occurs_check;
|
2011-12-15 11:06:48 -08:00
|
|
|
export closure_kind;
|
2012-01-10 06:49:15 -08:00
|
|
|
export ck_block;
|
|
|
|
export ck_box;
|
|
|
|
export ck_uniq;
|
2012-01-02 12:00:40 +01:00
|
|
|
export param_bound, param_bounds, bound_copy, bound_send, bound_iface;
|
2011-12-28 17:50:12 +01:00
|
|
|
export param_bounds_to_kind;
|
2012-02-02 16:50:17 -08:00
|
|
|
export default_arg_mode_for_ty;
|
2012-02-10 06:01:32 -08:00
|
|
|
export item_path;
|
2012-02-14 15:21:53 -08:00
|
|
|
export item_path_str;
|
2012-03-13 14:12:44 -07:00
|
|
|
export ast_ty_to_ty_cache_entry;
|
2012-03-23 11:37:10 -07:00
|
|
|
export atttce_unresolved, atttce_resolved;
|
2012-03-22 20:06:01 -07:00
|
|
|
export mach_sty;
|
2010-12-21 12:13:51 -08:00
|
|
|
|
2011-06-15 11:19:50 -07:00
|
|
|
// Data types
|
2011-05-09 12:27:03 -07:00
|
|
|
|
2012-02-02 16:50:17 -08:00
|
|
|
// Note: after typeck, you should use resolved_mode() to convert this mode
|
|
|
|
// into an rmode, which will take into account the results of mode inference.
|
2012-01-30 21:00:57 -08:00
|
|
|
type arg = {mode: ast::mode, ty: t};
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2011-07-27 14:19:39 +02:00
|
|
|
type field = {ident: ast::ident, mt: mt};
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2012-01-02 12:00:40 +01:00
|
|
|
type param_bounds = @[param_bound];
|
|
|
|
|
2012-02-13 18:56:09 +01:00
|
|
|
type method = {ident: ast::ident,
|
|
|
|
tps: @[param_bounds],
|
|
|
|
fty: fn_ty,
|
2012-03-28 18:50:33 -07:00
|
|
|
purity: ast::purity,
|
|
|
|
privacy: ast::privacy};
|
2010-12-21 12:13:51 -08:00
|
|
|
|
2011-08-12 07:15:18 -07:00
|
|
|
type constr_table = hashmap<ast::node_id, [constr]>;
|
2011-05-19 17:21:21 -07:00
|
|
|
|
2012-02-15 11:25:39 -08:00
|
|
|
type mt = {ty: t, mutbl: ast::mutability};
|
2011-03-17 17:39:47 -07:00
|
|
|
|
2012-03-19 10:19:00 -07:00
|
|
|
type field_ty = {
|
2012-03-06 08:02:13 -08:00
|
|
|
ident: ident,
|
2012-03-19 10:19:00 -07:00
|
|
|
id: def_id,
|
2012-03-27 22:08:48 -07:00
|
|
|
privacy: ast::privacy,
|
|
|
|
mutability: ast::class_mutability
|
2012-03-06 08:02:13 -08:00
|
|
|
};
|
|
|
|
|
2011-04-25 12:15:55 -07:00
|
|
|
// Contains information needed to resolve types and (in the future) look up
|
|
|
|
// the types of AST nodes.
|
2012-02-03 15:15:28 +01:00
|
|
|
type creader_cache = hashmap<{cnum: int, pos: uint, len: uint}, t>;
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2012-02-10 10:28:35 -08:00
|
|
|
type intern_key = {struct: sty, o_def_id: option<ast::def_id>};
|
|
|
|
|
2012-03-13 14:12:44 -07:00
|
|
|
enum ast_ty_to_ty_cache_entry {
|
|
|
|
atttce_unresolved, /* not resolved yet */
|
2012-03-23 11:37:10 -07:00
|
|
|
atttce_resolved(t) /* resolved to a type, irrespective of region */
|
2012-03-13 14:12:44 -07:00
|
|
|
}
|
|
|
|
|
2012-03-23 17:28:24 -07:00
|
|
|
enum borrowing {
|
|
|
|
bo_none,
|
|
|
|
bo_box(@borrowing),
|
|
|
|
bo_uniq(@borrowing)
|
|
|
|
}
|
|
|
|
|
2011-06-15 11:19:50 -07:00
|
|
|
type ctxt =
|
2012-02-10 10:28:35 -08:00
|
|
|
@{interner: hashmap<intern_key, t_box>,
|
2012-03-26 18:35:18 -07:00
|
|
|
mut next_id: uint,
|
2011-07-27 14:19:39 +02:00
|
|
|
sess: session::session,
|
|
|
|
def_map: resolve::def_map,
|
2012-03-09 16:39:54 -08:00
|
|
|
region_map: @middle::region::region_map,
|
2012-03-29 19:25:50 -07:00
|
|
|
|
|
|
|
// Stores the types for various nodes in the AST. Note that this table
|
|
|
|
// is not guaranteed to be populated until after typeck. See
|
|
|
|
// typeck::fn_ctxt for details.
|
2011-07-27 14:19:39 +02:00
|
|
|
node_types: node_type_table,
|
2012-03-29 19:25:50 -07:00
|
|
|
|
|
|
|
// Stores the type parameters which were substituted to obtain the type
|
|
|
|
// of this node. This only applies to nodes that refer to entities
|
|
|
|
// parameterized by type parameters, such as generic fns, types, or
|
|
|
|
// other items.
|
2012-01-30 17:28:30 +01:00
|
|
|
node_type_substs: hashmap<node_id, [t]>,
|
2012-03-29 19:25:50 -07:00
|
|
|
|
2011-07-27 14:19:39 +02:00
|
|
|
items: ast_map::map,
|
|
|
|
freevars: freevars::freevar_map,
|
|
|
|
tcache: type_cache,
|
|
|
|
rcache: creader_cache,
|
2011-09-02 15:34:58 -07:00
|
|
|
short_names_cache: hashmap<t, @str>,
|
2011-11-22 13:16:23 +01:00
|
|
|
needs_drop_cache: hashmap<t, bool>,
|
2012-03-23 17:52:20 -07:00
|
|
|
needs_unwind_cleanup_cache: hashmap<t, bool>,
|
2011-12-28 18:11:33 +01:00
|
|
|
kind_cache: hashmap<t, kind>,
|
2012-03-13 14:12:44 -07:00
|
|
|
ast_ty_to_ty_cache: hashmap<@ast::ty, ast_ty_to_ty_cache_entry>,
|
2012-01-25 14:34:31 +01:00
|
|
|
enum_var_cache: hashmap<def_id, @[variant_info]>,
|
2011-12-28 17:50:12 +01:00
|
|
|
iface_method_cache: hashmap<def_id, @[method]>,
|
2012-02-02 16:50:17 -08:00
|
|
|
ty_param_bounds: hashmap<ast::node_id, param_bounds>,
|
2012-03-23 17:28:24 -07:00
|
|
|
inferred_modes: hashmap<ast::node_id, ast::mode>,
|
|
|
|
borrowings: hashmap<ast::node_id, borrowing>};
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
type t_box = @{struct: sty,
|
|
|
|
id: uint,
|
|
|
|
has_params: bool,
|
|
|
|
has_vars: bool,
|
2012-03-12 13:24:37 -07:00
|
|
|
has_rptrs: bool,
|
2012-02-10 10:28:35 -08:00
|
|
|
o_def_id: option<ast::def_id>};
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
// To reduce refcounting cost, we're representing types as unsafe pointers
|
|
|
|
// throughout the compiler. These are simply casted t_box values. Use ty::get
|
|
|
|
// to cast them back to a box. (Without the cast, compiler performance suffers
|
|
|
|
// ~15%.) This does mean that a t value relies on the ctxt to keep its box
|
|
|
|
// alive, and using ty::get is unsafe when the ctxt is no longer alive.
|
|
|
|
enum t_opaque {}
|
|
|
|
type t = *t_opaque;
|
2011-05-11 17:05:39 -07:00
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
pure fn get(t: t) -> t_box unsafe {
|
|
|
|
let t2 = unsafe::reinterpret_cast::<t, t_box>(t);
|
|
|
|
let t3 = t2;
|
2012-03-20 15:12:30 -07:00
|
|
|
unsafe::forget(t2);
|
2012-02-03 15:15:28 +01:00
|
|
|
t3
|
|
|
|
}
|
|
|
|
|
|
|
|
fn type_has_params(t: t) -> bool { get(t).has_params }
|
|
|
|
fn type_has_vars(t: t) -> bool { get(t).has_vars }
|
2012-03-12 13:24:37 -07:00
|
|
|
fn type_has_rptrs(t: t) -> bool { get(t).has_rptrs }
|
2012-02-10 10:28:35 -08:00
|
|
|
fn type_def_id(t: t) -> option<ast::def_id> { get(t).o_def_id }
|
2012-02-03 15:15:28 +01:00
|
|
|
fn type_id(t: t) -> uint { get(t).id }
|
2011-04-20 10:51:41 -07:00
|
|
|
|
2012-01-19 14:24:03 -08:00
|
|
|
enum closure_kind {
|
2012-01-19 17:56:05 -08:00
|
|
|
ck_block,
|
|
|
|
ck_box,
|
|
|
|
ck_uniq,
|
2011-12-15 11:06:48 -08:00
|
|
|
}
|
|
|
|
|
2011-12-23 16:09:52 +01:00
|
|
|
type fn_ty = {proto: ast::proto,
|
|
|
|
inputs: [arg],
|
|
|
|
output: t,
|
|
|
|
ret_style: ret_style,
|
|
|
|
constraints: [@constr]};
|
|
|
|
|
2012-03-08 14:05:16 -08:00
|
|
|
enum region {
|
2012-03-23 11:37:10 -07:00
|
|
|
// The region of a block.
|
2012-03-13 15:52:45 -07:00
|
|
|
re_block(node_id),
|
2012-03-23 11:37:10 -07:00
|
|
|
// The self region. Only valid inside classes and typeclass
|
|
|
|
// implementations.
|
|
|
|
re_self,
|
|
|
|
// The inferred region, which also corresponds to &self in typedefs.
|
|
|
|
re_inferred,
|
2012-03-21 12:21:06 -07:00
|
|
|
|
2012-03-23 11:37:10 -07:00
|
|
|
// A region parameter.
|
|
|
|
re_param(uint),
|
|
|
|
|
|
|
|
// A region variable.
|
|
|
|
re_var(uint)
|
2012-03-08 14:05:16 -08:00
|
|
|
}
|
|
|
|
|
2010-12-21 12:13:51 -08:00
|
|
|
// NB: If you change this, you'll probably want to change the corresponding
|
2011-05-12 17:24:54 +02:00
|
|
|
// AST structure in front/ast::rs as well.
|
2012-01-19 14:24:03 -08:00
|
|
|
enum sty {
|
2012-01-19 17:56:05 -08:00
|
|
|
ty_nil,
|
|
|
|
ty_bot,
|
|
|
|
ty_bool,
|
|
|
|
ty_int(ast::int_ty),
|
|
|
|
ty_uint(ast::uint_ty),
|
|
|
|
ty_float(ast::float_ty),
|
|
|
|
ty_str,
|
2012-01-25 14:34:31 +01:00
|
|
|
ty_enum(def_id, [t]),
|
2012-01-19 17:56:05 -08:00
|
|
|
ty_box(mt),
|
|
|
|
ty_uniq(mt),
|
|
|
|
ty_vec(mt),
|
|
|
|
ty_ptr(mt),
|
2012-03-08 14:05:16 -08:00
|
|
|
ty_rptr(region, mt),
|
2012-01-19 17:56:05 -08:00
|
|
|
ty_rec([field]),
|
|
|
|
ty_fn(fn_ty),
|
|
|
|
ty_iface(def_id, [t]),
|
2012-02-09 14:16:12 -08:00
|
|
|
ty_class(def_id, [t]),
|
2012-01-19 17:56:05 -08:00
|
|
|
ty_res(def_id, t, [t]),
|
|
|
|
ty_tup([t]),
|
|
|
|
|
2012-01-30 11:52:34 +01:00
|
|
|
ty_var(int), // type variable during typechecking
|
|
|
|
ty_param(uint, def_id), // type parameter
|
|
|
|
ty_self([t]), // interface method self type
|
2012-01-19 17:56:05 -08:00
|
|
|
|
|
|
|
ty_type, // type_desc*
|
2012-03-08 13:01:08 -08:00
|
|
|
ty_opaque_box, // used by monomorphizer to represent any @ box
|
2012-01-19 17:56:05 -08:00
|
|
|
ty_constr(t, [@type_constr]),
|
|
|
|
ty_opaque_closure_ptr(closure_kind), // ptr to env for fn, fn@, fn~
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
|
2011-07-19 17:52:34 -07:00
|
|
|
// In the middle end, constraints have a def_id attached, referring
|
|
|
|
// to the definition of the operator in the constraint.
|
2011-08-12 06:36:51 -07:00
|
|
|
type constr_general<ARG> = spanned<constr_general_<ARG, def_id>>;
|
2011-11-30 13:38:38 +01:00
|
|
|
type type_constr = constr_general<@path>;
|
2011-08-12 07:15:18 -07:00
|
|
|
type constr = constr_general<uint>;
|
2010-12-21 17:47:13 -08:00
|
|
|
|
2011-06-15 11:19:50 -07:00
|
|
|
// Data structures used in type unification
|
2012-01-19 14:24:03 -08:00
|
|
|
enum type_err {
|
2012-01-19 17:56:05 -08:00
|
|
|
terr_mismatch,
|
|
|
|
terr_ret_style_mismatch(ast::ret_style, ast::ret_style),
|
2012-03-22 20:06:01 -07:00
|
|
|
terr_mutability,
|
|
|
|
terr_proto_mismatch(ast::proto, ast::proto),
|
2012-01-19 17:56:05 -08:00
|
|
|
terr_box_mutability,
|
2012-02-21 17:02:02 +01:00
|
|
|
terr_ptr_mutability,
|
2012-03-08 15:54:36 -08:00
|
|
|
terr_ref_mutability,
|
2012-01-19 17:56:05 -08:00
|
|
|
terr_vec_mutability,
|
|
|
|
terr_tuple_size(uint, uint),
|
2012-03-13 20:46:16 -04:00
|
|
|
terr_ty_param_size(uint, uint),
|
2012-01-19 17:56:05 -08:00
|
|
|
terr_record_size(uint, uint),
|
|
|
|
terr_record_mutability,
|
|
|
|
terr_record_fields(ast::ident, ast::ident),
|
|
|
|
terr_arg_count,
|
|
|
|
terr_mode_mismatch(mode, mode),
|
|
|
|
terr_constr_len(uint, uint),
|
|
|
|
terr_constr_mismatch(@type_constr, @type_constr),
|
2012-03-15 19:19:32 -07:00
|
|
|
terr_regions_differ(bool /* variance */, region, region),
|
2010-12-21 17:47:13 -08:00
|
|
|
}
|
|
|
|
|
2012-01-19 14:24:03 -08:00
|
|
|
enum param_bound {
|
2012-01-19 17:56:05 -08:00
|
|
|
bound_copy,
|
|
|
|
bound_send,
|
|
|
|
bound_iface(t),
|
2011-12-28 17:50:12 +01:00
|
|
|
}
|
|
|
|
|
2012-01-02 12:00:40 +01:00
|
|
|
fn param_bounds_to_kind(bounds: param_bounds) -> kind {
|
2012-03-15 09:47:03 -04:00
|
|
|
let mut kind = kind_noncopyable;
|
2011-12-28 17:50:12 +01:00
|
|
|
for bound in *bounds {
|
|
|
|
alt bound {
|
2012-01-18 22:37:22 -08:00
|
|
|
bound_copy {
|
2011-12-28 18:11:33 +01:00
|
|
|
if kind != kind_sendable { kind = kind_copyable; }
|
2011-12-28 17:50:12 +01:00
|
|
|
}
|
2012-01-18 22:37:22 -08:00
|
|
|
bound_send { kind = kind_sendable; }
|
2011-12-28 17:50:12 +01:00
|
|
|
_ {}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
kind
|
|
|
|
}
|
|
|
|
|
2012-01-02 12:09:26 +01:00
|
|
|
type ty_param_bounds_and_ty = {bounds: @[param_bounds], ty: t};
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2011-12-28 17:50:12 +01:00
|
|
|
type type_cache = hashmap<ast::def_id, ty_param_bounds_and_ty>;
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2012-01-30 17:28:30 +01:00
|
|
|
type node_type_table = @smallintmap::smallintmap<t>;
|
2011-05-12 16:36:47 -07:00
|
|
|
|
2011-04-29 15:26:28 +00:00
|
|
|
fn mk_rcache() -> creader_cache {
|
2011-07-27 14:19:39 +02:00
|
|
|
type val = {cnum: int, pos: uint, len: uint};
|
2011-09-12 11:27:30 +02:00
|
|
|
fn hash_cache_entry(k: val) -> uint {
|
2011-07-26 14:06:02 +02:00
|
|
|
ret (k.cnum as uint) + k.pos + k.len;
|
2011-04-29 15:26:28 +00:00
|
|
|
}
|
2011-09-12 11:27:30 +02:00
|
|
|
fn eq_cache_entries(a: val, b: val) -> bool {
|
2011-07-26 14:06:02 +02:00
|
|
|
ret a.cnum == b.cnum && a.pos == b.pos && a.len == b.len;
|
2011-04-29 15:26:28 +00:00
|
|
|
}
|
2012-03-14 12:07:23 -07:00
|
|
|
ret map::hashmap(hash_cache_entry, eq_cache_entries);
|
2011-04-29 15:26:28 +00:00
|
|
|
}
|
2011-04-25 12:15:55 -07:00
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn new_ty_hash<V: copy>() -> map::hashmap<t, V> {
|
2012-03-14 12:07:23 -07:00
|
|
|
map::hashmap({|&&t: t| type_id(t)},
|
2012-02-03 15:15:28 +01:00
|
|
|
{|&&a: t, &&b: t| type_id(a) == type_id(b)})
|
|
|
|
}
|
2011-07-19 17:52:34 -07:00
|
|
|
|
2011-12-20 04:17:47 +08:00
|
|
|
fn mk_ctxt(s: session::session, dm: resolve::def_map, amap: ast_map::map,
|
2012-03-09 16:39:54 -08:00
|
|
|
freevars: freevars::freevar_map,
|
|
|
|
region_map: @middle::region::region_map) -> ctxt {
|
2012-03-14 12:07:23 -07:00
|
|
|
let interner = map::hashmap({|&&k: intern_key|
|
2012-02-10 10:28:35 -08:00
|
|
|
hash_type_structure(k.struct) +
|
2012-04-02 15:34:49 -07:00
|
|
|
option::with_option(k.o_def_id, 0u, ast_util::hash_def_id)
|
2012-02-03 15:15:28 +01:00
|
|
|
}, {|&&a, &&b| a == b});
|
|
|
|
@{interner: interner,
|
2012-03-26 18:35:18 -07:00
|
|
|
mut next_id: 0u,
|
2012-02-03 15:15:28 +01:00
|
|
|
sess: s,
|
|
|
|
def_map: dm,
|
2012-03-09 16:39:54 -08:00
|
|
|
region_map: region_map,
|
2012-02-03 15:15:28 +01:00
|
|
|
node_types: @smallintmap::mk(),
|
2012-03-14 12:07:23 -07:00
|
|
|
node_type_substs: map::int_hash(),
|
2012-02-03 15:15:28 +01:00
|
|
|
items: amap,
|
|
|
|
freevars: freevars,
|
|
|
|
tcache: new_def_hash(),
|
|
|
|
rcache: mk_rcache(),
|
|
|
|
short_names_cache: new_ty_hash(),
|
|
|
|
needs_drop_cache: new_ty_hash(),
|
2012-03-23 17:52:20 -07:00
|
|
|
needs_unwind_cleanup_cache: new_ty_hash(),
|
2012-02-03 15:15:28 +01:00
|
|
|
kind_cache: new_ty_hash(),
|
2012-03-14 12:07:23 -07:00
|
|
|
ast_ty_to_ty_cache: map::hashmap(
|
2012-03-12 15:52:30 -07:00
|
|
|
ast_util::hash_ty, ast_util::eq_ty),
|
2012-02-03 15:15:28 +01:00
|
|
|
enum_var_cache: new_def_hash(),
|
|
|
|
iface_method_cache: new_def_hash(),
|
2012-03-14 12:07:23 -07:00
|
|
|
ty_param_bounds: map::int_hash(),
|
2012-03-23 17:28:24 -07:00
|
|
|
inferred_modes: map::int_hash(),
|
|
|
|
borrowings: map::int_hash()}
|
2011-04-29 15:26:28 +00:00
|
|
|
}
|
2011-05-11 17:05:39 -07:00
|
|
|
|
|
|
|
|
2011-04-20 10:51:41 -07:00
|
|
|
// Type constructors
|
2012-02-10 10:28:35 -08:00
|
|
|
fn mk_t(cx: ctxt, st: sty) -> t { mk_t_with_id(cx, st, none) }
|
2012-02-03 15:15:28 +01:00
|
|
|
|
|
|
|
// Interns a type/name combination, stores the resulting box in cx.interner,
|
|
|
|
// and returns the box as cast to an unsafe ptr (see comments for t above).
|
2012-02-10 10:28:35 -08:00
|
|
|
fn mk_t_with_id(cx: ctxt, st: sty, o_def_id: option<ast::def_id>) -> t {
|
|
|
|
let key = {struct: st, o_def_id: o_def_id};
|
2012-02-03 15:15:28 +01:00
|
|
|
alt cx.interner.find(key) {
|
|
|
|
some(t) { unsafe { ret unsafe::reinterpret_cast(t); } }
|
|
|
|
_ {}
|
|
|
|
}
|
2012-03-15 09:47:03 -04:00
|
|
|
let mut has_params = false, has_vars = false, has_rptrs = false;
|
2012-03-12 13:24:37 -07:00
|
|
|
fn derive_flags(&has_params: bool, &has_vars: bool, &has_rptrs: bool,
|
|
|
|
tt: t) {
|
2012-02-03 15:15:28 +01:00
|
|
|
let t = get(tt);
|
|
|
|
has_params |= t.has_params;
|
|
|
|
has_vars |= t.has_vars;
|
2012-03-12 13:24:37 -07:00
|
|
|
has_rptrs |= t.has_rptrs;
|
2011-04-25 16:17:14 -07:00
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
alt st {
|
2012-01-18 22:37:22 -08:00
|
|
|
ty_nil | ty_bot | ty_bool | ty_int(_) | ty_float(_) | ty_uint(_) |
|
2012-03-15 17:42:33 +01:00
|
|
|
ty_str | ty_type | ty_opaque_closure_ptr(_) |
|
2012-02-07 11:25:04 +01:00
|
|
|
ty_opaque_box {}
|
2011-08-19 15:16:48 -07:00
|
|
|
ty_param(_, _) { has_params = true; }
|
2012-01-30 11:52:34 +01:00
|
|
|
ty_var(_) | ty_self(_) { has_vars = true; }
|
2012-02-09 14:16:12 -08:00
|
|
|
ty_enum(_, tys) | ty_iface(_, tys) | ty_class(_, tys) {
|
2012-03-12 13:24:37 -07:00
|
|
|
for tt in tys { derive_flags(has_params, has_vars, has_rptrs, tt); }
|
2012-02-03 15:15:28 +01:00
|
|
|
}
|
2012-03-12 13:24:37 -07:00
|
|
|
ty_box(m) | ty_uniq(m) | ty_vec(m) | ty_ptr(m) {
|
|
|
|
derive_flags(has_params, has_vars, has_rptrs, m.ty);
|
|
|
|
}
|
|
|
|
ty_rptr(_, m) {
|
|
|
|
has_rptrs = true;
|
|
|
|
derive_flags(has_params, has_vars, has_rptrs, m.ty);
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
ty_rec(flds) {
|
2012-03-12 13:24:37 -07:00
|
|
|
for f in flds {
|
|
|
|
derive_flags(has_params, has_vars, has_rptrs, f.mt.ty);
|
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2011-08-15 12:08:05 +02:00
|
|
|
ty_tup(ts) {
|
2012-03-12 13:24:37 -07:00
|
|
|
for tt in ts { derive_flags(has_params, has_vars, has_rptrs, tt); }
|
2011-08-15 11:40:26 +02:00
|
|
|
}
|
2011-12-23 16:09:52 +01:00
|
|
|
ty_fn(f) {
|
2012-03-12 13:24:37 -07:00
|
|
|
for a in f.inputs {
|
|
|
|
derive_flags(has_params, has_vars, has_rptrs, a.ty);
|
|
|
|
}
|
|
|
|
derive_flags(has_params, has_vars, has_rptrs, f.output);
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
ty_res(_, tt, tps) {
|
2012-03-12 13:24:37 -07:00
|
|
|
derive_flags(has_params, has_vars, has_rptrs, tt);
|
|
|
|
for tt in tps { derive_flags(has_params, has_vars, has_rptrs, tt); }
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2012-02-03 15:15:28 +01:00
|
|
|
ty_constr(tt, _) {
|
2012-03-12 13:24:37 -07:00
|
|
|
derive_flags(has_params, has_vars, has_rptrs, tt);
|
2011-12-19 13:52:58 +01:00
|
|
|
}
|
2011-04-25 16:17:14 -07:00
|
|
|
}
|
2012-02-03 15:15:28 +01:00
|
|
|
let t = @{struct: st,
|
|
|
|
id: cx.next_id,
|
|
|
|
has_params: has_params,
|
|
|
|
has_vars: has_vars,
|
2012-03-12 13:24:37 -07:00
|
|
|
has_rptrs: has_rptrs,
|
2012-02-10 10:28:35 -08:00
|
|
|
o_def_id: o_def_id};
|
2012-02-03 15:15:28 +01:00
|
|
|
cx.interner.insert(key, t);
|
|
|
|
cx.next_id += 1u;
|
|
|
|
unsafe { unsafe::reinterpret_cast(t) }
|
2011-05-11 17:05:39 -07:00
|
|
|
}
|
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn mk_nil(cx: ctxt) -> t { mk_t(cx, ty_nil) }
|
2011-04-21 19:30:53 -07:00
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn mk_bot(cx: ctxt) -> t { mk_t(cx, ty_bot) }
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn mk_bool(cx: ctxt) -> t { mk_t(cx, ty_bool) }
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn mk_int(cx: ctxt) -> t { mk_t(cx, ty_int(ast::ty_i)) }
|
2011-04-25 16:17:14 -07:00
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn mk_float(cx: ctxt) -> t { mk_t(cx, ty_float(ast::ty_f)) }
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn mk_uint(cx: ctxt) -> t { mk_t(cx, ty_uint(ast::ty_u)) }
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn mk_mach_int(cx: ctxt, tm: ast::int_ty) -> t { mk_t(cx, ty_int(tm)) }
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn mk_mach_uint(cx: ctxt, tm: ast::uint_ty) -> t { mk_t(cx, ty_uint(tm)) }
|
2011-04-20 10:51:41 -07:00
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn mk_mach_float(cx: ctxt, tm: ast::float_ty) -> t { mk_t(cx, ty_float(tm)) }
|
2011-12-07 21:06:12 +01:00
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn mk_char(cx: ctxt) -> t { mk_t(cx, ty_int(ast::ty_char)) }
|
2011-12-07 21:06:12 +01:00
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn mk_str(cx: ctxt) -> t { mk_t(cx, ty_str) }
|
2011-04-20 11:23:36 -07:00
|
|
|
|
2012-01-25 14:34:31 +01:00
|
|
|
fn mk_enum(cx: ctxt, did: ast::def_id, tys: [t]) -> t {
|
2012-02-03 15:15:28 +01:00
|
|
|
mk_t(cx, ty_enum(did, tys))
|
2011-04-20 18:52:04 -07:00
|
|
|
}
|
2011-04-20 11:23:36 -07:00
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn mk_box(cx: ctxt, tm: mt) -> t { mk_t(cx, ty_box(tm)) }
|
2011-04-20 18:52:04 -07:00
|
|
|
|
2012-02-15 11:25:39 -08:00
|
|
|
fn mk_imm_box(cx: ctxt, ty: t) -> t { mk_box(cx, {ty: ty,
|
|
|
|
mutbl: ast::m_imm}) }
|
2011-09-21 18:54:54 -07:00
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn mk_uniq(cx: ctxt, tm: mt) -> t { mk_t(cx, ty_uniq(tm)) }
|
2011-08-10 17:23:46 -07:00
|
|
|
|
2012-02-15 11:25:39 -08:00
|
|
|
fn mk_imm_uniq(cx: ctxt, ty: t) -> t { mk_uniq(cx, {ty: ty,
|
|
|
|
mutbl: ast::m_imm}) }
|
2011-06-03 15:02:58 -04:00
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn mk_ptr(cx: ctxt, tm: mt) -> t { mk_t(cx, ty_ptr(tm)) }
|
2011-04-20 18:52:04 -07:00
|
|
|
|
2012-03-08 14:05:16 -08:00
|
|
|
fn mk_rptr(cx: ctxt, r: region, tm: mt) -> t { mk_t(cx, ty_rptr(r, tm)) }
|
|
|
|
|
2012-02-15 11:25:39 -08:00
|
|
|
fn mk_mut_ptr(cx: ctxt, ty: t) -> t { mk_ptr(cx, {ty: ty,
|
|
|
|
mutbl: ast::m_mutbl}) }
|
2011-08-02 16:24:38 -07:00
|
|
|
|
2012-03-21 15:42:20 +01:00
|
|
|
fn mk_imm_ptr(cx: ctxt, ty: t) -> t {
|
|
|
|
mk_ptr(cx, {ty: ty, mutbl: ast::m_imm})
|
|
|
|
}
|
|
|
|
|
2012-03-16 12:57:07 +01:00
|
|
|
fn mk_nil_ptr(cx: ctxt) -> t {
|
|
|
|
mk_ptr(cx, {ty: mk_nil(cx), mutbl: ast::m_imm})
|
|
|
|
}
|
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn mk_vec(cx: ctxt, tm: mt) -> t { mk_t(cx, ty_vec(tm)) }
|
2011-06-09 16:23:19 -07:00
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn mk_rec(cx: ctxt, fs: [field]) -> t { mk_t(cx, ty_rec(fs)) }
|
2011-04-20 18:52:04 -07:00
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn mk_constr(cx: ctxt, t: t, cs: [@type_constr]) -> t {
|
2012-02-03 15:15:28 +01:00
|
|
|
mk_t(cx, ty_constr(t, cs))
|
2011-07-19 17:52:34 -07:00
|
|
|
}
|
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn mk_tup(cx: ctxt, ts: [t]) -> t { mk_t(cx, ty_tup(ts)) }
|
2011-08-15 11:40:26 +02:00
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn mk_fn(cx: ctxt, fty: fn_ty) -> t { mk_t(cx, ty_fn(fty)) }
|
2011-04-20 18:52:04 -07:00
|
|
|
|
2011-12-20 16:33:55 +01:00
|
|
|
fn mk_iface(cx: ctxt, did: ast::def_id, tys: [t]) -> t {
|
2012-02-03 15:15:28 +01:00
|
|
|
mk_t(cx, ty_iface(did, tys))
|
2011-12-20 16:33:55 +01:00
|
|
|
}
|
|
|
|
|
2012-02-09 14:16:12 -08:00
|
|
|
fn mk_class(cx: ctxt, class_id: ast::def_id, tys: [t]) -> t {
|
|
|
|
mk_t(cx, ty_class(class_id, tys))
|
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn mk_res(cx: ctxt, did: ast::def_id, inner: t, tps: [t]) -> t {
|
2012-02-03 15:15:28 +01:00
|
|
|
mk_t(cx, ty_res(did, inner, tps))
|
2011-06-24 18:10:40 +02:00
|
|
|
}
|
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn mk_var(cx: ctxt, v: int) -> t { mk_t(cx, ty_var(v)) }
|
2011-04-20 18:52:04 -07:00
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn mk_self(cx: ctxt, tps: [t]) -> t { mk_t(cx, ty_self(tps)) }
|
2012-01-30 11:52:34 +01:00
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn mk_param(cx: ctxt, n: uint, k: def_id) -> t { mk_t(cx, ty_param(n, k)) }
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn mk_type(cx: ctxt) -> t { mk_t(cx, ty_type) }
|
2011-04-20 18:52:04 -07:00
|
|
|
|
2012-01-05 16:19:12 -08:00
|
|
|
fn mk_opaque_closure_ptr(cx: ctxt, ck: closure_kind) -> t {
|
2012-02-03 15:15:28 +01:00
|
|
|
mk_t(cx, ty_opaque_closure_ptr(ck))
|
2011-12-15 11:06:48 -08:00
|
|
|
}
|
|
|
|
|
2012-02-07 11:25:04 +01:00
|
|
|
fn mk_opaque_box(cx: ctxt) -> t { mk_t(cx, ty_opaque_box) }
|
|
|
|
|
2012-02-10 10:28:35 -08:00
|
|
|
fn mk_with_id(cx: ctxt, base: t, def_id: ast::def_id) -> t {
|
|
|
|
mk_t_with_id(cx, get(base).struct, some(def_id))
|
2011-12-23 01:57:35 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Converts s to its machine type equivalent
|
2012-02-03 15:15:28 +01:00
|
|
|
pure fn mach_sty(cfg: @session::config, t: t) -> sty {
|
|
|
|
alt get(t).struct {
|
2012-01-18 22:37:22 -08:00
|
|
|
ty_int(ast::ty_i) { ty_int(cfg.int_type) }
|
|
|
|
ty_uint(ast::ty_u) { ty_uint(cfg.uint_type) }
|
|
|
|
ty_float(ast::ty_f) { ty_float(cfg.float_type) }
|
2011-12-23 01:57:35 +01:00
|
|
|
s { s }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn default_arg_mode_for_ty(ty: ty::t) -> ast::rmode {
|
|
|
|
if ty::type_is_immediate(ty) { ast::by_val }
|
2012-02-02 16:50:17 -08:00
|
|
|
else { ast::by_ref }
|
|
|
|
}
|
|
|
|
|
2012-03-12 16:31:22 +01:00
|
|
|
fn walk_ty(ty: t, f: fn(t)) {
|
|
|
|
maybe_walk_ty(ty, {|t| f(t); true});
|
|
|
|
}
|
|
|
|
|
|
|
|
fn maybe_walk_ty(ty: t, f: fn(t) -> bool) {
|
|
|
|
if !f(ty) { ret; }
|
2012-02-03 15:15:28 +01:00
|
|
|
alt get(ty).struct {
|
2012-01-18 22:37:22 -08:00
|
|
|
ty_nil | ty_bot | ty_bool | ty_int(_) | ty_uint(_) | ty_float(_) |
|
2012-03-15 17:42:33 +01:00
|
|
|
ty_str | ty_type | ty_opaque_box |
|
2012-01-31 12:38:22 +01:00
|
|
|
ty_opaque_closure_ptr(_) | ty_var(_) | ty_param(_, _) {}
|
2012-03-08 14:05:16 -08:00
|
|
|
ty_box(tm) | ty_vec(tm) | ty_ptr(tm) | ty_rptr(_, tm) {
|
2012-03-12 16:31:22 +01:00
|
|
|
maybe_walk_ty(tm.ty, f);
|
2012-03-08 14:05:16 -08:00
|
|
|
}
|
2012-02-09 14:16:12 -08:00
|
|
|
ty_enum(_, subtys) | ty_iface(_, subtys) | ty_class(_, subtys)
|
|
|
|
| ty_self(subtys) {
|
2012-03-12 16:31:22 +01:00
|
|
|
for subty: t in subtys { maybe_walk_ty(subty, f); }
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
ty_rec(fields) {
|
2012-03-12 16:31:22 +01:00
|
|
|
for fl: field in fields { maybe_walk_ty(fl.mt.ty, f); }
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2012-03-12 16:31:22 +01:00
|
|
|
ty_tup(ts) { for tt in ts { maybe_walk_ty(tt, f); } }
|
2012-01-31 12:38:22 +01:00
|
|
|
ty_fn(ft) {
|
2012-03-12 16:31:22 +01:00
|
|
|
for a: arg in ft.inputs { maybe_walk_ty(a.ty, f); }
|
|
|
|
maybe_walk_ty(ft.output, f);
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
ty_res(_, sub, tps) {
|
2012-03-12 16:31:22 +01:00
|
|
|
maybe_walk_ty(sub, f);
|
|
|
|
for tp: t in tps { maybe_walk_ty(tp, f); }
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2012-03-12 16:31:22 +01:00
|
|
|
ty_constr(sub, _) { maybe_walk_ty(sub, f); }
|
|
|
|
ty_uniq(tm) { maybe_walk_ty(tm.ty, f); }
|
2011-04-15 12:23:00 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-01-19 14:24:03 -08:00
|
|
|
enum fold_mode {
|
2012-01-19 17:56:05 -08:00
|
|
|
fm_var(fn@(int) -> t),
|
|
|
|
fm_param(fn@(uint, def_id) -> t),
|
2012-03-26 19:00:11 -07:00
|
|
|
fm_rptr(fn@(region, bool /* under & */) -> region,
|
|
|
|
bool /* descend into outermost fn */),
|
2012-01-19 17:56:05 -08:00
|
|
|
fm_general(fn@(t) -> t),
|
2011-06-09 11:20:47 -07:00
|
|
|
}
|
2011-04-15 12:23:00 -07:00
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn fold_ty(cx: ctxt, fld: fold_mode, ty_0: t) -> t {
|
2012-03-23 11:37:10 -07:00
|
|
|
fn do_fold(cx: ctxt, fld: fold_mode, ty_0: t, under_rptr: bool) -> t {
|
|
|
|
let mut ty = ty_0;
|
|
|
|
|
|
|
|
let tb = get(ty);
|
|
|
|
alt fld {
|
|
|
|
fm_var(_) { if !tb.has_vars { ret ty; } }
|
|
|
|
fm_param(_) { if !tb.has_params { ret ty; } }
|
2012-03-26 19:00:11 -07:00
|
|
|
fm_rptr(_,_) { if !tb.has_rptrs { ret ty; } }
|
2012-03-23 11:37:10 -07:00
|
|
|
fm_general(_) {/* no fast path */ }
|
|
|
|
}
|
2012-02-02 16:50:17 -08:00
|
|
|
|
2012-03-23 11:37:10 -07:00
|
|
|
alt tb.struct {
|
|
|
|
ty_nil | ty_bot | ty_bool | ty_int(_) | ty_uint(_) | ty_float(_) |
|
|
|
|
ty_str | ty_type | ty_opaque_closure_ptr(_) |
|
|
|
|
ty_opaque_box {}
|
|
|
|
ty_box(tm) {
|
|
|
|
ty = mk_box(cx, {ty: do_fold(cx, fld, tm.ty, under_rptr),
|
|
|
|
mutbl: tm.mutbl});
|
|
|
|
}
|
|
|
|
ty_uniq(tm) {
|
|
|
|
ty = mk_uniq(cx, {ty: do_fold(cx, fld, tm.ty, under_rptr),
|
|
|
|
mutbl: tm.mutbl});
|
|
|
|
}
|
|
|
|
ty_ptr(tm) {
|
|
|
|
ty = mk_ptr(cx, {ty: do_fold(cx, fld, tm.ty, under_rptr),
|
|
|
|
mutbl: tm.mutbl});
|
|
|
|
}
|
|
|
|
ty_vec(tm) {
|
|
|
|
ty = mk_vec(cx, {ty: do_fold(cx, fld, tm.ty, under_rptr),
|
|
|
|
mutbl: tm.mutbl});
|
|
|
|
}
|
|
|
|
ty_enum(tid, subtys) {
|
|
|
|
ty = mk_enum(cx, tid,
|
|
|
|
vec::map(subtys, {|t|
|
|
|
|
do_fold(cx, fld, t, under_rptr)
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
ty_iface(did, subtys) {
|
|
|
|
ty = mk_iface(cx, did,
|
|
|
|
vec::map(subtys, {|t|
|
|
|
|
do_fold(cx, fld, t, under_rptr)
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
ty_self(subtys) {
|
|
|
|
ty = mk_self(cx, vec::map(subtys, {|t|
|
|
|
|
do_fold(cx, fld, t, under_rptr)
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
ty_rec(fields) {
|
|
|
|
let mut new_fields: [field] = [];
|
|
|
|
for fl: field in fields {
|
|
|
|
let new_ty = do_fold(cx, fld, fl.mt.ty, under_rptr);
|
|
|
|
let new_mt = {ty: new_ty, mutbl: fl.mt.mutbl};
|
|
|
|
new_fields += [{ident: fl.ident, mt: new_mt}];
|
|
|
|
}
|
|
|
|
ty = mk_rec(cx, new_fields);
|
|
|
|
}
|
|
|
|
ty_tup(ts) {
|
|
|
|
let mut new_ts = [];
|
|
|
|
for tt in ts { new_ts += [do_fold(cx, fld, tt, under_rptr)]; }
|
|
|
|
ty = mk_tup(cx, new_ts);
|
|
|
|
}
|
|
|
|
ty_fn(f) {
|
2012-03-26 19:00:11 -07:00
|
|
|
let mut new_fld;
|
2012-03-23 17:04:58 -07:00
|
|
|
alt fld {
|
2012-03-26 19:00:11 -07:00
|
|
|
fm_rptr(_, false) {
|
2012-03-23 17:04:58 -07:00
|
|
|
// Don't recurse into functions, because regions are
|
|
|
|
// universally quantified, well, universally, at function
|
|
|
|
// boundaries.
|
2012-03-26 19:00:11 -07:00
|
|
|
ret ty;
|
2012-03-23 17:04:58 -07:00
|
|
|
}
|
2012-03-26 19:00:11 -07:00
|
|
|
fm_rptr(f, true) {
|
|
|
|
new_fld = fm_rptr(f, false);
|
2012-03-23 17:04:58 -07:00
|
|
|
}
|
2012-03-26 19:00:11 -07:00
|
|
|
_ { new_fld = fld; }
|
2012-03-23 11:37:10 -07:00
|
|
|
}
|
2012-03-26 19:00:11 -07:00
|
|
|
|
|
|
|
let mut new_args: [arg] = [];
|
|
|
|
for a: arg in f.inputs {
|
|
|
|
let new_ty = do_fold(cx, new_fld, a.ty, under_rptr);
|
|
|
|
new_args += [{mode: a.mode, ty: new_ty}];
|
|
|
|
}
|
|
|
|
let new_output = do_fold(cx, new_fld, f.output, under_rptr);
|
|
|
|
ty = mk_fn(cx, {inputs: new_args, output: new_output with f});
|
2012-03-23 11:37:10 -07:00
|
|
|
}
|
|
|
|
ty_res(did, subty, tps) {
|
|
|
|
let mut new_tps = [];
|
|
|
|
for tp: t in tps {
|
|
|
|
new_tps += [do_fold(cx, fld, tp, under_rptr)];
|
|
|
|
}
|
|
|
|
ty = mk_res(cx, did, do_fold(cx, fld, subty, under_rptr),
|
|
|
|
new_tps);
|
|
|
|
}
|
|
|
|
ty_var(id) {
|
|
|
|
alt fld { fm_var(folder) { ty = folder(id); } _ {/* no-op */ } }
|
|
|
|
}
|
|
|
|
ty_param(id, did) {
|
|
|
|
alt fld { fm_param(folder) { ty = folder(id, did); } _ {} }
|
|
|
|
}
|
|
|
|
ty_rptr(r, tm) {
|
|
|
|
let region = alt fld {
|
2012-03-26 19:00:11 -07:00
|
|
|
fm_rptr(folder, _) { folder(r, under_rptr) }
|
2012-03-23 11:37:10 -07:00
|
|
|
_ { r }
|
|
|
|
};
|
|
|
|
ty = mk_rptr(cx, region,
|
|
|
|
{ty: do_fold(cx, fld, tm.ty, true),
|
|
|
|
mutbl: tm.mutbl});
|
|
|
|
}
|
|
|
|
ty_constr(subty, cs) {
|
|
|
|
ty = mk_constr(cx, do_fold(cx, fld, subty, under_rptr), cs);
|
|
|
|
}
|
|
|
|
_ {
|
|
|
|
cx.sess.bug("unsupported sort of type in fold_ty");
|
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2012-03-23 11:37:10 -07:00
|
|
|
alt tb.o_def_id {
|
|
|
|
some(did) { ty = mk_t_with_id(cx, get(ty).struct, some(did)); }
|
|
|
|
_ {}
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2012-03-23 11:37:10 -07:00
|
|
|
|
|
|
|
// If this is a general type fold, then we need to run it now.
|
|
|
|
alt fld { fm_general(folder) { ret folder(ty); } _ { ret ty; } }
|
2012-02-23 17:44:03 +01:00
|
|
|
}
|
2010-12-21 12:13:51 -08:00
|
|
|
|
2012-03-23 11:37:10 -07:00
|
|
|
ret do_fold(cx, fld, ty_0, false);
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-06-15 11:19:50 -07:00
|
|
|
// Type utilities
|
2011-07-20 19:04:45 -07:00
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn type_is_nil(ty: t) -> bool { get(ty).struct == ty_nil }
|
2011-05-14 19:02:30 -07:00
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn type_is_bot(ty: t) -> bool { get(ty).struct == ty_bot }
|
2010-12-21 12:13:51 -08:00
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn type_is_bool(ty: t) -> bool { get(ty).struct == ty_bool }
|
2011-04-19 15:22:57 -07:00
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn type_is_structural(ty: t) -> bool {
|
|
|
|
alt get(ty).struct {
|
2012-03-03 17:49:23 -08:00
|
|
|
ty_rec(_) | ty_class(_,_) | ty_tup(_) | ty_enum(_, _) | ty_fn(_) |
|
2012-02-10 11:32:03 +01:00
|
|
|
ty_iface(_, _) | ty_res(_, _, _) { true }
|
2011-12-20 16:33:55 +01:00
|
|
|
_ { false }
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn type_is_copyable(cx: ctxt, ty: t) -> bool {
|
2011-12-28 18:11:33 +01:00
|
|
|
ret kind_can_be_copied(type_kind(cx, ty));
|
2011-06-25 12:22:50 +02:00
|
|
|
}
|
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn type_is_sequence(ty: t) -> bool {
|
|
|
|
alt get(ty).struct {
|
2012-01-18 22:37:22 -08:00
|
|
|
ty_str { ret true; }
|
2011-08-18 14:11:06 -07:00
|
|
|
ty_vec(_) { ret true; }
|
2011-07-27 14:19:39 +02:00
|
|
|
_ { ret false; }
|
2011-03-02 16:42:09 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn type_is_str(ty: t) -> bool { get(ty).struct == ty_str }
|
2011-06-20 13:46:44 -07:00
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn sequence_element_type(cx: ctxt, ty: t) -> t {
|
2012-02-03 15:15:28 +01:00
|
|
|
alt get(ty).struct {
|
2012-01-18 22:37:22 -08:00
|
|
|
ty_str { ret mk_mach_uint(cx, ast::ty_u8); }
|
2011-08-18 14:11:06 -07:00
|
|
|
ty_vec(mt) { ret mt.ty; }
|
2011-09-02 15:34:58 -07:00
|
|
|
_ { cx.sess.bug("sequence_element_type called on non-sequence value"); }
|
2011-03-03 18:18:51 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
pure fn type_is_tup_like(ty: t) -> bool {
|
|
|
|
alt get(ty).struct {
|
2012-01-18 15:42:00 -08:00
|
|
|
ty_rec(_) | ty_tup(_) { true }
|
2011-09-17 10:17:40 -07:00
|
|
|
_ { false }
|
2011-01-19 16:29:14 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn get_element_type(ty: t, i: uint) -> t {
|
|
|
|
alt get(ty).struct {
|
2011-08-19 15:16:48 -07:00
|
|
|
ty_rec(flds) { ret flds[i].mt.ty; }
|
|
|
|
ty_tup(ts) { ret ts[i]; }
|
2012-02-03 15:15:28 +01:00
|
|
|
_ { fail "get_element_type called on invalid type"; }
|
2011-01-19 16:29:14 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
pure fn type_is_box(ty: t) -> bool {
|
|
|
|
alt get(ty).struct {
|
2011-09-21 14:00:11 -07:00
|
|
|
ty_box(_) { ret true; }
|
|
|
|
_ { ret false; }
|
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
pure fn type_is_boxed(ty: t) -> bool {
|
|
|
|
alt get(ty).struct {
|
2012-02-10 11:32:03 +01:00
|
|
|
ty_box(_) | ty_opaque_box { true }
|
2012-02-07 11:25:04 +01:00
|
|
|
_ { false }
|
2011-09-21 14:00:11 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
pure fn type_is_unique_box(ty: t) -> bool {
|
|
|
|
alt get(ty).struct {
|
2011-09-21 14:00:11 -07:00
|
|
|
ty_uniq(_) { ret true; }
|
|
|
|
_ { ret false; }
|
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2011-06-18 00:41:25 -07:00
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
pure fn type_is_unsafe_ptr(ty: t) -> bool {
|
|
|
|
alt get(ty).struct {
|
2011-11-02 12:15:50 +01:00
|
|
|
ty_ptr(_) { ret true; }
|
|
|
|
_ { ret false; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
pure fn type_is_vec(ty: t) -> bool {
|
|
|
|
ret alt get(ty).struct {
|
2011-09-02 15:34:58 -07:00
|
|
|
ty_vec(_) { true }
|
2012-01-18 22:37:22 -08:00
|
|
|
ty_str { true }
|
2011-09-02 15:34:58 -07:00
|
|
|
_ { false }
|
|
|
|
};
|
2011-08-25 10:18:02 +02:00
|
|
|
}
|
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
pure fn type_is_unique(ty: t) -> bool {
|
|
|
|
alt get(ty).struct {
|
2011-08-25 10:18:02 +02:00
|
|
|
ty_uniq(_) { ret true; }
|
|
|
|
ty_vec(_) { true }
|
2012-01-18 22:37:22 -08:00
|
|
|
ty_str { true }
|
2011-09-02 15:34:58 -07:00
|
|
|
_ { ret false; }
|
|
|
|
}
|
2011-08-16 12:38:42 -07:00
|
|
|
}
|
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
pure fn type_is_scalar(ty: t) -> bool {
|
|
|
|
alt get(ty).struct {
|
2012-01-18 22:37:22 -08:00
|
|
|
ty_nil | ty_bool | ty_int(_) | ty_float(_) | ty_uint(_) |
|
2012-03-15 17:42:33 +01:00
|
|
|
ty_type | ty_ptr(_) | ty_rptr(_, _) { true }
|
2011-12-07 21:06:12 +01:00
|
|
|
_ { false }
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2011-05-19 18:28:09 -07:00
|
|
|
}
|
|
|
|
|
2011-10-06 12:26:12 +02:00
|
|
|
// FIXME maybe inline this for speed?
|
2012-02-03 15:15:28 +01:00
|
|
|
fn type_is_immediate(ty: t) -> bool {
|
|
|
|
ret type_is_scalar(ty) || type_is_boxed(ty) ||
|
|
|
|
type_is_unique(ty);
|
2011-10-06 12:26:12 +02:00
|
|
|
}
|
|
|
|
|
2011-11-22 13:16:23 +01:00
|
|
|
fn type_needs_drop(cx: ctxt, ty: t) -> bool {
|
|
|
|
alt cx.needs_drop_cache.find(ty) {
|
2011-07-27 14:19:39 +02:00
|
|
|
some(result) { ret result; }
|
2012-01-18 22:37:22 -08:00
|
|
|
none {/* fall through */ }
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
|
2012-03-15 09:47:03 -04:00
|
|
|
let mut accum = false;
|
2012-02-03 15:15:28 +01:00
|
|
|
let result = alt get(ty).struct {
|
2011-07-27 14:19:39 +02:00
|
|
|
// scalar types
|
2012-01-18 22:37:22 -08:00
|
|
|
ty_nil | ty_bot | ty_bool | ty_int(_) | ty_float(_) | ty_uint(_) |
|
2012-03-08 17:22:10 -08:00
|
|
|
ty_type | ty_ptr(_) | ty_rptr(_, _) { false }
|
2011-07-27 14:19:39 +02:00
|
|
|
ty_rec(flds) {
|
2011-11-22 13:16:23 +01:00
|
|
|
for f in flds { if type_needs_drop(cx, f.mt.ty) { accum = true; } }
|
|
|
|
accum
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2012-03-27 22:08:48 -07:00
|
|
|
ty_class(did,_) {
|
|
|
|
for f in ty::class_items_as_fields(cx, did)
|
|
|
|
{ if type_needs_drop(cx, f.mt.ty) { accum = true; } }
|
|
|
|
accum
|
|
|
|
}
|
|
|
|
|
2011-08-15 11:40:26 +02:00
|
|
|
ty_tup(elts) {
|
2011-11-22 13:16:23 +01:00
|
|
|
for m in elts { if type_needs_drop(cx, m) { accum = true; } }
|
|
|
|
accum
|
2011-08-15 11:40:26 +02:00
|
|
|
}
|
2012-01-25 14:34:31 +01:00
|
|
|
ty_enum(did, tps) {
|
|
|
|
let variants = enum_variants(cx, did);
|
2011-12-15 17:14:58 -08:00
|
|
|
for variant in *variants {
|
2011-11-22 13:16:23 +01:00
|
|
|
for aty in variant.args {
|
2011-07-27 14:19:39 +02:00
|
|
|
// Perform any type parameter substitutions.
|
|
|
|
let arg_ty = substitute_type_params(cx, tps, aty);
|
2011-11-22 13:16:23 +01:00
|
|
|
if type_needs_drop(cx, arg_ty) { accum = true; }
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
2011-11-22 13:16:23 +01:00
|
|
|
if accum { break; }
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
2011-11-22 13:16:23 +01:00
|
|
|
accum
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2011-11-22 13:16:23 +01:00
|
|
|
_ { true }
|
|
|
|
};
|
2010-12-21 12:13:51 -08:00
|
|
|
|
2011-11-22 13:16:23 +01:00
|
|
|
cx.needs_drop_cache.insert(ty, result);
|
2011-07-27 14:19:39 +02:00
|
|
|
ret result;
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
|
2012-03-23 17:52:20 -07:00
|
|
|
// Some things don't need cleanups during unwinding because the
|
|
|
|
// task can free them all at once later. Currently only things
|
|
|
|
// that only contain scalars and shared boxes can avoid unwind
|
|
|
|
// cleanups.
|
|
|
|
fn type_needs_unwind_cleanup(cx: ctxt, ty: t) -> bool {
|
|
|
|
alt cx.needs_unwind_cleanup_cache.find(ty) {
|
|
|
|
some(result) { ret result; }
|
|
|
|
none { }
|
|
|
|
}
|
|
|
|
|
2012-03-26 16:25:29 -07:00
|
|
|
let tycache = new_ty_hash();
|
|
|
|
let needs_unwind_cleanup =
|
|
|
|
type_needs_unwind_cleanup_(cx, ty, tycache, false);
|
|
|
|
cx.needs_unwind_cleanup_cache.insert(ty, needs_unwind_cleanup);
|
|
|
|
ret needs_unwind_cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
fn type_needs_unwind_cleanup_(cx: ctxt, ty: t,
|
|
|
|
tycache: map::hashmap<t, ()>,
|
|
|
|
encountered_box: bool) -> bool {
|
|
|
|
|
2012-03-23 17:52:20 -07:00
|
|
|
// Prevent infinite recursion
|
2012-03-26 16:25:29 -07:00
|
|
|
alt tycache.find(ty) {
|
|
|
|
some(_) { ret false; }
|
|
|
|
none { tycache.insert(ty, ()); }
|
|
|
|
}
|
2012-03-23 17:52:20 -07:00
|
|
|
|
2012-03-26 16:25:29 -07:00
|
|
|
let mut encountered_box = encountered_box;
|
2012-03-23 17:52:20 -07:00
|
|
|
let mut needs_unwind_cleanup = false;
|
|
|
|
maybe_walk_ty(ty) {|ty|
|
|
|
|
alt get(ty).struct {
|
2012-03-26 16:25:29 -07:00
|
|
|
ty_box(_) | ty_opaque_box {
|
|
|
|
encountered_box = true;
|
|
|
|
true
|
|
|
|
}
|
2012-03-23 17:52:20 -07:00
|
|
|
ty_nil | ty_bot | ty_bool |
|
|
|
|
ty_int(_) | ty_uint(_) | ty_float(_) |
|
2012-03-26 16:25:29 -07:00
|
|
|
ty_rec(_) | ty_tup(_) | ty_ptr(_) {
|
2012-03-23 17:52:20 -07:00
|
|
|
true
|
|
|
|
}
|
|
|
|
ty_enum(did, tps) {
|
|
|
|
for v in *enum_variants(cx, did) {
|
|
|
|
for aty in v.args {
|
|
|
|
let t = substitute_type_params(cx, tps, aty);
|
2012-03-26 16:25:29 -07:00
|
|
|
needs_unwind_cleanup |=
|
|
|
|
type_needs_unwind_cleanup_(cx, t, tycache,
|
|
|
|
encountered_box);
|
2012-03-23 17:52:20 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
!needs_unwind_cleanup
|
|
|
|
}
|
2012-03-26 16:25:29 -07:00
|
|
|
ty_uniq(_) | ty_str | ty_vec(_) | ty_res(_, _, _) {
|
|
|
|
// Once we're inside a box, the annihilator will find
|
|
|
|
// it and destroy it.
|
|
|
|
if !encountered_box {
|
|
|
|
needs_unwind_cleanup = true;
|
|
|
|
false
|
|
|
|
} else {
|
|
|
|
true
|
|
|
|
}
|
|
|
|
}
|
2012-03-23 17:52:20 -07:00
|
|
|
_ {
|
|
|
|
needs_unwind_cleanup = true;
|
|
|
|
false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ret needs_unwind_cleanup;
|
|
|
|
}
|
|
|
|
|
2012-01-19 17:56:05 -08:00
|
|
|
enum kind { kind_sendable, kind_copyable, kind_noncopyable, }
|
2011-12-28 18:11:33 +01:00
|
|
|
|
|
|
|
// Using these query functons is preferable to direct comparison or matching
|
|
|
|
// against the kind constants, as we may modify the kind hierarchy in the
|
|
|
|
// future.
|
|
|
|
pure fn kind_can_be_copied(k: kind) -> bool {
|
|
|
|
ret alt k {
|
2012-01-18 22:37:22 -08:00
|
|
|
kind_sendable { true }
|
|
|
|
kind_copyable { true }
|
|
|
|
kind_noncopyable { false }
|
2011-12-28 18:11:33 +01:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
pure fn kind_can_be_sent(k: kind) -> bool {
|
|
|
|
ret alt k {
|
2012-01-18 22:37:22 -08:00
|
|
|
kind_sendable { true }
|
|
|
|
kind_copyable { false }
|
|
|
|
kind_noncopyable { false }
|
2011-12-28 18:11:33 +01:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
fn proto_kind(p: proto) -> kind {
|
|
|
|
alt p {
|
2012-01-18 22:37:22 -08:00
|
|
|
ast::proto_any { kind_noncopyable }
|
|
|
|
ast::proto_block { kind_noncopyable }
|
|
|
|
ast::proto_box { kind_copyable }
|
|
|
|
ast::proto_uniq { kind_sendable }
|
|
|
|
ast::proto_bare { kind_sendable }
|
2011-12-28 18:11:33 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-11-15 18:15:35 +01:00
|
|
|
fn kind_lteq(a: kind, b: kind) -> bool {
|
|
|
|
alt a {
|
2012-01-18 22:37:22 -08:00
|
|
|
kind_noncopyable { true }
|
|
|
|
kind_copyable { b != kind_noncopyable }
|
|
|
|
kind_sendable { b == kind_sendable }
|
2011-11-15 18:15:35 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn lower_kind(a: kind, b: kind) -> kind {
|
2012-02-03 15:15:28 +01:00
|
|
|
if kind_lteq(a, b) { a } else { b }
|
2011-11-15 18:15:35 +01:00
|
|
|
}
|
|
|
|
|
2011-12-28 18:11:33 +01:00
|
|
|
fn type_kind(cx: ctxt, ty: t) -> kind {
|
2011-07-27 17:49:00 -07:00
|
|
|
alt cx.kind_cache.find(ty) {
|
|
|
|
some(result) { ret result; }
|
2012-01-18 22:37:22 -08:00
|
|
|
none {/* fall through */ }
|
2011-07-27 17:49:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// Insert a default in case we loop back on self recursively.
|
2011-12-28 18:11:33 +01:00
|
|
|
cx.kind_cache.insert(ty, kind_sendable);
|
2011-07-27 17:49:00 -07:00
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
let result = alt get(ty).struct {
|
2011-11-15 18:15:35 +01:00
|
|
|
// Scalar and unique types are sendable
|
2012-01-18 22:37:22 -08:00
|
|
|
ty_nil | ty_bot | ty_bool | ty_int(_) | ty_uint(_) | ty_float(_) |
|
2012-03-15 17:42:33 +01:00
|
|
|
ty_ptr(_) | ty_str { kind_sendable }
|
2012-01-18 22:37:22 -08:00
|
|
|
ty_type { kind_copyable }
|
2011-12-28 18:11:33 +01:00
|
|
|
ty_fn(f) { proto_kind(f.proto) }
|
2012-01-18 22:37:22 -08:00
|
|
|
ty_opaque_closure_ptr(ck_block) { kind_noncopyable }
|
|
|
|
ty_opaque_closure_ptr(ck_box) { kind_copyable }
|
|
|
|
ty_opaque_closure_ptr(ck_uniq) { kind_sendable }
|
2011-07-27 21:23:40 -07:00
|
|
|
// Those with refcounts-to-inner raise pinned to shared,
|
|
|
|
// lower unique to shared. Therefore just set result to shared.
|
2012-02-07 11:25:04 +01:00
|
|
|
ty_box(_) | ty_iface(_, _) | ty_opaque_box { kind_copyable }
|
2012-03-08 17:22:10 -08:00
|
|
|
ty_rptr(_, _) { kind_copyable }
|
2011-11-23 13:25:26 +01:00
|
|
|
// Boxes and unique pointers raise pinned to shared.
|
|
|
|
ty_vec(tm) | ty_uniq(tm) { type_kind(cx, tm.ty) }
|
2011-07-27 17:49:00 -07:00
|
|
|
// Records lower to the lowest of their members.
|
|
|
|
ty_rec(flds) {
|
2012-03-15 09:47:03 -04:00
|
|
|
let mut lowest = kind_sendable;
|
2011-11-15 18:15:35 +01:00
|
|
|
for f in flds { lowest = lower_kind(lowest, type_kind(cx, f.mt.ty)); }
|
|
|
|
lowest
|
2011-07-27 17:49:00 -07:00
|
|
|
}
|
2011-08-18 18:06:00 -07:00
|
|
|
// Tuples lower to the lowest of their members.
|
|
|
|
ty_tup(tys) {
|
2012-03-15 09:47:03 -04:00
|
|
|
let mut lowest = kind_sendable;
|
2011-11-15 18:15:35 +01:00
|
|
|
for ty in tys { lowest = lower_kind(lowest, type_kind(cx, ty)); }
|
|
|
|
lowest
|
2011-08-18 18:06:00 -07:00
|
|
|
}
|
2012-01-25 14:34:31 +01:00
|
|
|
// Enums lower to the lowest of their variants.
|
|
|
|
ty_enum(did, tps) {
|
2012-03-15 09:47:03 -04:00
|
|
|
let mut lowest = kind_sendable;
|
2012-01-25 14:34:31 +01:00
|
|
|
for variant in *enum_variants(cx, did) {
|
2011-11-15 18:15:35 +01:00
|
|
|
for aty in variant.args {
|
2011-07-27 17:49:00 -07:00
|
|
|
// Perform any type parameter substitutions.
|
|
|
|
let arg_ty = substitute_type_params(cx, tps, aty);
|
2011-11-15 18:15:35 +01:00
|
|
|
lowest = lower_kind(lowest, type_kind(cx, arg_ty));
|
2011-12-28 18:11:33 +01:00
|
|
|
if lowest == kind_noncopyable { break; }
|
2011-07-27 17:49:00 -07:00
|
|
|
}
|
|
|
|
}
|
2011-11-15 18:15:35 +01:00
|
|
|
lowest
|
2011-07-27 17:49:00 -07:00
|
|
|
}
|
2011-11-15 18:15:35 +01:00
|
|
|
// Resources are always noncopyable.
|
2011-12-28 18:11:33 +01:00
|
|
|
ty_res(did, inner, tps) { kind_noncopyable }
|
2012-01-02 12:13:26 +01:00
|
|
|
ty_param(_, did) {
|
|
|
|
param_bounds_to_kind(cx.ty_param_bounds.get(did.node))
|
|
|
|
}
|
2011-11-15 18:15:35 +01:00
|
|
|
ty_constr(t, _) { type_kind(cx, t) }
|
2012-03-27 15:42:37 -07:00
|
|
|
ty_class(_, _) { fail "FIXME"; }
|
|
|
|
ty_var(_) { fail "FIXME"; }
|
|
|
|
ty_self(_) { kind_noncopyable }
|
2011-11-15 18:15:35 +01:00
|
|
|
};
|
2011-07-27 17:49:00 -07:00
|
|
|
|
|
|
|
cx.kind_cache.insert(ty, result);
|
|
|
|
ret result;
|
|
|
|
}
|
|
|
|
|
2012-03-28 13:42:33 -07:00
|
|
|
// True if instantiating an instance of `ty` requires an instead of `r_ty`.
|
|
|
|
fn is_instantiable(cx: ctxt, r_ty: t) -> bool {
|
|
|
|
|
|
|
|
fn type_requires(cx: ctxt, seen: @mut [def_id],
|
|
|
|
r_ty: t, ty: t) -> bool {
|
|
|
|
#debug["type_requires(%s, %s)?",
|
|
|
|
ty_to_str(cx, r_ty),
|
|
|
|
ty_to_str(cx, ty)];
|
|
|
|
|
|
|
|
let r = {
|
|
|
|
get(r_ty).struct == get(ty).struct ||
|
|
|
|
subtypes_require(cx, seen, r_ty, ty)
|
|
|
|
};
|
|
|
|
|
|
|
|
#debug["type_requires(%s, %s)? %b",
|
|
|
|
ty_to_str(cx, r_ty),
|
|
|
|
ty_to_str(cx, ty),
|
|
|
|
r];
|
|
|
|
ret r;
|
|
|
|
}
|
|
|
|
|
|
|
|
fn subtypes_require(cx: ctxt, seen: @mut [def_id],
|
|
|
|
r_ty: t, ty: t) -> bool {
|
|
|
|
#debug["subtypes_require(%s, %s)?",
|
|
|
|
ty_to_str(cx, r_ty),
|
|
|
|
ty_to_str(cx, ty)];
|
|
|
|
|
|
|
|
let r = alt get(ty).struct {
|
|
|
|
ty_nil |
|
|
|
|
ty_bot |
|
|
|
|
ty_bool |
|
|
|
|
ty_int(_) |
|
|
|
|
ty_uint(_) |
|
|
|
|
ty_float(_) |
|
|
|
|
ty_str |
|
|
|
|
ty_fn(_) |
|
|
|
|
ty_var(_) |
|
|
|
|
ty_param(_, _) |
|
|
|
|
ty_self(_) |
|
|
|
|
ty_type |
|
|
|
|
ty_opaque_box |
|
|
|
|
ty_opaque_closure_ptr(_) |
|
|
|
|
ty_vec(_) {
|
|
|
|
false
|
|
|
|
}
|
|
|
|
|
|
|
|
ty_constr(t, _) {
|
|
|
|
type_requires(cx, seen, r_ty, t)
|
|
|
|
}
|
|
|
|
|
|
|
|
ty_box(mt) |
|
|
|
|
ty_uniq(mt) |
|
|
|
|
ty_ptr(mt) |
|
|
|
|
ty_rptr(_, mt) {
|
|
|
|
be type_requires(cx, seen, r_ty, mt.ty);
|
|
|
|
}
|
|
|
|
|
|
|
|
ty_rec(fields) {
|
|
|
|
vec::any(fields) {|field|
|
|
|
|
type_requires(cx, seen, r_ty, field.mt.ty)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ty_iface(_, _) {
|
|
|
|
false
|
|
|
|
}
|
|
|
|
|
|
|
|
ty_class(did, _) if vec::contains(*seen, did) {
|
|
|
|
false
|
|
|
|
}
|
|
|
|
|
|
|
|
ty_class(did, tps) {
|
|
|
|
vec::push(*seen, did);
|
|
|
|
let r = vec::any(lookup_class_fields(cx, did)) {|f|
|
|
|
|
let fty = ty::lookup_item_type(cx, f.id);
|
|
|
|
let sty = substitute_type_params(cx, tps, fty.ty);
|
|
|
|
type_requires(cx, seen, r_ty, sty)
|
|
|
|
};
|
|
|
|
vec::pop(*seen);
|
|
|
|
r
|
|
|
|
}
|
|
|
|
|
|
|
|
ty_res(did, _, _) if vec::contains(*seen, did) {
|
|
|
|
false
|
|
|
|
}
|
|
|
|
|
|
|
|
ty_res(did, sub, tps) {
|
|
|
|
vec::push(*seen, did);
|
|
|
|
let sty = substitute_type_params(cx, tps, sub);
|
|
|
|
let r = type_requires(cx, seen, r_ty, sty);
|
|
|
|
vec::pop(*seen);
|
|
|
|
r
|
|
|
|
}
|
|
|
|
|
|
|
|
ty_tup(ts) {
|
|
|
|
vec::any(ts) {|t|
|
|
|
|
type_requires(cx, seen, r_ty, t)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ty_enum(did, _) if vec::contains(*seen, did) {
|
|
|
|
false
|
|
|
|
}
|
|
|
|
|
|
|
|
ty_enum(did, tps) {
|
|
|
|
vec::push(*seen, did);
|
|
|
|
let vs = enum_variants(cx, did);
|
|
|
|
let r = vec::len(*vs) > 0u && vec::all(*vs) {|variant|
|
|
|
|
vec::any(variant.args) {|aty|
|
|
|
|
let sty = substitute_type_params(cx, tps, aty);
|
|
|
|
type_requires(cx, seen, r_ty, sty)
|
|
|
|
}
|
|
|
|
};
|
|
|
|
vec::pop(*seen);
|
|
|
|
r
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
#debug["subtypes_require(%s, %s)? %b",
|
|
|
|
ty_to_str(cx, r_ty),
|
|
|
|
ty_to_str(cx, ty),
|
|
|
|
r];
|
|
|
|
|
|
|
|
ret r;
|
|
|
|
}
|
|
|
|
|
|
|
|
let seen = @mut [];
|
|
|
|
!subtypes_require(cx, seen, r_ty, r_ty)
|
|
|
|
}
|
|
|
|
|
2012-01-23 14:59:00 -08:00
|
|
|
fn type_structurally_contains(cx: ctxt, ty: t, test: fn(sty) -> bool) ->
|
2011-09-02 15:34:58 -07:00
|
|
|
bool {
|
2012-02-03 15:15:28 +01:00
|
|
|
let sty = get(ty).struct;
|
2011-08-22 13:39:32 +02:00
|
|
|
if test(sty) { ret true; }
|
|
|
|
alt sty {
|
2012-01-25 14:34:31 +01:00
|
|
|
ty_enum(did, tps) {
|
|
|
|
for variant in *enum_variants(cx, did) {
|
2011-08-22 13:39:32 +02:00
|
|
|
for aty in variant.args {
|
|
|
|
let sty = substitute_type_params(cx, tps, aty);
|
|
|
|
if type_structurally_contains(cx, sty, test) { ret true; }
|
|
|
|
}
|
2011-06-24 14:31:52 -04:00
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
ret false;
|
|
|
|
}
|
|
|
|
ty_rec(fields) {
|
2011-08-22 13:39:32 +02:00
|
|
|
for field in fields {
|
|
|
|
if type_structurally_contains(cx, field.mt.ty, test) { ret true; }
|
2011-06-30 14:46:17 +02:00
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
ret false;
|
|
|
|
}
|
2011-08-15 12:08:05 +02:00
|
|
|
ty_tup(ts) {
|
2011-08-22 13:39:32 +02:00
|
|
|
for tt in ts {
|
|
|
|
if type_structurally_contains(cx, tt, test) { ret true; }
|
|
|
|
}
|
2011-08-15 11:40:26 +02:00
|
|
|
ret false;
|
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
ty_res(_, sub, tps) {
|
2011-08-22 13:39:32 +02:00
|
|
|
let sty = substitute_type_params(cx, tps, sub);
|
|
|
|
ret type_structurally_contains(cx, sty, test);
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2011-08-22 13:39:32 +02:00
|
|
|
_ { ret false; }
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-11-18 09:46:44 +01:00
|
|
|
// Returns true for noncopyable types and types where a copy of a value can be
|
2012-03-26 18:35:18 -07:00
|
|
|
// distinguished from the value itself. I.e. types with mut content that's
|
2011-11-18 09:46:44 +01:00
|
|
|
// not shared through a pointer.
|
2011-09-12 11:27:30 +02:00
|
|
|
fn type_allows_implicit_copy(cx: ctxt, ty: t) -> bool {
|
2012-01-11 09:58:05 -08:00
|
|
|
ret !type_structurally_contains(cx, ty, {|sty|
|
|
|
|
alt sty {
|
2011-09-12 14:43:41 +02:00
|
|
|
ty_param(_, _) { true }
|
|
|
|
ty_vec(mt) {
|
2012-02-15 11:25:39 -08:00
|
|
|
mt.mutbl != ast::m_imm
|
2011-09-12 14:43:41 +02:00
|
|
|
}
|
|
|
|
ty_rec(fields) {
|
2012-03-26 12:39:20 +02:00
|
|
|
vec::any(fields, {|f| f.mt.mutbl != ast::m_imm})
|
2011-09-12 14:43:41 +02:00
|
|
|
}
|
|
|
|
_ { false }
|
2012-01-11 09:58:05 -08:00
|
|
|
}
|
2011-12-28 18:11:33 +01:00
|
|
|
}) && type_kind(cx, ty) != kind_noncopyable;
|
2011-09-12 11:27:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
fn type_structurally_contains_uniques(cx: ctxt, ty: t) -> bool {
|
2012-01-11 09:58:05 -08:00
|
|
|
ret type_structurally_contains(cx, ty, {|sty|
|
2012-03-26 12:39:20 +02:00
|
|
|
alt sty {
|
|
|
|
ty_uniq(_) { true }
|
2011-09-12 14:43:41 +02:00
|
|
|
ty_vec(_) { true }
|
2012-01-18 22:37:22 -08:00
|
|
|
ty_str { true }
|
2012-03-26 12:39:20 +02:00
|
|
|
_ { false }
|
|
|
|
}
|
2011-09-12 14:43:41 +02:00
|
|
|
});
|
2011-09-12 11:27:30 +02:00
|
|
|
}
|
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn type_is_integral(ty: t) -> bool {
|
|
|
|
alt get(ty).struct {
|
2012-01-18 22:37:22 -08:00
|
|
|
ty_int(_) | ty_uint(_) | ty_bool { true }
|
2011-12-07 21:06:12 +01:00
|
|
|
_ { false }
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn type_is_fp(ty: t) -> bool {
|
|
|
|
alt get(ty).struct {
|
2011-12-07 21:06:12 +01:00
|
|
|
ty_float(_) { true }
|
|
|
|
_ { false }
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn type_is_numeric(ty: t) -> bool {
|
|
|
|
ret type_is_integral(ty) || type_is_fp(ty);
|
2011-12-02 13:42:51 +01:00
|
|
|
}
|
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn type_is_signed(ty: t) -> bool {
|
|
|
|
alt get(ty).struct {
|
2011-12-07 21:06:12 +01:00
|
|
|
ty_int(_) { true }
|
|
|
|
_ { false }
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-12-20 20:47:03 +01:00
|
|
|
// Whether a type is Plain Old Data -- meaning it does not contain pointers
|
|
|
|
// that the cycle collector might care about.
|
2011-09-12 11:27:30 +02:00
|
|
|
fn type_is_pod(cx: ctxt, ty: t) -> bool {
|
2012-03-15 09:47:03 -04:00
|
|
|
let mut result = true;
|
2012-02-03 15:15:28 +01:00
|
|
|
alt get(ty).struct {
|
2011-08-19 15:16:48 -07:00
|
|
|
// Scalar types
|
2012-01-18 22:37:22 -08:00
|
|
|
ty_nil | ty_bot | ty_bool | ty_int(_) | ty_float(_) | ty_uint(_) |
|
2012-03-15 17:42:33 +01:00
|
|
|
ty_type | ty_ptr(_) { result = true; }
|
2011-08-19 15:16:48 -07:00
|
|
|
// Boxed types
|
2012-01-18 22:37:22 -08:00
|
|
|
ty_str | ty_box(_) | ty_uniq(_) | ty_vec(_) | ty_fn(_) |
|
2012-03-08 17:22:10 -08:00
|
|
|
ty_iface(_, _) | ty_rptr(_,_) | ty_opaque_box { result = false; }
|
2011-08-19 15:16:48 -07:00
|
|
|
// Structural types
|
2012-01-25 14:34:31 +01:00
|
|
|
ty_enum(did, tps) {
|
|
|
|
let variants = enum_variants(cx, did);
|
2011-12-15 17:14:58 -08:00
|
|
|
for variant: variant_info in *variants {
|
2011-08-19 15:16:48 -07:00
|
|
|
let tup_ty = mk_tup(cx, variant.args);
|
|
|
|
|
|
|
|
// Perform any type parameter substitutions.
|
2012-03-15 09:47:03 -04:00
|
|
|
let tup_ty = substitute_type_params(cx, tps, tup_ty);
|
2011-08-19 15:16:48 -07:00
|
|
|
if !type_is_pod(cx, tup_ty) { result = false; }
|
2011-08-15 11:40:26 +02:00
|
|
|
}
|
2011-08-19 15:16:48 -07:00
|
|
|
}
|
|
|
|
ty_rec(flds) {
|
|
|
|
for f: field in flds {
|
|
|
|
if !type_is_pod(cx, f.mt.ty) { result = false; }
|
2011-08-04 10:46:10 -07:00
|
|
|
}
|
2011-08-19 15:16:48 -07:00
|
|
|
}
|
|
|
|
ty_tup(elts) {
|
|
|
|
for elt in elts { if !type_is_pod(cx, elt) { result = false; } }
|
|
|
|
}
|
|
|
|
ty_res(_, inner, tps) {
|
|
|
|
result = type_is_pod(cx, substitute_type_params(cx, tps, inner));
|
|
|
|
}
|
|
|
|
ty_constr(subt, _) { result = type_is_pod(cx, subt); }
|
|
|
|
ty_param(_, _) { result = false; }
|
2012-01-30 21:00:57 -08:00
|
|
|
ty_opaque_closure_ptr(_) { result = true; }
|
2012-01-30 11:52:34 +01:00
|
|
|
_ { cx.sess.bug("unexpected type in type_is_pod"); }
|
2011-08-04 10:46:10 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
ret result;
|
|
|
|
}
|
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn type_is_enum(ty: t) -> bool {
|
|
|
|
alt get(ty).struct {
|
2012-01-25 14:34:31 +01:00
|
|
|
ty_enum(_, _) { ret true; }
|
2012-01-10 14:55:54 -07:00
|
|
|
_ { ret false;}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-01-19 14:24:03 -08:00
|
|
|
// Whether a type is enum like, that is a enum type with only nullary
|
2012-01-10 14:55:54 -07:00
|
|
|
// constructors
|
2012-01-10 18:33:26 -07:00
|
|
|
fn type_is_c_like_enum(cx: ctxt, ty: t) -> bool {
|
2012-02-03 15:15:28 +01:00
|
|
|
alt get(ty).struct {
|
2012-01-25 14:34:31 +01:00
|
|
|
ty_enum(did, tps) {
|
|
|
|
let variants = enum_variants(cx, did);
|
2012-01-10 14:55:54 -07:00
|
|
|
let some_n_ary = vec::any(*variants, {|v| vec::len(v.args) > 0u});
|
|
|
|
ret !some_n_ary;
|
|
|
|
}
|
|
|
|
_ { ret false;}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn type_param(ty: t) -> option<uint> {
|
|
|
|
alt get(ty).struct {
|
2011-08-19 15:16:48 -07:00
|
|
|
ty_param(id, _) { ret some(id); }
|
2011-07-27 14:19:39 +02:00
|
|
|
_ {/* fall through */ }
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
2011-07-21 15:46:03 -07:00
|
|
|
ret none;
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
|
2011-08-18 15:00:12 -07:00
|
|
|
// Returns a vec of all the type variables
|
2011-08-03 18:06:57 -07:00
|
|
|
// occurring in t. It may contain duplicates.
|
2012-03-12 16:31:22 +01:00
|
|
|
fn vars_in_type(ty: t) -> [int] {
|
2012-03-15 09:47:03 -04:00
|
|
|
let mut rslt = [];
|
2012-03-12 16:31:22 +01:00
|
|
|
walk_ty(ty) {|ty|
|
2012-02-03 15:15:28 +01:00
|
|
|
alt get(ty).struct { ty_var(v) { rslt += [v]; } _ { } }
|
2011-08-03 18:06:57 -07:00
|
|
|
}
|
2012-01-31 12:38:22 +01:00
|
|
|
rslt
|
2011-08-03 18:06:57 -07:00
|
|
|
}
|
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn type_autoderef(cx: ctxt, t: t) -> t {
|
2012-03-15 09:47:03 -04:00
|
|
|
let mut t1 = t;
|
2012-03-10 20:35:41 -08:00
|
|
|
loop {
|
2012-02-03 15:15:28 +01:00
|
|
|
alt get(t1).struct {
|
2012-03-14 16:13:05 -07:00
|
|
|
ty_box(mt) | ty_uniq(mt) | ty::ty_rptr(_, mt) { t1 = mt.ty; }
|
2011-09-29 11:18:40 +02:00
|
|
|
ty_res(_, inner, tps) {
|
2011-07-27 14:19:39 +02:00
|
|
|
t1 = substitute_type_params(cx, tps, inner);
|
|
|
|
}
|
2012-01-25 14:34:31 +01:00
|
|
|
ty_enum(did, tps) {
|
|
|
|
let variants = enum_variants(cx, did);
|
2011-12-15 17:14:58 -08:00
|
|
|
if vec::len(*variants) != 1u || vec::len(variants[0].args) != 1u {
|
2011-07-27 14:19:39 +02:00
|
|
|
break;
|
2011-07-01 13:57:03 +02:00
|
|
|
}
|
2011-08-19 15:16:48 -07:00
|
|
|
t1 = substitute_type_params(cx, tps, variants[0].args[0]);
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
_ { break; }
|
2011-07-01 01:48:32 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
ret t1;
|
|
|
|
}
|
|
|
|
|
2011-12-22 14:24:36 +01:00
|
|
|
// Type hashing.
|
2011-09-12 11:27:30 +02:00
|
|
|
fn hash_type_structure(st: sty) -> uint {
|
2012-02-03 15:15:28 +01:00
|
|
|
fn hash_uint(id: uint, n: uint) -> uint { (id << 2u) + n }
|
2011-07-27 14:19:39 +02:00
|
|
|
fn hash_def(id: uint, did: ast::def_id) -> uint {
|
2012-02-03 15:15:28 +01:00
|
|
|
let h = (id << 2u) + (did.crate as uint);
|
|
|
|
(h << 2u) + (did.node as uint)
|
2011-04-20 14:34:17 -07:00
|
|
|
}
|
2012-02-03 15:15:28 +01:00
|
|
|
fn hash_subty(id: uint, subty: t) -> uint { (id << 2u) + type_id(subty) }
|
2011-12-15 11:06:48 -08:00
|
|
|
fn hash_subtys(id: uint, subtys: [t]) -> uint {
|
2012-03-15 09:47:03 -04:00
|
|
|
let mut h = id;
|
2012-02-03 15:15:28 +01:00
|
|
|
for s in subtys { h = (h << 2u) + type_id(s) }
|
|
|
|
h
|
2011-12-15 11:06:48 -08:00
|
|
|
}
|
2011-09-12 11:27:30 +02:00
|
|
|
fn hash_type_constr(id: uint, c: @type_constr) -> uint {
|
2012-03-15 09:47:03 -04:00
|
|
|
let mut h = id;
|
2012-02-03 15:15:28 +01:00
|
|
|
h = (h << 2u) + hash_def(h, c.node.id);
|
|
|
|
// FIXME this makes little sense
|
|
|
|
for a in c.node.args {
|
2011-07-27 14:19:39 +02:00
|
|
|
alt a.node {
|
2012-02-03 15:15:28 +01:00
|
|
|
carg_base { h += h << 2u; }
|
|
|
|
carg_lit(_) { fail "lit args not implemented yet"; }
|
|
|
|
carg_ident(p) { h += h << 2u; }
|
2011-07-19 17:52:34 -07:00
|
|
|
}
|
|
|
|
}
|
2012-02-03 15:15:28 +01:00
|
|
|
h
|
2011-04-20 14:34:17 -07:00
|
|
|
}
|
2012-03-08 14:05:16 -08:00
|
|
|
fn hash_region(r: region) -> uint {
|
|
|
|
alt r {
|
2012-03-23 11:37:10 -07:00
|
|
|
re_block(_) { 0u }
|
|
|
|
re_self { 1u }
|
|
|
|
re_inferred { 2u }
|
|
|
|
re_param(_) { 3u }
|
|
|
|
re_var(_) { 4u }
|
2012-03-08 14:05:16 -08:00
|
|
|
}
|
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
alt st {
|
2012-01-18 22:37:22 -08:00
|
|
|
ty_nil { 0u } ty_bool { 1u }
|
2011-12-07 21:06:12 +01:00
|
|
|
ty_int(t) {
|
|
|
|
alt t {
|
2012-01-18 22:37:22 -08:00
|
|
|
ast::ty_i { 2u } ast::ty_char { 3u } ast::ty_i8 { 4u }
|
|
|
|
ast::ty_i16 { 5u } ast::ty_i32 { 6u } ast::ty_i64 { 7u }
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
}
|
2011-12-07 21:06:12 +01:00
|
|
|
ty_uint(t) {
|
|
|
|
alt t {
|
2012-01-18 22:37:22 -08:00
|
|
|
ast::ty_u { 8u } ast::ty_u8 { 9u } ast::ty_u16 { 10u }
|
|
|
|
ast::ty_u32 { 11u } ast::ty_u64 { 12u }
|
2011-12-07 21:06:12 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
ty_float(t) {
|
2012-01-18 22:37:22 -08:00
|
|
|
alt t { ast::ty_f { 13u } ast::ty_f32 { 14u } ast::ty_f64 { 15u } }
|
2011-12-07 21:06:12 +01:00
|
|
|
}
|
2012-02-03 15:15:28 +01:00
|
|
|
ty_str { 17u }
|
2012-01-25 14:34:31 +01:00
|
|
|
ty_enum(did, tys) {
|
2012-03-15 09:47:03 -04:00
|
|
|
let mut h = hash_def(18u, did);
|
2012-02-03 15:15:28 +01:00
|
|
|
for typ: t in tys { h = hash_subty(h, typ); }
|
|
|
|
h
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2012-02-03 15:15:28 +01:00
|
|
|
ty_box(mt) { hash_subty(19u, mt.ty) }
|
|
|
|
ty_vec(mt) { hash_subty(21u, mt.ty) }
|
2011-07-27 14:19:39 +02:00
|
|
|
ty_rec(fields) {
|
2012-03-15 09:47:03 -04:00
|
|
|
let mut h = 26u;
|
2012-02-03 15:15:28 +01:00
|
|
|
for f in fields { h = hash_subty(h, f.mt.ty); }
|
|
|
|
h
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2012-02-03 15:15:28 +01:00
|
|
|
ty_tup(ts) { hash_subtys(25u, ts) }
|
|
|
|
ty_fn(f) {
|
2012-03-15 09:47:03 -04:00
|
|
|
let mut h = 27u;
|
2012-02-03 15:15:28 +01:00
|
|
|
for a in f.inputs { h = hash_subty(h, a.ty); }
|
|
|
|
hash_subty(h, f.output)
|
|
|
|
}
|
|
|
|
ty_var(v) { hash_uint(30u, v as uint) }
|
|
|
|
ty_param(pid, did) { hash_def(hash_uint(31u, pid), did) }
|
2012-01-30 11:52:34 +01:00
|
|
|
ty_self(ts) {
|
2012-03-15 09:47:03 -04:00
|
|
|
let mut h = 28u;
|
2012-02-03 15:15:28 +01:00
|
|
|
for t in ts { h = hash_subty(h, t); }
|
|
|
|
h
|
2012-01-30 11:52:34 +01:00
|
|
|
}
|
2012-02-03 15:15:28 +01:00
|
|
|
ty_type { 32u }
|
|
|
|
ty_bot { 34u }
|
|
|
|
ty_ptr(mt) { hash_subty(35u, mt.ty) }
|
2012-03-08 14:05:16 -08:00
|
|
|
ty_rptr(region, mt) {
|
2012-03-15 09:47:03 -04:00
|
|
|
let mut h = (46u << 2u) + hash_region(region);
|
2012-03-08 14:05:16 -08:00
|
|
|
hash_subty(h, mt.ty)
|
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
ty_res(did, sub, tps) {
|
2012-03-15 09:47:03 -04:00
|
|
|
let mut h = hash_subty(hash_def(18u, did), sub);
|
2012-02-03 15:15:28 +01:00
|
|
|
hash_subtys(h, tps)
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
ty_constr(t, cs) {
|
2012-03-15 09:47:03 -04:00
|
|
|
let mut h = hash_subty(36u, t);
|
2012-02-03 15:15:28 +01:00
|
|
|
for c in cs { h = (h << 2u) + hash_type_constr(h, c); }
|
|
|
|
h
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2012-02-03 15:15:28 +01:00
|
|
|
ty_uniq(mt) { hash_subty(37u, mt.ty) }
|
2011-12-20 16:33:55 +01:00
|
|
|
ty_iface(did, tys) {
|
2012-03-15 09:47:03 -04:00
|
|
|
let mut h = hash_def(40u, did);
|
2012-01-05 16:19:12 -08:00
|
|
|
for typ: t in tys { h = hash_subty(h, typ); }
|
2012-02-03 15:15:28 +01:00
|
|
|
h
|
2011-12-20 16:33:55 +01:00
|
|
|
}
|
2012-02-03 15:15:28 +01:00
|
|
|
ty_opaque_closure_ptr(ck_block) { 41u }
|
|
|
|
ty_opaque_closure_ptr(ck_box) { 42u }
|
|
|
|
ty_opaque_closure_ptr(ck_uniq) { 43u }
|
2012-02-07 11:25:04 +01:00
|
|
|
ty_opaque_box { 44u }
|
2012-02-09 14:16:12 -08:00
|
|
|
ty_class(did, tys) {
|
2012-03-15 09:47:03 -04:00
|
|
|
let mut h = hash_def(45u, did);
|
2012-02-09 14:16:12 -08:00
|
|
|
for typ: t in tys { h = hash_subty(h, typ); }
|
|
|
|
h
|
|
|
|
}
|
2011-04-20 14:34:17 -07:00
|
|
|
}
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
|
2012-01-23 14:59:00 -08:00
|
|
|
fn arg_eq<T>(eq: fn(T, T) -> bool,
|
2012-01-11 09:58:05 -08:00
|
|
|
a: @sp_constr_arg<T>,
|
|
|
|
b: @sp_constr_arg<T>)
|
2011-09-12 11:27:30 +02:00
|
|
|
-> bool {
|
2011-07-27 14:19:39 +02:00
|
|
|
alt a.node {
|
2012-01-18 22:37:22 -08:00
|
|
|
ast::carg_base {
|
|
|
|
alt b.node { ast::carg_base { ret true; } _ { ret false; } }
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
ast::carg_ident(s) {
|
|
|
|
alt b.node { ast::carg_ident(t) { ret eq(s, t); } _ { ret false; } }
|
|
|
|
}
|
|
|
|
ast::carg_lit(l) {
|
2011-11-22 11:49:29 +01:00
|
|
|
alt b.node {
|
2012-03-22 14:56:56 -07:00
|
|
|
ast::carg_lit(m) { ret const_eval::lit_eq(l, m); } _ { ret false; }
|
2011-11-22 11:49:29 +01:00
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-01-23 14:59:00 -08:00
|
|
|
fn args_eq<T>(eq: fn(T, T) -> bool,
|
2012-01-11 09:58:05 -08:00
|
|
|
a: [@sp_constr_arg<T>],
|
2011-09-12 11:27:30 +02:00
|
|
|
b: [@sp_constr_arg<T>]) -> bool {
|
2012-03-15 09:47:03 -04:00
|
|
|
let mut i: uint = 0u;
|
2011-08-12 07:15:18 -07:00
|
|
|
for arg: @sp_constr_arg<T> in a {
|
2011-08-19 15:16:48 -07:00
|
|
|
if !arg_eq(eq, arg, b[i]) { ret false; }
|
2011-06-09 09:48:16 -07:00
|
|
|
i += 1u;
|
|
|
|
}
|
|
|
|
ret true;
|
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn constr_eq(c: @constr, d: @constr) -> bool {
|
2011-12-22 14:24:36 +01:00
|
|
|
fn eq_int(&&x: uint, &&y: uint) -> bool { ret x == y; }
|
2011-06-15 11:19:50 -07:00
|
|
|
ret path_to_str(c.node.path) == path_to_str(d.node.path) &&
|
2011-06-16 16:55:46 -07:00
|
|
|
// FIXME: hack
|
2011-06-15 11:19:50 -07:00
|
|
|
args_eq(eq_int, c.node.args, d.node.args);
|
2011-06-09 09:48:16 -07:00
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn constrs_eq(cs: [@constr], ds: [@constr]) -> bool {
|
2011-08-15 16:38:23 -07:00
|
|
|
if vec::len(cs) != vec::len(ds) { ret false; }
|
2012-03-15 09:47:03 -04:00
|
|
|
let mut i = 0u;
|
2011-08-19 15:16:48 -07:00
|
|
|
for c: @constr in cs { if !constr_eq(c, ds[i]) { ret false; } i += 1u; }
|
2011-06-09 09:48:16 -07:00
|
|
|
ret true;
|
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn node_id_to_type(cx: ctxt, id: ast::node_id) -> t {
|
2012-03-03 17:49:23 -08:00
|
|
|
alt smallintmap::find(*cx.node_types, id as uint) {
|
|
|
|
some(t) { t }
|
2012-03-28 12:54:06 -07:00
|
|
|
none { cx.sess.bug(#fmt("node_id_to_type: unbound node ID %s",
|
|
|
|
ast_map::node_id_to_str(cx.items, id))); }
|
2012-03-03 17:49:23 -08:00
|
|
|
}
|
2011-05-13 16:40:21 -07:00
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn node_id_to_type_params(cx: ctxt, id: ast::node_id) -> [t] {
|
2012-01-30 17:28:30 +01:00
|
|
|
alt cx.node_type_substs.find(id) {
|
2012-01-18 22:37:22 -08:00
|
|
|
none { ret []; }
|
2012-01-30 17:28:30 +01:00
|
|
|
some(ts) { ret ts; }
|
2011-05-13 16:40:21 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn node_id_has_type_params(cx: ctxt, id: ast::node_id) -> bool {
|
2012-01-30 17:28:30 +01:00
|
|
|
ret cx.node_type_substs.contains_key(id);
|
2011-05-13 16:53:35 -07:00
|
|
|
}
|
|
|
|
|
2011-06-15 11:19:50 -07:00
|
|
|
// Type accessors for substructures of types
|
2012-02-03 15:15:28 +01:00
|
|
|
fn ty_fn_args(fty: t) -> [arg] {
|
|
|
|
alt get(fty).struct {
|
|
|
|
ty_fn(f) { f.inputs }
|
|
|
|
_ { fail "ty_fn_args() called on non-fn type"; }
|
2011-02-18 17:30:57 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn ty_fn_proto(fty: t) -> ast::proto {
|
|
|
|
alt get(fty).struct {
|
|
|
|
ty_fn(f) { f.proto }
|
|
|
|
_ { fail "ty_fn_proto() called on non-fn type"; }
|
2011-07-29 20:51:18 +02:00
|
|
|
}
|
2011-01-03 18:22:39 -08:00
|
|
|
}
|
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
pure fn ty_fn_ret(fty: t) -> t {
|
|
|
|
alt get(fty).struct {
|
|
|
|
ty_fn(f) { f.output }
|
|
|
|
_ { fail "ty_fn_ret() called on non-fn type"; }
|
2011-02-18 17:30:57 -08:00
|
|
|
}
|
2011-01-03 18:22:39 -08:00
|
|
|
}
|
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn ty_fn_ret_style(fty: t) -> ast::ret_style {
|
|
|
|
alt get(fty).struct {
|
|
|
|
ty_fn(f) { f.ret_style }
|
|
|
|
_ { fail "ty_fn_ret_style() called on non-fn type"; }
|
2011-09-14 14:34:50 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn is_fn_ty(fty: t) -> bool {
|
|
|
|
alt get(fty).struct {
|
|
|
|
ty_fn(_) { ret true; }
|
2011-07-27 14:19:39 +02:00
|
|
|
_ { ret false; }
|
2011-02-18 17:30:57 -08:00
|
|
|
}
|
2011-01-03 18:22:39 -08:00
|
|
|
}
|
|
|
|
|
2011-08-24 17:24:58 -07:00
|
|
|
// Just checks whether it's a fn that returns bool,
|
|
|
|
// not its purity.
|
2012-02-03 15:15:28 +01:00
|
|
|
fn is_pred_ty(fty: t) -> bool {
|
|
|
|
is_fn_ty(fty) && type_is_bool(ty_fn_ret(fty))
|
2011-08-24 17:24:58 -07:00
|
|
|
}
|
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn ty_var_id(typ: t) -> int {
|
|
|
|
alt get(typ).struct {
|
|
|
|
ty_var(vid) { ret vid; }
|
2011-12-22 14:42:52 -08:00
|
|
|
_ { #error("ty_var_id called on non-var ty"); fail; }
|
2011-05-20 18:36:35 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-01-03 18:22:39 -08:00
|
|
|
|
2010-12-21 12:13:51 -08:00
|
|
|
// Type accessors for AST nodes
|
2011-09-12 11:27:30 +02:00
|
|
|
fn block_ty(cx: ctxt, b: ast::blk) -> t {
|
2011-06-19 22:41:21 +02:00
|
|
|
ret node_id_to_type(cx, b.node.id);
|
|
|
|
}
|
2010-12-21 12:13:51 -08:00
|
|
|
|
|
|
|
|
2011-04-08 21:27:54 -07:00
|
|
|
// Returns the type of a pattern as a monotype. Like @expr_ty, this function
|
|
|
|
// doesn't provide type parameter substitutions.
|
2011-09-12 11:27:30 +02:00
|
|
|
fn pat_ty(cx: ctxt, pat: @ast::pat) -> t {
|
2012-01-30 17:28:30 +01:00
|
|
|
ret node_id_to_type(cx, pat.id);
|
2011-06-19 22:41:21 +02:00
|
|
|
}
|
|
|
|
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2011-04-08 21:27:54 -07:00
|
|
|
// Returns the type of an expression as a monotype.
|
|
|
|
//
|
|
|
|
// NB: This type doesn't provide type parameter substitutions; e.g. if you
|
|
|
|
// ask for the type of "id" in "id(3)", it will return "fn(&int) -> int"
|
2011-08-22 12:45:18 +02:00
|
|
|
// instead of "fn(t) -> T with T = int". If this isn't what you want, see
|
2011-04-08 21:27:54 -07:00
|
|
|
// expr_ty_params_and_ty() below.
|
2011-09-12 11:27:30 +02:00
|
|
|
fn expr_ty(cx: ctxt, expr: @ast::expr) -> t {
|
2012-01-30 17:28:30 +01:00
|
|
|
ret node_id_to_type(cx, expr.id);
|
2011-04-08 21:27:54 -07:00
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn expr_ty_params_and_ty(cx: ctxt, expr: @ast::expr) -> {params: [t], ty: t} {
|
2011-07-27 14:19:39 +02:00
|
|
|
ret {params: node_id_to_type_params(cx, expr.id),
|
|
|
|
ty: node_id_to_type(cx, expr.id)};
|
2011-04-08 21:27:54 -07:00
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn expr_has_ty_params(cx: ctxt, expr: @ast::expr) -> bool {
|
2011-06-21 22:16:40 +02:00
|
|
|
ret node_id_has_type_params(cx, expr.id);
|
2011-04-08 21:27:54 -07:00
|
|
|
}
|
|
|
|
|
2012-01-13 10:58:31 +01:00
|
|
|
fn expr_is_lval(method_map: typeck::method_map, e: @ast::expr) -> bool {
|
2011-11-15 18:15:35 +01:00
|
|
|
alt e.node {
|
2012-01-26 15:23:04 +01:00
|
|
|
ast::expr_path(_) | ast::expr_unary(ast::deref, _) { true }
|
|
|
|
ast::expr_field(_, _, _) | ast::expr_index(_, _) {
|
|
|
|
!method_map.contains_key(e.id)
|
|
|
|
}
|
2011-11-15 18:15:35 +01:00
|
|
|
_ { false }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn stmt_node_id(s: @ast::stmt) -> ast::node_id {
|
2011-07-27 14:19:39 +02:00
|
|
|
alt s.node {
|
2012-01-04 14:16:41 -08:00
|
|
|
ast::stmt_decl(_, id) | stmt_expr(_, id) | stmt_semi(_, id) {
|
|
|
|
ret id;
|
|
|
|
}
|
2011-05-17 19:00:29 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-01-31 17:05:20 -08:00
|
|
|
fn field_idx(id: ast::ident, fields: [field]) -> option<uint> {
|
2012-03-15 09:47:03 -04:00
|
|
|
let mut i = 0u;
|
2011-12-18 19:30:40 +01:00
|
|
|
for f in fields { if f.ident == id { ret some(i); } i += 1u; }
|
|
|
|
ret none;
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
|
2012-02-09 14:33:00 +01:00
|
|
|
fn get_field(rec_ty: t, id: ast::ident) -> field {
|
2012-02-21 15:59:48 +01:00
|
|
|
alt check vec::find(get_fields(rec_ty), {|f| str::eq(f.ident, id) }) {
|
2012-02-09 14:33:00 +01:00
|
|
|
some(f) { f }
|
2012-01-30 21:00:57 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-02-09 14:33:00 +01:00
|
|
|
fn get_fields(rec_ty:t) -> [field] {
|
2012-02-21 15:59:48 +01:00
|
|
|
alt check get(rec_ty).struct {
|
2012-02-09 14:33:00 +01:00
|
|
|
ty_rec(fields) { fields }
|
2011-09-13 12:14:30 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-01-31 17:05:20 -08:00
|
|
|
fn method_idx(id: ast::ident, meths: [method]) -> option<uint> {
|
2012-03-15 09:47:03 -04:00
|
|
|
let mut i = 0u;
|
2011-12-18 19:30:40 +01:00
|
|
|
for m in meths { if m.ident == id { ret some(i); } i += 1u; }
|
|
|
|
ret none;
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn sort_methods(meths: [method]) -> [method] {
|
|
|
|
fn method_lteq(a: method, b: method) -> bool {
|
2012-02-03 03:28:49 -08:00
|
|
|
ret str::le(a.ident, b.ident);
|
2011-03-17 11:40:05 -07:00
|
|
|
}
|
2011-12-23 16:09:52 +01:00
|
|
|
ret std::sort::merge_sort(bind method_lteq(_, _), meths);
|
2011-03-17 11:40:05 -07:00
|
|
|
}
|
|
|
|
|
2012-03-10 20:35:41 -08:00
|
|
|
fn occurs_check(tcx: ctxt, sp: span, vid: int, rt: t) {
|
2012-02-03 15:15:28 +01:00
|
|
|
// Fast path
|
2012-03-10 20:35:41 -08:00
|
|
|
if !type_has_vars(rt) { ret; }
|
2011-08-19 15:16:48 -07:00
|
|
|
|
2011-08-03 18:06:57 -07:00
|
|
|
// Occurs check!
|
2012-03-12 16:31:22 +01:00
|
|
|
if vec::contains(vars_in_type(rt), vid) {
|
2011-08-03 18:06:57 -07:00
|
|
|
// Maybe this should be span_err -- however, there's an
|
|
|
|
// assertion later on that the type doesn't contain
|
|
|
|
// variables, so in this case we have to be sure to die.
|
2011-09-12 12:39:38 +02:00
|
|
|
tcx.sess.span_fatal
|
2012-03-10 20:35:41 -08:00
|
|
|
(sp, "type inference failed because I \
|
2011-09-12 12:39:38 +02:00
|
|
|
could not find a type\n that's both of the form "
|
2012-02-03 15:15:28 +01:00
|
|
|
+ ty_to_str(tcx, mk_var(tcx, vid)) +
|
2011-09-12 12:39:38 +02:00
|
|
|
" and of the form " + ty_to_str(tcx, rt) +
|
2012-03-05 16:27:27 -08:00
|
|
|
" - such a type would have to be infinitely large.");
|
2012-03-10 20:35:41 -08:00
|
|
|
}
|
2011-08-03 18:06:57 -07:00
|
|
|
}
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2012-02-02 16:50:17 -08:00
|
|
|
// Maintains a little union-set tree for inferred modes. `canon()` returns
|
|
|
|
// the current head value for `m0`.
|
|
|
|
fn canon<T:copy>(tbl: hashmap<ast::node_id, ast::inferable<T>>,
|
|
|
|
m0: ast::inferable<T>) -> ast::inferable<T> {
|
|
|
|
alt m0 {
|
|
|
|
ast::infer(id) {
|
|
|
|
alt tbl.find(id) {
|
|
|
|
none { m0 }
|
|
|
|
some(m1) {
|
|
|
|
let cm1 = canon(tbl, m1);
|
|
|
|
// path compression:
|
|
|
|
if cm1 != m1 { tbl.insert(id, cm1); }
|
|
|
|
cm1
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_ { m0 }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Maintains a little union-set tree for inferred modes. `resolve_mode()`
|
|
|
|
// returns the current head value for `m0`.
|
|
|
|
fn canon_mode(cx: ctxt, m0: ast::mode) -> ast::mode {
|
|
|
|
canon(cx.inferred_modes, m0)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Returns the head value for mode, failing if `m` was a infer(_) that
|
|
|
|
// was never inferred. This should be safe for use after typeck.
|
|
|
|
fn resolved_mode(cx: ctxt, m: ast::mode) -> ast::rmode {
|
|
|
|
alt canon_mode(cx, m) {
|
|
|
|
ast::infer(_) {
|
|
|
|
cx.sess.bug(#fmt["mode %? was never resolved", m]);
|
|
|
|
}
|
|
|
|
ast::expl(m0) { m0 }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn arg_mode(cx: ctxt, a: arg) -> ast::rmode { resolved_mode(cx, a.mode) }
|
2012-02-02 16:50:17 -08:00
|
|
|
|
|
|
|
// Unifies `m1` and `m2`. Returns unified value or failure code.
|
|
|
|
fn unify_mode(cx: ctxt, m1: ast::mode, m2: ast::mode)
|
2012-03-13 14:39:28 -07:00
|
|
|
-> result<ast::mode, type_err> {
|
2012-02-02 16:50:17 -08:00
|
|
|
alt (canon_mode(cx, m1), canon_mode(cx, m2)) {
|
|
|
|
(m1, m2) if (m1 == m2) {
|
|
|
|
result::ok(m1)
|
|
|
|
}
|
|
|
|
(ast::infer(id1), ast::infer(id2)) {
|
|
|
|
cx.inferred_modes.insert(id2, m1);
|
|
|
|
result::ok(m1)
|
|
|
|
}
|
|
|
|
(ast::infer(id), m) | (m, ast::infer(id)) {
|
|
|
|
cx.inferred_modes.insert(id, m);
|
|
|
|
result::ok(m1)
|
|
|
|
}
|
|
|
|
(m1, m2) {
|
|
|
|
result::err(terr_mode_mismatch(m1, m2))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// If `m` was never unified, unifies it with `m_def`. Returns the final value
|
|
|
|
// for `m`.
|
|
|
|
fn set_default_mode(cx: ctxt, m: ast::mode, m_def: ast::rmode) {
|
|
|
|
alt canon_mode(cx, m) {
|
|
|
|
ast::infer(id) {
|
|
|
|
cx.inferred_modes.insert(id, ast::expl(m_def));
|
|
|
|
}
|
|
|
|
ast::expl(_) { }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-02-18 14:52:33 -08:00
|
|
|
// Type unification via Robinson's algorithm (Robinson 1965). Implemented as
|
|
|
|
// described in Hoder and Voronkov:
|
|
|
|
//
|
|
|
|
// http://www.cs.man.ac.uk/~hoderk/ubench/unification_full.pdf
|
2011-05-14 21:13:47 -04:00
|
|
|
mod unify {
|
2012-03-13 20:46:16 -04:00
|
|
|
import result::{result, ok, err, chain, map, map2};
|
|
|
|
|
2012-03-24 09:20:20 -07:00
|
|
|
export mk_region_bindings;
|
|
|
|
export region_bindings;
|
|
|
|
export precise, in_region_bindings;
|
2011-06-19 02:34:20 -07:00
|
|
|
|
2012-03-13 20:46:16 -04:00
|
|
|
type ures<T> = result<T,type_err>;
|
|
|
|
|
2012-03-21 13:28:50 -07:00
|
|
|
type region_bindings =
|
|
|
|
{sets: ufind::ufind, regions: smallintmap::smallintmap<region>};
|
2011-05-20 18:36:35 -07:00
|
|
|
|
2012-01-19 14:24:03 -08:00
|
|
|
enum unify_style {
|
2012-01-19 17:56:05 -08:00
|
|
|
precise,
|
2012-03-24 09:20:20 -07:00
|
|
|
in_region_bindings(@region_bindings)
|
2011-12-29 13:12:52 +01:00
|
|
|
}
|
2012-02-03 15:15:28 +01:00
|
|
|
type uctxt = {st: unify_style, tcx: ctxt};
|
2011-05-18 11:53:26 -07:00
|
|
|
|
2012-03-21 13:28:50 -07:00
|
|
|
fn mk_region_bindings() -> @region_bindings {
|
|
|
|
ret @{sets: ufind::make(), regions: smallintmap::mk::<region>()};
|
|
|
|
}
|
|
|
|
|
|
|
|
// Unifies two region sets.
|
|
|
|
//
|
|
|
|
// FIXME: This is a straight copy of the code above. We should use
|
|
|
|
// polymorphism to make this better.
|
|
|
|
fn union_region_sets<T:copy>(
|
|
|
|
cx: @uctxt, set_a: uint, set_b: uint,
|
2012-03-24 09:20:20 -07:00
|
|
|
nxt: fn() -> ures<T>) -> ures<T> {
|
2012-03-21 13:28:50 -07:00
|
|
|
|
|
|
|
let rb = alt cx.st {
|
2012-03-24 09:20:20 -07:00
|
|
|
in_region_bindings(rb) { rb }
|
|
|
|
precise {
|
2012-03-21 13:28:50 -07:00
|
|
|
cx.tcx.sess.bug("attempted to unify two region sets without \
|
|
|
|
a set of region bindings present");
|
|
|
|
}
|
|
|
|
};
|
|
|
|
ufind::grow(rb.sets, uint::max(set_a, set_b) + 1u);
|
|
|
|
let root_a = ufind::find(rb.sets, set_a);
|
|
|
|
let root_b = ufind::find(rb.sets, set_b);
|
|
|
|
|
|
|
|
let replace_region = (
|
|
|
|
fn@(rb: @region_bindings, r: region) {
|
|
|
|
ufind::union(rb.sets, set_a, set_b);
|
|
|
|
let root_c: uint = ufind::find(rb.sets, set_a);
|
2012-03-23 11:37:10 -07:00
|
|
|
smallintmap::insert(rb.regions, root_c, r);
|
2012-03-21 13:28:50 -07:00
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
alt smallintmap::find(rb.regions, root_a) {
|
|
|
|
none {
|
|
|
|
alt smallintmap::find(rb.regions, root_b) {
|
|
|
|
none { ufind::union(rb.sets, set_a, set_b); ret nxt(); }
|
|
|
|
some(r_b) { replace_region(rb, r_b); ret nxt(); }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
some(r_a) {
|
|
|
|
alt smallintmap::find(rb.regions, root_b) {
|
|
|
|
none { replace_region(rb, r_a); ret nxt(); }
|
|
|
|
some(r_b) {
|
2012-03-24 09:20:20 -07:00
|
|
|
ret unify_regions(cx, r_a, r_b) {|r_c|
|
2012-03-21 13:28:50 -07:00
|
|
|
replace_region(rb, r_c);
|
|
|
|
nxt()
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-03-21 15:59:57 -07:00
|
|
|
fn record_region_binding<T:copy>(
|
|
|
|
cx: @uctxt, key: uint,
|
2012-03-24 09:20:20 -07:00
|
|
|
r: region,
|
2012-03-21 16:12:12 -07:00
|
|
|
nxt: fn(region) -> ures<T>) -> ures<T> {
|
2012-03-21 15:59:57 -07:00
|
|
|
|
|
|
|
let rb = alt cx.st {
|
2012-03-24 09:20:20 -07:00
|
|
|
in_region_bindings(rb) { rb }
|
|
|
|
precise { fail; }
|
2012-03-21 15:59:57 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
ufind::grow(rb.sets, key + 1u);
|
|
|
|
let root = ufind::find(rb.sets, key);
|
|
|
|
let mut result_region = r;
|
|
|
|
alt smallintmap::find(rb.regions, root) {
|
|
|
|
some(old_region) {
|
2012-03-24 09:20:20 -07:00
|
|
|
alt unify_regions(cx, old_region, r, {|v| ok(v)}) {
|
2012-03-21 15:59:57 -07:00
|
|
|
ok(unified_region) { result_region = unified_region; }
|
|
|
|
err(e) { ret err(e); }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
none {/* fall through */ }
|
|
|
|
}
|
|
|
|
smallintmap::insert(rb.regions, root, result_region);
|
|
|
|
|
|
|
|
// FIXME: This should be re_var instead.
|
2012-03-21 16:12:12 -07:00
|
|
|
ret nxt(re_param(key));
|
2012-03-21 15:59:57 -07:00
|
|
|
}
|
|
|
|
|
2012-03-13 20:46:16 -04:00
|
|
|
fn unify_regions<T:copy>(
|
2012-03-24 18:30:53 -07:00
|
|
|
_cx: @uctxt, _e_region: region, _a_region: region,
|
|
|
|
_nxt: fn(region) -> ures<T>) -> ures<T> {
|
|
|
|
fail; // unused
|
2012-03-13 20:46:16 -04:00
|
|
|
}
|
2011-12-22 14:52:30 +01:00
|
|
|
}
|
|
|
|
|
2012-03-15 19:19:32 -07:00
|
|
|
fn type_err_to_str(cx: ctxt, err: type_err) -> str {
|
2011-07-27 14:19:39 +02:00
|
|
|
alt err {
|
2012-01-18 22:37:22 -08:00
|
|
|
terr_mismatch { ret "types differ"; }
|
2011-09-14 11:01:42 +02:00
|
|
|
terr_ret_style_mismatch(expect, actual) {
|
|
|
|
fn to_str(s: ast::ret_style) -> str {
|
|
|
|
alt s {
|
2012-01-18 22:37:22 -08:00
|
|
|
ast::noreturn { "non-returning" }
|
2012-01-19 01:03:57 -08:00
|
|
|
ast::return_val { "return-by-value" }
|
2011-09-14 11:01:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
ret to_str(actual) + " function found where " + to_str(expect) +
|
|
|
|
" function was expected";
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2012-03-22 20:06:01 -07:00
|
|
|
terr_proto_mismatch(e, a) {
|
|
|
|
ret #fmt["closure protocol mismatch (%s vs %s)",
|
|
|
|
proto_to_str(e), proto_to_str(a)];
|
|
|
|
}
|
|
|
|
terr_mutability { ret "values differ in mutability"; }
|
2012-01-18 22:37:22 -08:00
|
|
|
terr_box_mutability { ret "boxed values differ in mutability"; }
|
|
|
|
terr_vec_mutability { ret "vectors differ in mutability"; }
|
2012-02-21 17:02:02 +01:00
|
|
|
terr_ptr_mutability { ret "pointers differ in mutability"; }
|
2012-03-08 15:54:36 -08:00
|
|
|
terr_ref_mutability { ret "references differ in mutability"; }
|
2012-03-13 20:46:16 -04:00
|
|
|
terr_ty_param_size(e_sz, a_sz) {
|
|
|
|
ret "expected a type with " + uint::to_str(e_sz, 10u) +
|
|
|
|
" type params but found one with " + uint::to_str(a_sz, 10u) +
|
|
|
|
" type params";
|
|
|
|
}
|
2011-08-15 11:40:26 +02:00
|
|
|
terr_tuple_size(e_sz, a_sz) {
|
2011-09-02 15:34:58 -07:00
|
|
|
ret "expected a tuple with " + uint::to_str(e_sz, 10u) +
|
|
|
|
" elements but found one with " + uint::to_str(a_sz, 10u) +
|
|
|
|
" elements";
|
2011-08-15 11:40:26 +02:00
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
terr_record_size(e_sz, a_sz) {
|
2011-09-02 15:34:58 -07:00
|
|
|
ret "expected a record with " + uint::to_str(e_sz, 10u) +
|
|
|
|
" fields but found one with " + uint::to_str(a_sz, 10u) +
|
|
|
|
" fields";
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2012-01-18 22:37:22 -08:00
|
|
|
terr_record_mutability { ret "record elements differ in mutability"; }
|
2011-07-27 14:19:39 +02:00
|
|
|
terr_record_fields(e_fld, a_fld) {
|
2011-09-02 15:34:58 -07:00
|
|
|
ret "expected a record with field '" + e_fld +
|
|
|
|
"' but found one with field '" + a_fld + "'";
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2012-01-18 22:37:22 -08:00
|
|
|
terr_arg_count { ret "incorrect number of function parameters"; }
|
2011-07-27 14:19:39 +02:00
|
|
|
terr_mode_mismatch(e_mode, a_mode) {
|
2012-02-02 16:50:17 -08:00
|
|
|
ret "expected argument mode " + mode_to_str(e_mode) + " but found " +
|
|
|
|
mode_to_str(a_mode);
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
terr_constr_len(e_len, a_len) {
|
2012-03-05 16:27:27 -08:00
|
|
|
ret "expected a type with " + uint::str(e_len) +
|
2011-09-02 15:34:58 -07:00
|
|
|
" constraints, but found one with " + uint::str(a_len) +
|
|
|
|
" constraints";
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
terr_constr_mismatch(e_constr, a_constr) {
|
2012-03-05 16:27:27 -08:00
|
|
|
ret "expected a type with constraint " + ty_constr_to_str(e_constr) +
|
2011-09-02 15:34:58 -07:00
|
|
|
" but found one with constraint " +
|
|
|
|
ty_constr_to_str(a_constr);
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2012-03-15 19:19:32 -07:00
|
|
|
terr_regions_differ(true, region_a, region_b) {
|
|
|
|
ret #fmt("reference lifetime %s does not match reference lifetime %s",
|
|
|
|
region_to_str(cx, region_a), region_to_str(cx, region_b));
|
|
|
|
}
|
|
|
|
terr_regions_differ(false, subregion, superregion) {
|
|
|
|
ret #fmt("references with lifetime %s do not outlive references with \
|
|
|
|
lifetime %s",
|
|
|
|
region_to_str(cx, subregion),
|
|
|
|
region_to_str(cx, superregion));
|
2012-03-11 13:28:43 -07:00
|
|
|
}
|
2010-12-21 17:47:13 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-05-20 18:36:35 -07:00
|
|
|
// Replaces type parameters in the given type using the given list of
|
|
|
|
// substitions.
|
2011-09-12 11:27:30 +02:00
|
|
|
fn substitute_type_params(cx: ctxt, substs: [ty::t], typ: t) -> t {
|
2012-03-12 10:05:15 +01:00
|
|
|
if !type_has_params(typ) { ret typ; }
|
2012-01-14 16:05:07 -08:00
|
|
|
// Precondition? idx < vec::len(substs)
|
2012-01-30 11:52:34 +01:00
|
|
|
fold_ty(cx, fm_param({|idx, _id| substs[idx]}), typ)
|
2011-04-08 21:27:54 -07:00
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn def_has_ty_params(def: ast::def) -> bool {
|
2011-07-27 14:19:39 +02:00
|
|
|
alt def {
|
2012-01-26 10:29:47 +01:00
|
|
|
ast::def_fn(_, _) | ast::def_variant(_, _) { true }
|
2012-02-07 11:25:04 +01:00
|
|
|
_ { false }
|
2011-03-30 17:23:25 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-12-22 08:45:18 +01:00
|
|
|
fn store_iface_methods(cx: ctxt, id: ast::node_id, ms: @[method]) {
|
|
|
|
cx.iface_method_cache.insert(ast_util::local_def(id), ms);
|
|
|
|
}
|
|
|
|
|
|
|
|
fn iface_methods(cx: ctxt, id: ast::def_id) -> @[method] {
|
|
|
|
alt cx.iface_method_cache.find(id) {
|
|
|
|
some(ms) { ret ms; }
|
|
|
|
_ {}
|
|
|
|
}
|
|
|
|
// Local interfaces are supposed to have been added explicitly.
|
|
|
|
assert id.crate != ast::local_crate;
|
2012-01-05 13:57:27 +01:00
|
|
|
let result = csearch::get_iface_methods(cx, id);
|
2011-12-22 08:45:18 +01:00
|
|
|
cx.iface_method_cache.insert(id, result);
|
|
|
|
result
|
|
|
|
}
|
2011-05-19 17:21:21 -07:00
|
|
|
|
2012-01-31 17:05:20 -08:00
|
|
|
fn impl_iface(cx: ctxt, id: ast::def_id) -> option<t> {
|
2012-01-05 10:57:19 +01:00
|
|
|
if id.crate == ast::local_crate {
|
2012-03-07 12:54:00 +01:00
|
|
|
alt cx.items.get(id.node) {
|
|
|
|
ast_map::node_item(@{node: ast::item_impl(
|
|
|
|
_, some(@{node: ast::ty_path(_, id), _}), _, _), _}, _) {
|
|
|
|
some(node_id_to_type(cx, id))
|
|
|
|
}
|
|
|
|
_ { none }
|
|
|
|
}
|
2012-01-05 10:57:19 +01:00
|
|
|
} else {
|
|
|
|
csearch::get_impl_iface(cx, id)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-03-08 16:10:25 +01:00
|
|
|
fn ty_to_def_id(ty: t) -> ast::def_id {
|
|
|
|
alt check get(ty).struct {
|
|
|
|
ty_iface(id, _) | ty_class(id, _) | ty_res(id, _, _) | ty_enum(id, _) {
|
|
|
|
id
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-01-25 14:34:31 +01:00
|
|
|
// Enum information
|
2012-02-03 15:15:28 +01:00
|
|
|
type variant_info = @{args: [t], ctor_ty: t, name: str,
|
2012-01-15 21:42:10 -08:00
|
|
|
id: ast::def_id, disr_val: int};
|
2011-07-27 14:19:39 +02:00
|
|
|
|
2012-02-07 21:19:53 -08:00
|
|
|
fn substd_enum_variants(cx: ctxt, id: ast::def_id, tps: [ty::t])
|
|
|
|
-> [variant_info] {
|
|
|
|
vec::map(*enum_variants(cx, id)) { |variant_info|
|
|
|
|
let substd_args = vec::map(variant_info.args) {|aty|
|
|
|
|
substitute_type_params(cx, tps, aty)
|
|
|
|
};
|
|
|
|
|
|
|
|
let substd_ctor_ty =
|
|
|
|
substitute_type_params(cx, tps, variant_info.ctor_ty);
|
|
|
|
|
|
|
|
@{args: substd_args, ctor_ty: substd_ctor_ty with *variant_info}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-02-14 15:21:53 -08:00
|
|
|
fn item_path_str(cx: ctxt, id: ast::def_id) -> str {
|
|
|
|
ast_map::path_to_str(item_path(cx, id))
|
|
|
|
}
|
|
|
|
|
2012-02-10 06:01:32 -08:00
|
|
|
fn item_path(cx: ctxt, id: ast::def_id) -> ast_map::path {
|
|
|
|
if id.crate != ast::local_crate {
|
|
|
|
csearch::get_item_path(cx, id)
|
|
|
|
} else {
|
|
|
|
let node = cx.items.get(id.node);
|
|
|
|
alt node {
|
|
|
|
ast_map::node_item(item, path) {
|
|
|
|
let item_elt = alt item.node {
|
|
|
|
item_mod(_) | item_native_mod(_) {
|
|
|
|
ast_map::path_mod(item.ident)
|
|
|
|
}
|
|
|
|
_ {
|
|
|
|
ast_map::path_name(item.ident)
|
|
|
|
}
|
|
|
|
};
|
|
|
|
*path + [item_elt]
|
|
|
|
}
|
|
|
|
|
2012-03-07 12:21:08 +01:00
|
|
|
ast_map::node_native_item(nitem, _, path) {
|
2012-02-10 06:01:32 -08:00
|
|
|
*path + [ast_map::path_name(nitem.ident)]
|
|
|
|
}
|
|
|
|
|
2012-02-10 16:38:33 -08:00
|
|
|
ast_map::node_method(method, _, path) {
|
2012-02-10 06:01:32 -08:00
|
|
|
*path + [ast_map::path_name(method.ident)]
|
|
|
|
}
|
|
|
|
|
|
|
|
ast_map::node_variant(variant, _, path) {
|
|
|
|
vec::init(*path) + [ast_map::path_name(variant.node.name)]
|
|
|
|
}
|
|
|
|
|
2012-03-06 08:02:13 -08:00
|
|
|
ast_map::node_ctor(i, path) {
|
|
|
|
*path + [ast_map::path_name(i.ident)]
|
2012-03-12 16:31:22 +01:00
|
|
|
}
|
|
|
|
|
2012-02-10 06:01:32 -08:00
|
|
|
ast_map::node_expr(_) | ast_map::node_arg(_, _) |
|
2012-03-15 18:46:18 -07:00
|
|
|
ast_map::node_local(_) | ast_map::node_export(_, _) |
|
|
|
|
ast_map::node_block(_) {
|
2012-02-10 06:01:32 -08:00
|
|
|
cx.sess.bug(#fmt["cannot find item_path for node %?", node]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-01-25 14:34:31 +01:00
|
|
|
fn enum_variants(cx: ctxt, id: ast::def_id) -> @[variant_info] {
|
|
|
|
alt cx.enum_var_cache.find(id) {
|
2011-12-22 16:23:49 +01:00
|
|
|
some(variants) { ret variants; }
|
2011-12-15 16:27:18 -08:00
|
|
|
_ { /* fallthrough */ }
|
|
|
|
}
|
2011-12-22 16:23:49 +01:00
|
|
|
let result = if ast::local_crate != id.crate {
|
2012-01-25 14:34:31 +01:00
|
|
|
@csearch::get_enum_variants(cx, id)
|
2011-12-22 16:23:49 +01:00
|
|
|
} else {
|
2012-01-16 02:36:47 -07:00
|
|
|
// FIXME: Now that the variants are run through the type checker (to
|
2012-01-16 21:04:02 -07:00
|
|
|
// check the disr_expr if it exists), this code should likely be
|
|
|
|
// moved there to avoid having to call eval_const_expr twice.
|
2011-12-22 16:23:49 +01:00
|
|
|
alt cx.items.get(id.node) {
|
2012-02-03 09:53:37 +01:00
|
|
|
ast_map::node_item(@{node: ast::item_enum(variants, _), _}, _) {
|
2012-03-15 09:47:03 -04:00
|
|
|
let mut disr_val = -1;
|
2011-12-22 08:45:18 +01:00
|
|
|
@vec::map(variants, {|variant|
|
2012-01-30 17:28:30 +01:00
|
|
|
let ctor_ty = node_id_to_type(cx, variant.node.id);
|
2011-12-22 08:45:18 +01:00
|
|
|
let arg_tys = if vec::len(variant.node.args) > 0u {
|
2012-02-03 15:15:28 +01:00
|
|
|
vec::map(ty_fn_args(ctor_ty), {|a| a.ty})
|
2011-12-22 08:45:18 +01:00
|
|
|
} else { [] };
|
2012-01-16 02:36:47 -07:00
|
|
|
alt variant.node.disr_expr {
|
|
|
|
some (ex) {
|
|
|
|
// FIXME: issue #1417
|
2012-03-22 14:56:56 -07:00
|
|
|
disr_val = alt const_eval::eval_const_expr(cx, ex) {
|
|
|
|
const_eval::const_int(val) {val as int}
|
2012-01-30 21:00:57 -08:00
|
|
|
_ { cx.sess.bug("tag_variants: bad disr expr"); }
|
2012-01-16 02:36:47 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
_ {disr_val += 1;}
|
|
|
|
}
|
2011-12-22 08:45:18 +01:00
|
|
|
@{args: arg_tys,
|
|
|
|
ctor_ty: ctor_ty,
|
2012-01-15 21:42:10 -08:00
|
|
|
name: variant.node.name,
|
2012-01-10 14:50:40 -07:00
|
|
|
id: ast_util::local_def(variant.node.id),
|
2012-01-16 02:36:47 -07:00
|
|
|
disr_val: disr_val
|
2012-01-10 14:50:40 -07:00
|
|
|
}
|
2011-12-22 08:45:18 +01:00
|
|
|
})
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2012-01-30 21:00:57 -08:00
|
|
|
_ { cx.sess.bug("tag_variants: id not bound to an enum"); }
|
2011-05-19 17:21:21 -07:00
|
|
|
}
|
2011-12-22 16:23:49 +01:00
|
|
|
};
|
2012-01-25 14:34:31 +01:00
|
|
|
cx.enum_var_cache.insert(id, result);
|
2011-12-22 16:23:49 +01:00
|
|
|
result
|
2011-05-19 17:21:21 -07:00
|
|
|
}
|
|
|
|
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2012-01-19 14:24:03 -08:00
|
|
|
// Returns information about the enum variant with the given ID:
|
2012-01-25 14:34:31 +01:00
|
|
|
fn enum_variant_with_id(cx: ctxt, enum_id: ast::def_id,
|
|
|
|
variant_id: ast::def_id) -> variant_info {
|
|
|
|
let variants = enum_variants(cx, enum_id);
|
2012-03-15 09:47:03 -04:00
|
|
|
let mut i = 0u;
|
2011-12-15 17:14:58 -08:00
|
|
|
while i < vec::len::<variant_info>(*variants) {
|
2011-08-19 15:16:48 -07:00
|
|
|
let variant = variants[i];
|
2011-07-27 14:19:39 +02:00
|
|
|
if def_eq(variant.id, variant_id) { ret variant; }
|
2011-05-19 17:21:21 -07:00
|
|
|
i += 1u;
|
|
|
|
}
|
2012-01-25 14:34:31 +01:00
|
|
|
cx.sess.bug("enum_variant_with_id(): no variant exists with that ID");
|
2011-05-19 17:21:21 -07:00
|
|
|
}
|
|
|
|
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2011-03-30 17:23:25 -07:00
|
|
|
// If the given item is in an external crate, looks up its type and adds it to
|
|
|
|
// the type cache. Returns the type parameters and type.
|
2011-12-28 17:50:12 +01:00
|
|
|
fn lookup_item_type(cx: ctxt, did: ast::def_id) -> ty_param_bounds_and_ty {
|
2012-03-06 08:02:13 -08:00
|
|
|
/*
|
|
|
|
Are we putting class ids in the tcache (where does that happen?)
|
|
|
|
*/
|
2011-07-27 14:19:39 +02:00
|
|
|
alt cx.tcache.find(did) {
|
|
|
|
some(tpt) { ret tpt; }
|
2012-01-18 22:37:22 -08:00
|
|
|
none {
|
2012-03-06 08:02:13 -08:00
|
|
|
#debug("lookup_item_type: looking up %?", did);
|
2012-02-02 12:37:17 +01:00
|
|
|
// The item is in this crate. The caller should have added it to the
|
|
|
|
// type cache already
|
|
|
|
assert did.crate != ast::local_crate;
|
2011-07-27 14:19:39 +02:00
|
|
|
let tyt = csearch::get_type(cx, did);
|
|
|
|
cx.tcache.insert(did, tyt);
|
|
|
|
ret tyt;
|
|
|
|
}
|
2011-03-30 17:23:25 -07:00
|
|
|
}
|
2012-03-03 17:49:23 -08:00
|
|
|
}
|
|
|
|
|
2012-03-19 10:19:00 -07:00
|
|
|
// Look up a field ID, whether or not it's local
|
|
|
|
fn lookup_field_type(tcx: ctxt, class_id: def_id, id: def_id) -> ty::t {
|
|
|
|
if id.crate == ast::local_crate {
|
|
|
|
node_id_to_type(tcx, id.node)
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
alt tcx.tcache.find(id) {
|
|
|
|
some(tpt) { ret tpt.ty; }
|
|
|
|
none {
|
|
|
|
let tpt = csearch::get_field_type(tcx, class_id, id);
|
|
|
|
// ok b/c fields are monomorphic
|
|
|
|
// TODO: Comment might be a lie, what if it mentions
|
|
|
|
// class-bound ty params?
|
|
|
|
tcx.tcache.insert(id, tpt);
|
|
|
|
ret tpt.ty;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-03-21 12:42:34 -07:00
|
|
|
// Look up the list of field names and IDs for a given class
|
2012-03-03 17:49:23 -08:00
|
|
|
// Fails if the id is not bound to a class.
|
2012-03-19 10:19:00 -07:00
|
|
|
fn lookup_class_fields(cx: ctxt, did: ast::def_id) -> [field_ty] {
|
2012-03-16 20:36:07 -07:00
|
|
|
if did.crate == ast::local_crate {
|
2012-03-03 17:49:23 -08:00
|
|
|
alt cx.items.find(did.node) {
|
|
|
|
some(ast_map::node_item(i,_)) {
|
2012-03-06 08:02:13 -08:00
|
|
|
alt i.node {
|
|
|
|
ast::item_class(_, items, _) {
|
2012-03-19 10:19:00 -07:00
|
|
|
class_field_tys(items)
|
2012-03-03 17:49:23 -08:00
|
|
|
}
|
2012-03-06 08:02:13 -08:00
|
|
|
_ { cx.sess.bug("class ID bound to non-class"); }
|
|
|
|
}
|
2012-03-03 17:49:23 -08:00
|
|
|
}
|
|
|
|
_ { cx.sess.bug("class ID not bound to an item"); }
|
|
|
|
}
|
2012-03-06 08:02:13 -08:00
|
|
|
}
|
2012-03-16 20:36:07 -07:00
|
|
|
else {
|
2012-03-19 10:19:00 -07:00
|
|
|
ret csearch::get_class_fields(cx, did);
|
2012-03-06 08:02:13 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-03-27 22:08:48 -07:00
|
|
|
fn lookup_class_field(cx: ctxt, parent: ast::def_id, field_id: ast::def_id)
|
|
|
|
-> field_ty {
|
|
|
|
alt vec::find(lookup_class_fields(cx, parent))
|
|
|
|
{|f| f.id.node == field_id.node} {
|
|
|
|
some(t) { t }
|
|
|
|
none { cx.sess.bug("class ID not found in parent's fields"); }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-03-26 09:59:59 -07:00
|
|
|
fn lookup_public_fields(cx: ctxt, did: ast::def_id) -> [field_ty] {
|
|
|
|
vec::filter(lookup_class_fields(cx, did), is_public)
|
|
|
|
}
|
|
|
|
|
|
|
|
pure fn is_public(f: field_ty) -> bool {
|
|
|
|
alt f.privacy {
|
|
|
|
pub { true }
|
|
|
|
priv { false }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-03-21 12:42:34 -07:00
|
|
|
// Look up the list of method names and IDs for a given class
|
|
|
|
// Fails if the id is not bound to a class.
|
|
|
|
fn lookup_class_method_ids(cx: ctxt, did: ast::def_id)
|
2012-03-26 09:59:59 -07:00
|
|
|
: is_local(did) -> [{name: ident, id: node_id, privacy: privacy}] {
|
2012-03-21 12:42:34 -07:00
|
|
|
alt cx.items.find(did.node) {
|
|
|
|
some(ast_map::node_item(@{node: item_class(_,items,_), _}, _)) {
|
|
|
|
let (_,ms) = split_class_items(items);
|
2012-03-28 18:50:33 -07:00
|
|
|
vec::map(ms, {|m| {name: m.ident, id: m.id,
|
2012-03-26 09:59:59 -07:00
|
|
|
privacy: m.privacy}})
|
2012-03-21 12:42:34 -07:00
|
|
|
}
|
|
|
|
_ {
|
|
|
|
cx.sess.bug("lookup_class_method_ids: id not bound to a class");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Given a class def_id and a method name, return the method's
|
2012-03-26 09:59:59 -07:00
|
|
|
def_id. Needed so we can do static dispatch for methods
|
2012-03-28 18:50:33 -07:00
|
|
|
Doesn't care about the method's privacy. (It's assumed that
|
|
|
|
the caller already checked that.)
|
|
|
|
*/
|
2012-03-21 12:42:34 -07:00
|
|
|
fn lookup_class_method_by_name(cx:ctxt, did: ast::def_id, name: ident,
|
2012-03-26 09:59:59 -07:00
|
|
|
sp: span) -> def_id {
|
2012-03-21 12:42:34 -07:00
|
|
|
if check is_local(did) {
|
|
|
|
let ms = lookup_class_method_ids(cx, did);
|
|
|
|
for m in ms {
|
2012-03-28 18:50:33 -07:00
|
|
|
if m.name == name {
|
2012-03-21 12:42:34 -07:00
|
|
|
ret ast_util::local_def(m.id);
|
|
|
|
}
|
|
|
|
}
|
2012-03-28 18:50:33 -07:00
|
|
|
cx.sess.span_fatal(sp, #fmt("Class doesn't have a method \
|
2012-03-26 09:59:59 -07:00
|
|
|
named %s", name));
|
2012-03-21 12:42:34 -07:00
|
|
|
}
|
|
|
|
else {
|
2012-03-22 18:03:12 -07:00
|
|
|
csearch::get_class_method(cx.sess.cstore, did, name)
|
2012-03-21 12:42:34 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-03-28 18:50:33 -07:00
|
|
|
fn class_field_tys(items: [@class_member]) -> [field_ty] {
|
2012-03-15 09:47:03 -04:00
|
|
|
let mut rslt = [];
|
2012-03-06 08:02:13 -08:00
|
|
|
for it in items {
|
2012-03-28 18:50:33 -07:00
|
|
|
alt it.node {
|
|
|
|
instance_var(nm, _, cm, id, privacy) {
|
2012-03-26 09:59:59 -07:00
|
|
|
rslt += [{ident: nm, id: ast_util::local_def(id),
|
2012-03-28 18:50:33 -07:00
|
|
|
privacy: privacy, mutability: cm}];
|
2012-03-06 08:02:13 -08:00
|
|
|
}
|
2012-03-28 18:50:33 -07:00
|
|
|
class_method(_) { }
|
2012-03-06 08:02:13 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
rslt
|
2012-03-03 17:49:23 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Return a list of fields corresponding to the class's items
|
|
|
|
// (as if the class was a record). trans uses this
|
|
|
|
fn class_items_as_fields(cx:ctxt, did: ast::def_id) -> [field] {
|
2012-03-15 09:47:03 -04:00
|
|
|
let mut rslt = [];
|
2012-03-19 10:19:00 -07:00
|
|
|
for f in lookup_class_fields(cx, did) {
|
2012-03-26 18:35:18 -07:00
|
|
|
// consider all instance vars mut, because the
|
2012-03-19 10:19:00 -07:00
|
|
|
// constructor may mutate all vars
|
|
|
|
rslt += [{ident: f.ident, mt: {ty: lookup_field_type(cx, did, f.id),
|
|
|
|
mutbl: m_mutbl}}];
|
2012-03-03 17:49:23 -08:00
|
|
|
}
|
|
|
|
rslt
|
|
|
|
}
|
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn is_binopable(_cx: ctxt, ty: t, op: ast::binop) -> bool {
|
2011-07-27 14:19:39 +02:00
|
|
|
const tycat_other: int = 0;
|
|
|
|
const tycat_bool: int = 1;
|
|
|
|
const tycat_int: int = 2;
|
|
|
|
const tycat_float: int = 3;
|
|
|
|
const tycat_str: int = 4;
|
|
|
|
const tycat_vec: int = 5;
|
|
|
|
const tycat_struct: int = 6;
|
2011-08-02 18:04:24 -07:00
|
|
|
const tycat_bot: int = 7;
|
2011-07-27 14:19:39 +02:00
|
|
|
|
|
|
|
const opcat_add: int = 0;
|
|
|
|
const opcat_sub: int = 1;
|
|
|
|
const opcat_mult: int = 2;
|
|
|
|
const opcat_shift: int = 3;
|
|
|
|
const opcat_rel: int = 4;
|
|
|
|
const opcat_eq: int = 5;
|
|
|
|
const opcat_bit: int = 6;
|
|
|
|
const opcat_logic: int = 7;
|
|
|
|
|
|
|
|
fn opcat(op: ast::binop) -> int {
|
|
|
|
alt op {
|
2012-01-18 22:37:22 -08:00
|
|
|
ast::add { opcat_add }
|
|
|
|
ast::subtract { opcat_sub }
|
|
|
|
ast::mul { opcat_mult }
|
|
|
|
ast::div { opcat_mult }
|
|
|
|
ast::rem { opcat_mult }
|
|
|
|
ast::and { opcat_logic }
|
|
|
|
ast::or { opcat_logic }
|
|
|
|
ast::bitxor { opcat_bit }
|
|
|
|
ast::bitand { opcat_bit }
|
|
|
|
ast::bitor { opcat_bit }
|
|
|
|
ast::lsl { opcat_shift }
|
|
|
|
ast::lsr { opcat_shift }
|
|
|
|
ast::asr { opcat_shift }
|
|
|
|
ast::eq { opcat_eq }
|
|
|
|
ast::ne { opcat_eq }
|
|
|
|
ast::lt { opcat_rel }
|
|
|
|
ast::le { opcat_rel }
|
|
|
|
ast::ge { opcat_rel }
|
|
|
|
ast::gt { opcat_rel }
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn tycat(ty: t) -> int {
|
|
|
|
alt get(ty).struct {
|
2012-01-18 22:37:22 -08:00
|
|
|
ty_bool { tycat_bool }
|
2011-12-07 21:06:12 +01:00
|
|
|
ty_int(_) { tycat_int }
|
|
|
|
ty_uint(_) { tycat_int }
|
|
|
|
ty_float(_) { tycat_float }
|
2012-01-18 22:37:22 -08:00
|
|
|
ty_str { tycat_str }
|
2011-08-18 14:11:06 -07:00
|
|
|
ty_vec(_) { tycat_vec }
|
2011-07-27 14:19:39 +02:00
|
|
|
ty_rec(_) { tycat_struct }
|
2011-08-15 11:40:26 +02:00
|
|
|
ty_tup(_) { tycat_struct }
|
2012-01-25 14:34:31 +01:00
|
|
|
ty_enum(_, _) { tycat_struct }
|
2012-01-18 22:37:22 -08:00
|
|
|
ty_bot { tycat_bot }
|
2011-07-27 14:19:39 +02:00
|
|
|
_ { tycat_other }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const t: bool = true;
|
|
|
|
const f: bool = false;
|
2011-06-24 14:19:58 -07:00
|
|
|
|
|
|
|
/*. add, shift, bit
|
|
|
|
. sub, rel, logic
|
|
|
|
. mult, eq, */
|
2011-09-02 15:34:58 -07:00
|
|
|
/*other*/
|
|
|
|
/*bool*/
|
|
|
|
/*int*/
|
|
|
|
/*float*/
|
|
|
|
/*str*/
|
|
|
|
/*vec*/
|
|
|
|
/*bot*/
|
2011-08-19 15:16:48 -07:00
|
|
|
let tbl =
|
|
|
|
[[f, f, f, f, t, t, f, f], [f, f, f, f, t, t, t, t],
|
|
|
|
[t, t, t, t, t, t, t, f], [t, t, t, f, t, t, f, f],
|
|
|
|
[t, f, f, f, t, t, f, f], [t, f, f, f, t, t, f, f],
|
|
|
|
[f, f, f, f, t, t, f, f], [t, t, t, t, t, t, t, t]]; /*struct*/
|
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
ret tbl[tycat(ty)][opcat(op)];
|
2011-06-24 14:19:58 -07:00
|
|
|
}
|
|
|
|
|
2012-02-03 15:15:28 +01:00
|
|
|
fn ast_constr_to_constr<T>(tcx: ctxt, c: @ast::constr_general<T>) ->
|
|
|
|
@constr_general<T> {
|
2011-07-27 14:19:39 +02:00
|
|
|
alt tcx.def_map.find(c.node.id) {
|
2012-01-18 22:37:22 -08:00
|
|
|
some(ast::def_fn(pred_id, ast::pure_fn)) {
|
2011-08-21 21:44:41 -07:00
|
|
|
ret @ast_util::respan(c.span,
|
|
|
|
{path: c.node.path,
|
|
|
|
args: c.node.args,
|
|
|
|
id: pred_id});
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
_ {
|
2011-09-12 11:27:30 +02:00
|
|
|
tcx.sess.span_fatal(c.span,
|
2012-03-05 16:27:27 -08:00
|
|
|
"predicate " + path_to_str(c.node.path) +
|
2011-09-12 12:39:38 +02:00
|
|
|
" is unbound or bound to a non-function or an \
|
2011-09-02 15:34:58 -07:00
|
|
|
impure function");
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2011-07-21 15:59:41 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-03-23 17:52:20 -07:00
|
|
|
|
2011-01-13 17:42:28 -08:00
|
|
|
// Local Variables:
|
|
|
|
// mode: rust
|
|
|
|
// fill-column: 78;
|
|
|
|
// indent-tabs-mode: nil
|
|
|
|
// c-basic-offset: 4
|
|
|
|
// buffer-file-coding-system: utf-8-unix
|
|
|
|
// End:
|