2011-12-13 16:25:51 -08:00
|
|
|
import vec;
|
|
|
|
import str;
|
|
|
|
import uint;
|
2011-08-15 16:21:55 -07:00
|
|
|
import std::ufind;
|
2011-05-12 17:24:54 +02:00
|
|
|
import std::map;
|
|
|
|
import std::map::hashmap;
|
2011-12-13 16:25:51 -08:00
|
|
|
import option;
|
|
|
|
import option::none;
|
|
|
|
import option::some;
|
2011-05-20 18:36:35 -07:00
|
|
|
import std::smallintmap;
|
2011-05-12 17:24:54 +02:00
|
|
|
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;
|
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::*;
|
2011-07-05 11:48:19 +02:00
|
|
|
import syntax::util::interner;
|
|
|
|
import util::ppaux::ty_to_str;
|
2011-07-19 17:52:34 -07:00
|
|
|
import util::ppaux::ty_constr_to_str;
|
2012-01-04 11:32:26 +01:00
|
|
|
import util::ppaux::mode_str;
|
2011-07-19 17:52:34 -07:00
|
|
|
import syntax::print::pprust::*;
|
2011-05-16 13:58:13 -07:00
|
|
|
|
2011-06-19 22:41:21 +02:00
|
|
|
export node_id_to_monotype;
|
|
|
|
export node_id_to_type;
|
|
|
|
export node_id_to_type_params;
|
|
|
|
export node_id_to_ty_param_substs_opt_and_ty;
|
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;
|
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 count_ty_params;
|
|
|
|
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;
|
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;
|
2011-06-19 02:34:20 -07:00
|
|
|
export fm_general;
|
|
|
|
export get_element_type;
|
|
|
|
export idx_nil;
|
2011-06-24 14:19:58 -07:00
|
|
|
export is_binopable;
|
2011-08-24 17:24:58 -07:00
|
|
|
export is_pred_ty;
|
2011-06-19 02:34:20 -07:00
|
|
|
export lookup_item_type;
|
|
|
|
export method;
|
|
|
|
export method_idx;
|
|
|
|
export mk_bool;
|
|
|
|
export mk_bot;
|
|
|
|
export mk_box;
|
|
|
|
export mk_char;
|
2011-07-19 17:52:34 -07:00
|
|
|
export mk_constr;
|
2011-06-19 02:34:20 -07:00
|
|
|
export mk_ctxt;
|
|
|
|
export mk_float;
|
|
|
|
export mk_fn;
|
|
|
|
export mk_imm_box;
|
2011-09-21 18:54:54 -07:00
|
|
|
export mk_imm_uniq;
|
2011-08-02 16:24:38 -07:00
|
|
|
export mk_mut_ptr;
|
2011-06-19 02:34:20 -07:00
|
|
|
export mk_int;
|
2011-09-02 16:52:14 -07:00
|
|
|
export mk_str;
|
2011-08-18 14:32:25 -07:00
|
|
|
export mk_vec;
|
2011-12-07 21:06:12 +01:00
|
|
|
export mk_mach_int;
|
|
|
|
export mk_mach_uint;
|
|
|
|
export mk_mach_float;
|
2011-06-19 02:34:20 -07:00
|
|
|
export mk_native;
|
|
|
|
export mk_native_fn;
|
|
|
|
export mk_nil;
|
2011-12-20 16:33:55 +01:00
|
|
|
export mk_iface;
|
2011-06-24 18:10:40 +02:00
|
|
|
export mk_res;
|
2011-06-19 02:34:20 -07:00
|
|
|
export mk_param;
|
|
|
|
export mk_ptr;
|
|
|
|
export mk_rec;
|
|
|
|
export mk_tag;
|
2011-08-15 11:40:26 +02:00
|
|
|
export mk_tup;
|
2011-06-19 02:34:20 -07:00
|
|
|
export mk_type;
|
2011-12-15 11:06:48 -08:00
|
|
|
export mk_send_type;
|
2011-06-19 02:34:20 -07:00
|
|
|
export mk_uint;
|
2011-08-10 17:23:46 -07:00
|
|
|
export mk_uniq;
|
2011-06-19 02:34:20 -07:00
|
|
|
export mk_var;
|
2012-01-05 16:19:12 -08:00
|
|
|
export mk_opaque_closure_ptr;
|
2011-12-19 13:52:58 +01:00
|
|
|
export mk_named;
|
2011-12-14 15:23:11 +01:00
|
|
|
export gen_ty;
|
2011-06-19 02:34:20 -07:00
|
|
|
export mode;
|
|
|
|
export mt;
|
|
|
|
export node_type_table;
|
|
|
|
export pat_ty;
|
|
|
|
export ret_ty_of_fn;
|
|
|
|
export sequence_element_type;
|
2012-01-16 11:45:18 +01:00
|
|
|
export struct, struct_raw;
|
2011-12-19 13:52:58 +01:00
|
|
|
export ty_name;
|
2011-06-19 02:34:20 -07:00
|
|
|
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;
|
2011-06-19 02:34:20 -07:00
|
|
|
export tag_variants;
|
2012-01-05 10:57:19 +01:00
|
|
|
export iface_methods, store_iface_methods, impl_iface;
|
2011-06-19 02:34:20 -07:00
|
|
|
export tag_variant_with_id;
|
|
|
|
export ty_param_substs_opt_and_ty;
|
2011-12-28 17:50:12 +01:00
|
|
|
export ty_param_bounds_and_ty;
|
2011-06-19 02:34:20 -07:00
|
|
|
export ty_native_fn;
|
|
|
|
export ty_bool;
|
|
|
|
export ty_bot;
|
|
|
|
export ty_box;
|
2011-07-19 17:52:34 -07:00
|
|
|
export ty_constr;
|
2012-01-05 16:19:12 -08:00
|
|
|
export ty_opaque_closure_ptr;
|
2011-07-19 20:15:27 -07:00
|
|
|
export ty_constr_arg;
|
2011-06-19 02:34:20 -07:00
|
|
|
export ty_float;
|
2011-12-23 16:09:52 +01:00
|
|
|
export ty_fn, fn_ty;
|
2011-06-19 02:34:20 -07:00
|
|
|
export ty_fn_proto;
|
|
|
|
export ty_fn_ret;
|
2011-09-14 14:34:50 +02:00
|
|
|
export ty_fn_ret_style;
|
2011-06-19 02:34:20 -07:00
|
|
|
export ty_int;
|
2011-09-02 16:45:00 -07:00
|
|
|
export ty_str;
|
2011-08-18 14:11:06 -07:00
|
|
|
export ty_vec;
|
2011-06-19 02:34:20 -07:00
|
|
|
export ty_native;
|
|
|
|
export ty_nil;
|
2011-12-20 16:33:55 +01:00
|
|
|
export ty_iface;
|
2011-06-24 18:10:40 +02:00
|
|
|
export ty_res;
|
2011-06-19 02:34:20 -07:00
|
|
|
export ty_param;
|
|
|
|
export ty_ptr;
|
|
|
|
export ty_rec;
|
|
|
|
export ty_tag;
|
2011-08-15 11:40:26 +02:00
|
|
|
export ty_tup;
|
2011-06-19 02:34:20 -07:00
|
|
|
export ty_type;
|
2011-12-15 11:06:48 -08:00
|
|
|
export ty_send_type;
|
2011-06-19 02:34:20 -07:00
|
|
|
export ty_uint;
|
2011-08-10 17:23:46 -07:00
|
|
|
export ty_uniq;
|
2011-06-19 02:34:20 -07:00
|
|
|
export ty_var;
|
2011-12-19 13:52:58 +01:00
|
|
|
export ty_named;
|
2012-01-13 10:58:31 +01:00
|
|
|
export same_type;
|
2011-06-19 02:34:20 -07:00
|
|
|
export ty_var_id;
|
|
|
|
export ty_param_substs_opt_and_ty_to_monotype;
|
|
|
|
export ty_fn_args;
|
2011-07-19 17:52:34 -07:00
|
|
|
export type_constr;
|
2011-06-19 02:34:20 -07:00
|
|
|
export type_contains_params;
|
|
|
|
export type_contains_vars;
|
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;
|
|
|
|
export type_has_dynamic_size;
|
2011-07-29 12:53:58 +02:00
|
|
|
export type_needs_drop;
|
2011-06-19 02:34:20 -07:00
|
|
|
export type_is_bool;
|
|
|
|
export type_is_bot;
|
|
|
|
export type_is_box;
|
|
|
|
export type_is_boxed;
|
2011-09-21 14:00:11 -07:00
|
|
|
export type_is_unique_box;
|
2011-11-02 12:15:50 +01:00
|
|
|
export type_is_unsafe_ptr;
|
2011-09-02 16:09:41 +02:00
|
|
|
export type_is_vec;
|
2011-06-19 02:34:20 -07:00
|
|
|
export type_is_fp;
|
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-06-19 02:34:20 -07:00
|
|
|
export type_is_native;
|
|
|
|
export type_is_nil;
|
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-06-20 13:46:44 -07:00
|
|
|
export type_is_str;
|
2011-08-16 12:38:42 -07:00
|
|
|
export type_is_unique;
|
2012-01-10 14:55:54 -07:00
|
|
|
export type_is_tag;
|
2012-01-10 18:33:26 -07:00
|
|
|
export type_is_c_like_enum;
|
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;
|
|
|
|
export unify;
|
|
|
|
export variant_info;
|
|
|
|
export walk_ty;
|
2011-08-03 18:06:57 -07:00
|
|
|
export occurs_check_fails;
|
2011-12-15 11:06:48 -08:00
|
|
|
export closure_kind;
|
2012-01-12 11:39:23 -08:00
|
|
|
export ck_any;
|
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;
|
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
|
|
|
|
2011-07-27 14:19:39 +02:00
|
|
|
type arg = {mode: 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-01-02 12:09:26 +01:00
|
|
|
type method = {ident: ast::ident, tps: @[param_bounds], fty: fn_ty};
|
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
|
|
|
|
2011-07-27 14:19:39 +02:00
|
|
|
type mt = {ty: t, mut: ast::mutability};
|
2011-03-17 17:39:47 -07:00
|
|
|
|
2011-06-15 11:19:50 -07: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.
|
2011-08-12 07:15:18 -07:00
|
|
|
type creader_cache = hashmap<{cnum: int, pos: uint, len: uint}, ty::t>;
|
2011-06-15 11:19:50 -07:00
|
|
|
|
|
|
|
type ctxt =
|
2011-07-27 14:19:39 +02:00
|
|
|
@{ts: @type_store,
|
|
|
|
sess: session::session,
|
|
|
|
def_map: resolve::def_map,
|
|
|
|
node_types: node_type_table,
|
|
|
|
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>,
|
2011-12-28 18:11:33 +01:00
|
|
|
kind_cache: hashmap<t, kind>,
|
2011-12-15 16:27:18 -08:00
|
|
|
ast_ty_to_ty_cache: hashmap<@ast::ty, option::t<t>>,
|
2011-12-29 11:23:35 +01:00
|
|
|
tag_var_cache: hashmap<def_id, @[variant_info]>,
|
2011-12-28 17:50:12 +01:00
|
|
|
iface_method_cache: hashmap<def_id, @[method]>,
|
2012-01-02 12:13:26 +01:00
|
|
|
ty_param_bounds: hashmap<ast::node_id, param_bounds>};
|
2011-06-15 11:19:50 -07:00
|
|
|
|
|
|
|
type ty_ctxt = ctxt;
|
|
|
|
|
2011-05-11 17:05:39 -07:00
|
|
|
// Never construct these manually. These are interned.
|
2011-12-19 13:52:58 +01:00
|
|
|
type raw_t = {struct: sty,
|
|
|
|
hash: uint,
|
|
|
|
has_params: bool,
|
|
|
|
has_vars: bool};
|
2011-05-11 17:05:39 -07:00
|
|
|
|
|
|
|
type t = uint;
|
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_any,
|
|
|
|
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]};
|
|
|
|
|
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,
|
|
|
|
ty_tag(def_id, [t]),
|
|
|
|
ty_box(mt),
|
|
|
|
ty_uniq(mt),
|
|
|
|
ty_vec(mt),
|
|
|
|
ty_ptr(mt),
|
|
|
|
ty_rec([field]),
|
|
|
|
ty_fn(fn_ty),
|
|
|
|
ty_native_fn([arg], t),
|
|
|
|
ty_iface(def_id, [t]),
|
|
|
|
ty_res(def_id, t, [t]),
|
|
|
|
ty_tup([t]),
|
|
|
|
ty_var(int), // type variable
|
|
|
|
|
|
|
|
ty_param(uint, def_id), // fn/enum type param
|
|
|
|
|
|
|
|
ty_type, // type_desc*
|
|
|
|
ty_send_type, // type_desc* that has been cloned into exchange heap
|
|
|
|
ty_native(def_id),
|
|
|
|
ty_constr(t, [@type_constr]),
|
|
|
|
ty_opaque_closure_ptr(closure_kind), // ptr to env for fn, fn@, fn~
|
|
|
|
ty_named(t, @str),
|
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),
|
|
|
|
terr_box_mutability,
|
|
|
|
terr_vec_mutability,
|
|
|
|
terr_tuple_size(uint, uint),
|
|
|
|
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),
|
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 {
|
2011-12-28 18:11:33 +01:00
|
|
|
let 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
|
|
|
|
2011-07-27 14:19:39 +02:00
|
|
|
const idx_nil: uint = 0u;
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2011-07-27 14:19:39 +02:00
|
|
|
const idx_bool: uint = 1u;
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2011-07-27 14:19:39 +02:00
|
|
|
const idx_int: uint = 2u;
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2011-07-27 14:19:39 +02:00
|
|
|
const idx_float: uint = 3u;
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2011-07-27 14:19:39 +02:00
|
|
|
const idx_uint: uint = 4u;
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2011-07-27 14:19:39 +02:00
|
|
|
const idx_i8: uint = 5u;
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2011-07-27 14:19:39 +02:00
|
|
|
const idx_i16: uint = 6u;
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2011-07-27 14:19:39 +02:00
|
|
|
const idx_i32: uint = 7u;
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2011-07-27 14:19:39 +02:00
|
|
|
const idx_i64: uint = 8u;
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2011-07-27 14:19:39 +02:00
|
|
|
const idx_u8: uint = 9u;
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2011-07-27 14:19:39 +02:00
|
|
|
const idx_u16: uint = 10u;
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2011-07-27 14:19:39 +02:00
|
|
|
const idx_u32: uint = 11u;
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2011-07-27 14:19:39 +02:00
|
|
|
const idx_u64: uint = 12u;
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2011-07-27 14:19:39 +02:00
|
|
|
const idx_f32: uint = 13u;
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2011-07-27 14:19:39 +02:00
|
|
|
const idx_f64: uint = 14u;
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2011-07-27 14:19:39 +02:00
|
|
|
const idx_char: uint = 15u;
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2011-09-02 16:52:14 -07:00
|
|
|
const idx_str: uint = 16u;
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2011-09-01 22:08:59 -07:00
|
|
|
const idx_type: uint = 17u;
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2011-12-15 11:06:48 -08:00
|
|
|
const idx_send_type: uint = 18u;
|
|
|
|
|
|
|
|
const idx_bot: uint = 19u;
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2012-01-05 16:19:12 -08:00
|
|
|
const idx_first_others: uint = 20u;
|
2011-05-11 17:05:39 -07:00
|
|
|
|
2011-08-12 07:15:18 -07:00
|
|
|
type type_store = interner::interner<@raw_t>;
|
2011-04-20 18:52:04 -07:00
|
|
|
|
2012-01-14 16:05:07 -08:00
|
|
|
// substs is a list of actuals that correspond to ty's
|
|
|
|
// formal parameters
|
2011-08-12 07:15:18 -07:00
|
|
|
type ty_param_substs_opt_and_ty = {substs: option::t<[ty::t]>, ty: ty::t};
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2011-05-13 13:05:55 -07:00
|
|
|
type node_type_table =
|
2011-08-12 07:15:18 -07:00
|
|
|
@smallintmap::smallintmap<ty::ty_param_substs_opt_and_ty>;
|
2011-05-12 16:36:47 -07:00
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn populate_type_store(cx: ctxt) {
|
2011-12-19 13:52:58 +01:00
|
|
|
intern(cx, ty_nil);
|
|
|
|
intern(cx, ty_bool);
|
|
|
|
intern(cx, ty_int(ast::ty_i));
|
|
|
|
intern(cx, ty_float(ast::ty_f));
|
|
|
|
intern(cx, ty_uint(ast::ty_u));
|
|
|
|
intern(cx, ty_int(ast::ty_i8));
|
|
|
|
intern(cx, ty_int(ast::ty_i16));
|
|
|
|
intern(cx, ty_int(ast::ty_i32));
|
|
|
|
intern(cx, ty_int(ast::ty_i64));
|
|
|
|
intern(cx, ty_uint(ast::ty_u8));
|
|
|
|
intern(cx, ty_uint(ast::ty_u16));
|
|
|
|
intern(cx, ty_uint(ast::ty_u32));
|
|
|
|
intern(cx, ty_uint(ast::ty_u64));
|
|
|
|
intern(cx, ty_float(ast::ty_f32));
|
|
|
|
intern(cx, ty_float(ast::ty_f64));
|
|
|
|
intern(cx, ty_int(ast::ty_char));
|
|
|
|
intern(cx, ty_str);
|
|
|
|
intern(cx, ty_type);
|
|
|
|
intern(cx, ty_send_type);
|
|
|
|
intern(cx, ty_bot);
|
2011-08-15 16:38:23 -07:00
|
|
|
assert (vec::len(cx.ts.vect) == idx_first_others);
|
2011-04-20 18:52:04 -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
|
|
|
}
|
2011-07-25 15:07:48 +02:00
|
|
|
ret map::mk_hashmap(hash_cache_entry, eq_cache_entries);
|
2011-04-29 15:26:28 +00:00
|
|
|
}
|
2011-04-25 12:15:55 -07:00
|
|
|
|
2012-01-05 15:35:37 +01:00
|
|
|
fn new_ty_hash<V: copy>() -> map::hashmap<t, V> { map::new_uint_hash() }
|
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,
|
2011-07-27 14:19:39 +02:00
|
|
|
freevars: freevars::freevar_map) -> ctxt {
|
|
|
|
let ntt: node_type_table =
|
2011-08-13 00:09:25 -07:00
|
|
|
@smallintmap::mk::<ty::ty_param_substs_opt_and_ty>();
|
2011-12-22 14:24:36 +01:00
|
|
|
fn eq_raw_ty(&&a: @raw_t, &&b: @raw_t) -> bool {
|
|
|
|
ret a.hash == b.hash && a.struct == b.struct;
|
|
|
|
}
|
2011-08-13 00:09:25 -07:00
|
|
|
let ts = @interner::mk::<@raw_t>(hash_raw_ty, eq_raw_ty);
|
2011-07-27 14:19:39 +02:00
|
|
|
let cx =
|
|
|
|
@{ts: ts,
|
|
|
|
sess: s,
|
|
|
|
def_map: dm,
|
|
|
|
node_types: ntt,
|
|
|
|
items: amap,
|
|
|
|
freevars: freevars,
|
2011-12-22 16:23:49 +01:00
|
|
|
tcache: new_def_hash(),
|
2011-07-27 14:19:39 +02:00
|
|
|
rcache: mk_rcache(),
|
2011-12-22 14:24:36 +01:00
|
|
|
short_names_cache: new_ty_hash(),
|
|
|
|
needs_drop_cache: new_ty_hash(),
|
|
|
|
kind_cache: new_ty_hash(),
|
2011-09-02 15:34:58 -07:00
|
|
|
ast_ty_to_ty_cache:
|
2011-12-15 16:27:18 -08:00
|
|
|
map::mk_hashmap(ast_util::hash_ty, ast_util::eq_ty),
|
2011-12-22 08:45:18 +01:00
|
|
|
tag_var_cache: new_def_hash(),
|
2011-12-28 17:50:12 +01:00
|
|
|
iface_method_cache: new_def_hash(),
|
2012-01-02 12:13:26 +01:00
|
|
|
ty_param_bounds: map::new_int_hash()};
|
2011-05-19 17:21:21 -07:00
|
|
|
populate_type_store(cx);
|
|
|
|
ret cx;
|
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
|
2011-12-19 13:52:58 +01:00
|
|
|
fn mk_raw_ty(cx: ctxt, st: sty) -> @raw_t {
|
|
|
|
let h = hash_type_structure(st);
|
2011-07-27 14:19:39 +02:00
|
|
|
let has_params: bool = false;
|
|
|
|
let has_vars: bool = false;
|
2011-09-12 12:39:38 +02:00
|
|
|
fn derive_flags_t(cx: ctxt, &has_params: bool, &has_vars: bool, tt: t) {
|
2011-08-13 00:09:25 -07:00
|
|
|
let rt = interner::get::<@raw_t>(*cx.ts, tt);
|
2011-05-11 17:05:39 -07:00
|
|
|
has_params = has_params || rt.has_params;
|
|
|
|
has_vars = has_vars || rt.has_vars;
|
2011-04-25 16:17:14 -07:00
|
|
|
}
|
2011-09-12 12:39:38 +02:00
|
|
|
fn derive_flags_mt(cx: ctxt, &has_params: bool, &has_vars: bool, m: mt) {
|
2011-05-20 18:36:35 -07:00
|
|
|
derive_flags_t(cx, has_params, has_vars, m.ty);
|
2011-04-25 16:17:14 -07:00
|
|
|
}
|
2011-09-12 12:39:38 +02:00
|
|
|
fn derive_flags_arg(cx: ctxt, &has_params: bool, &has_vars: bool,
|
|
|
|
a: arg) {
|
2011-05-20 18:36:35 -07:00
|
|
|
derive_flags_t(cx, has_params, has_vars, a.ty);
|
2011-04-25 16:17:14 -07:00
|
|
|
}
|
2011-09-12 12:39:38 +02:00
|
|
|
fn derive_flags_sig(cx: ctxt, &has_params: bool, &has_vars: bool,
|
|
|
|
args: [arg], tt: t) {
|
2011-08-15 21:54:52 -07:00
|
|
|
for a: arg in args { derive_flags_arg(cx, has_params, has_vars, a); }
|
2011-05-20 18:36:35 -07:00
|
|
|
derive_flags_t(cx, has_params, has_vars, tt);
|
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(_) |
|
|
|
|
ty_str | ty_send_type | ty_type | ty_native(_) |
|
2012-01-05 16:19:12 -08:00
|
|
|
ty_opaque_closure_ptr(_) {
|
2011-12-15 11:06:48 -08:00
|
|
|
/* no-op */
|
|
|
|
}
|
2011-08-19 15:16:48 -07:00
|
|
|
ty_param(_, _) { has_params = true; }
|
2011-07-27 14:19:39 +02:00
|
|
|
ty_var(_) { has_vars = true; }
|
2011-12-20 16:33:55 +01:00
|
|
|
ty_tag(_, tys) | ty_iface(_, tys) {
|
2011-08-15 21:54:52 -07:00
|
|
|
for tt: t in tys { derive_flags_t(cx, has_params, has_vars, tt); }
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
ty_box(m) { derive_flags_mt(cx, has_params, has_vars, m); }
|
2011-09-21 18:54:54 -07:00
|
|
|
ty_uniq(m) { derive_flags_mt(cx, has_params, has_vars, m); }
|
2011-08-18 14:11:06 -07:00
|
|
|
ty_vec(m) { derive_flags_mt(cx, has_params, has_vars, m); }
|
2011-07-27 14:19:39 +02:00
|
|
|
ty_ptr(m) { derive_flags_mt(cx, has_params, has_vars, m); }
|
|
|
|
ty_rec(flds) {
|
2011-08-15 21:54:52 -07:00
|
|
|
for f: field in flds {
|
2011-07-27 14:19:39 +02:00
|
|
|
derive_flags_mt(cx, has_params, has_vars, f.mt);
|
|
|
|
}
|
|
|
|
}
|
2011-08-15 12:08:05 +02:00
|
|
|
ty_tup(ts) {
|
2011-08-19 15:16:48 -07:00
|
|
|
for tt in ts { derive_flags_t(cx, has_params, has_vars, tt); }
|
2011-08-15 11:40:26 +02:00
|
|
|
}
|
2011-12-23 16:09:52 +01:00
|
|
|
ty_fn(f) {
|
|
|
|
derive_flags_sig(cx, has_params, has_vars, f.inputs, f.output);
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2011-11-21 02:15:40 +08:00
|
|
|
ty_native_fn(args, tt) {
|
2011-07-27 14:19:39 +02:00
|
|
|
derive_flags_sig(cx, has_params, has_vars, args, tt);
|
|
|
|
}
|
|
|
|
ty_res(_, tt, tps) {
|
|
|
|
derive_flags_t(cx, has_params, has_vars, tt);
|
2011-08-15 21:54:52 -07:00
|
|
|
for tt: t in tps { derive_flags_t(cx, has_params, has_vars, tt); }
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2011-12-19 13:52:58 +01:00
|
|
|
ty_constr(tt, _) | ty_named(tt, _) {
|
|
|
|
derive_flags_t(cx, has_params, has_vars, tt);
|
|
|
|
}
|
2011-04-25 16:17:14 -07:00
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
ret @{struct: st,
|
|
|
|
hash: h,
|
|
|
|
has_params: has_params,
|
|
|
|
has_vars: has_vars};
|
2011-05-11 17:05:39 -07:00
|
|
|
}
|
|
|
|
|
2011-12-19 13:52:58 +01:00
|
|
|
fn intern(cx: ctxt, st: sty) {
|
|
|
|
interner::intern(*cx.ts, mk_raw_ty(cx, st));
|
2011-04-25 09:49:08 -07:00
|
|
|
}
|
2011-04-21 19:30:53 -07:00
|
|
|
|
2011-04-25 16:17:14 -07:00
|
|
|
// These are private constructors to this module. External users should always
|
|
|
|
// use the mk_foo() functions below.
|
2011-12-19 13:52:58 +01:00
|
|
|
fn gen_ty(cx: ctxt, st: sty) -> t {
|
|
|
|
let raw_type = mk_raw_ty(cx, st);
|
|
|
|
ret interner::intern(*cx.ts, raw_type);
|
|
|
|
}
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn mk_nil(_cx: ctxt) -> t { ret idx_nil; }
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn mk_bot(_cx: ctxt) -> t { ret idx_bot; }
|
2011-04-25 16:17:14 -07:00
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn mk_bool(_cx: ctxt) -> t { ret idx_bool; }
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn mk_int(_cx: ctxt) -> t { ret idx_int; }
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn mk_float(_cx: ctxt) -> t { ret idx_float; }
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn mk_uint(_cx: ctxt) -> t { ret idx_uint; }
|
2011-04-20 10:51:41 -07:00
|
|
|
|
2011-12-07 21:06:12 +01:00
|
|
|
fn mk_mach_int(_cx: ctxt, tm: ast::int_ty) -> t {
|
2011-07-27 14:19:39 +02:00
|
|
|
alt tm {
|
2012-01-18 22:37:22 -08:00
|
|
|
ast::ty_i { ret idx_int; }
|
|
|
|
ast::ty_char { ret idx_char; }
|
|
|
|
ast::ty_i8 { ret idx_i8; }
|
|
|
|
ast::ty_i16 { ret idx_i16; }
|
|
|
|
ast::ty_i32 { ret idx_i32; }
|
|
|
|
ast::ty_i64 { ret idx_i64; }
|
2011-12-07 21:06:12 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn mk_mach_uint(_cx: ctxt, tm: ast::uint_ty) -> t {
|
|
|
|
alt tm {
|
2012-01-18 22:37:22 -08:00
|
|
|
ast::ty_u { ret idx_uint; }
|
|
|
|
ast::ty_u8 { ret idx_u8; }
|
|
|
|
ast::ty_u16 { ret idx_u16; }
|
|
|
|
ast::ty_u32 { ret idx_u32; }
|
|
|
|
ast::ty_u64 { ret idx_u64; }
|
2011-12-07 21:06:12 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn mk_mach_float(_cx: ctxt, tm: ast::float_ty) -> t {
|
|
|
|
alt tm {
|
2012-01-18 22:37:22 -08:00
|
|
|
ast::ty_f { ret idx_float; }
|
|
|
|
ast::ty_f32 { ret idx_f32; }
|
|
|
|
ast::ty_f64 { ret idx_f64; }
|
2011-04-25 09:49:08 -07:00
|
|
|
}
|
2011-04-20 10:51:41 -07:00
|
|
|
}
|
|
|
|
|
2011-12-07 21:06:12 +01:00
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn mk_char(_cx: ctxt) -> t { ret idx_char; }
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn mk_str(_cx: ctxt) -> t { ret idx_str; }
|
2011-04-20 11:23:36 -07:00
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn mk_tag(cx: ctxt, did: ast::def_id, tys: [t]) -> t {
|
2011-04-25 12:15:55 -07:00
|
|
|
ret gen_ty(cx, ty_tag(did, tys));
|
2011-04-20 18:52:04 -07:00
|
|
|
}
|
2011-04-20 11:23:36 -07:00
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn mk_box(cx: ctxt, tm: mt) -> t { ret gen_ty(cx, ty_box(tm)); }
|
2011-04-20 18:52:04 -07:00
|
|
|
|
2011-09-21 18:54:54 -07:00
|
|
|
fn mk_uniq(cx: ctxt, tm: mt) -> t { ret gen_ty(cx, ty_uniq(tm)); }
|
|
|
|
|
|
|
|
fn mk_imm_uniq(cx: ctxt, ty: t) -> t {
|
|
|
|
ret mk_uniq(cx, {ty: ty, mut: ast::imm});
|
|
|
|
}
|
2011-08-10 17:23:46 -07:00
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn mk_ptr(cx: ctxt, tm: mt) -> t { ret gen_ty(cx, ty_ptr(tm)); }
|
2011-06-03 15:02:58 -04:00
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn mk_imm_box(cx: ctxt, ty: t) -> t {
|
2011-07-27 14:19:39 +02:00
|
|
|
ret mk_box(cx, {ty: ty, mut: ast::imm});
|
2011-04-20 18:52:04 -07:00
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn mk_mut_ptr(cx: ctxt, ty: t) -> t {
|
2011-08-02 16:24:38 -07:00
|
|
|
ret mk_ptr(cx, {ty: ty, mut: ast::mut});
|
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn mk_vec(cx: ctxt, tm: mt) -> t { ret gen_ty(cx, ty_vec(tm)); }
|
2011-06-09 16:23:19 -07:00
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn mk_rec(cx: ctxt, fs: [field]) -> t { ret gen_ty(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 {
|
2011-07-19 17:52:34 -07:00
|
|
|
ret gen_ty(cx, ty_constr(t, cs));
|
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn mk_tup(cx: ctxt, ts: [t]) -> t { ret gen_ty(cx, ty_tup(ts)); }
|
2011-08-15 11:40:26 +02:00
|
|
|
|
2011-12-23 16:09:52 +01:00
|
|
|
fn mk_fn(cx: ctxt, fty: fn_ty) -> t {
|
|
|
|
ret gen_ty(cx, ty_fn(fty));
|
2011-04-20 18:52:04 -07:00
|
|
|
}
|
|
|
|
|
2011-11-21 02:15:40 +08:00
|
|
|
fn mk_native_fn(cx: ctxt, args: [arg], ty: t) -> t {
|
|
|
|
ret gen_ty(cx, ty_native_fn(args, ty));
|
2011-04-20 18:52:04 -07:00
|
|
|
}
|
2011-04-20 10:51:41 -07:00
|
|
|
|
2011-12-20 16:33:55 +01:00
|
|
|
fn mk_iface(cx: ctxt, did: ast::def_id, tys: [t]) -> t {
|
|
|
|
ret gen_ty(cx, ty_iface(did, tys));
|
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn mk_res(cx: ctxt, did: ast::def_id, inner: t, tps: [t]) -> t {
|
2011-06-30 14:46:17 +02:00
|
|
|
ret gen_ty(cx, ty_res(did, inner, tps));
|
2011-06-24 18:10:40 +02:00
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn mk_var(cx: ctxt, v: int) -> t { ret gen_ty(cx, ty_var(v)); }
|
2011-04-20 18:52:04 -07:00
|
|
|
|
2011-12-29 11:23:35 +01:00
|
|
|
fn mk_param(cx: ctxt, n: uint, k: def_id) -> t {
|
2011-07-28 13:29:29 -07:00
|
|
|
ret gen_ty(cx, ty_param(n, k));
|
|
|
|
}
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn mk_type(_cx: ctxt) -> t { ret idx_type; }
|
2011-04-20 18:52:04 -07:00
|
|
|
|
2011-12-15 11:06:48 -08:00
|
|
|
fn mk_send_type(_cx: ctxt) -> t { ret idx_send_type; }
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn mk_native(cx: ctxt, did: def_id) -> t { ret gen_ty(cx, ty_native(did)); }
|
2011-04-20 10:51:41 -07:00
|
|
|
|
2012-01-05 16:19:12 -08:00
|
|
|
fn mk_opaque_closure_ptr(cx: ctxt, ck: closure_kind) -> t {
|
|
|
|
ret gen_ty(cx, ty_opaque_closure_ptr(ck));
|
2011-12-15 11:06:48 -08:00
|
|
|
}
|
|
|
|
|
2011-12-19 13:52:58 +01:00
|
|
|
fn mk_named(cx: ctxt, base: t, name: @str) -> t {
|
|
|
|
gen_ty(cx, ty_named(base, name))
|
|
|
|
}
|
2011-04-22 12:08:23 -07:00
|
|
|
|
2011-12-19 13:52:58 +01:00
|
|
|
// Returns the one-level-deep type structure of the given type.
|
|
|
|
pure fn struct(cx: ctxt, typ: t) -> sty {
|
|
|
|
alt interner::get(*cx.ts, typ).struct {
|
|
|
|
ty_named(t, _) { struct(cx, t) }
|
|
|
|
s { s }
|
|
|
|
}
|
|
|
|
}
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2012-01-16 11:45:18 +01:00
|
|
|
pure fn struct_raw(cx: ctxt, typ: t) -> sty {
|
|
|
|
interner::get(*cx.ts, typ).struct
|
|
|
|
}
|
|
|
|
|
2011-12-23 01:57:35 +01:00
|
|
|
// Returns struact(cx, typ) but replaces all occurences of platform
|
|
|
|
// dependent primitive types with their machine type equivalent
|
|
|
|
pure fn mach_struct(cx: ctxt, cfg: @session::config, typ: t) -> sty {
|
|
|
|
alt interner::get(*cx.ts, typ).struct {
|
|
|
|
ty_named(t, _) { mach_struct(cx, cfg, t) }
|
|
|
|
s { mach_sty(cfg, s) }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Converts s to its machine type equivalent
|
|
|
|
pure fn mach_sty(cfg: @session::config, s: sty) -> sty {
|
|
|
|
alt s {
|
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 }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-12-19 13:52:58 +01:00
|
|
|
pure fn ty_name(cx: ctxt, typ: t) -> option::t<@str> {
|
|
|
|
alt interner::get(*cx.ts, typ).struct {
|
|
|
|
ty_named(_, n) { some(n) }
|
|
|
|
_ { none }
|
|
|
|
}
|
2011-05-16 13:58:13 -07:00
|
|
|
}
|
2011-04-22 13:45:21 -07:00
|
|
|
|
2011-04-22 12:08:23 -07:00
|
|
|
|
2011-06-15 11:19:50 -07:00
|
|
|
// Type folds
|
2011-10-18 15:07:40 -07:00
|
|
|
type ty_walk = fn@(t);
|
2011-04-15 12:23:00 -07:00
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn walk_ty(cx: ctxt, walker: ty_walk, ty: t) {
|
2011-07-27 14:19:39 +02:00
|
|
|
alt struct(cx, ty) {
|
2012-01-18 22:37:22 -08:00
|
|
|
ty_nil | ty_bot | ty_bool | ty_int(_) | ty_uint(_) | ty_float(_) |
|
|
|
|
ty_str | ty_send_type | ty_type | ty_native(_) |
|
2012-01-05 16:19:12 -08:00
|
|
|
ty_opaque_closure_ptr(_) {
|
2011-12-15 11:06:48 -08:00
|
|
|
/* no-op */
|
|
|
|
}
|
2011-12-07 21:06:12 +01:00
|
|
|
ty_box(tm) | ty_vec(tm) | ty_ptr(tm) { walk_ty(cx, walker, tm.ty); }
|
2011-12-20 16:33:55 +01:00
|
|
|
ty_tag(_, subtys) | ty_iface(_, subtys) {
|
2011-08-15 21:54:52 -07:00
|
|
|
for subty: t in subtys { walk_ty(cx, walker, subty); }
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
ty_rec(fields) {
|
2011-08-15 21:54:52 -07:00
|
|
|
for fl: field in fields { walk_ty(cx, walker, fl.mt.ty); }
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2011-08-19 15:16:48 -07:00
|
|
|
ty_tup(ts) { for tt in ts { walk_ty(cx, walker, tt); } }
|
2011-12-23 16:09:52 +01:00
|
|
|
ty_fn(f) {
|
|
|
|
for a: arg in f.inputs { walk_ty(cx, walker, a.ty); }
|
|
|
|
walk_ty(cx, walker, f.output);
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2011-11-21 02:15:40 +08:00
|
|
|
ty_native_fn(args, ret_ty) {
|
2011-08-15 21:54:52 -07:00
|
|
|
for a: arg in args { walk_ty(cx, walker, a.ty); }
|
2011-07-27 14:19:39 +02:00
|
|
|
walk_ty(cx, walker, ret_ty);
|
|
|
|
}
|
|
|
|
ty_res(_, sub, tps) {
|
|
|
|
walk_ty(cx, walker, sub);
|
2011-08-15 21:54:52 -07:00
|
|
|
for tp: t in tps { walk_ty(cx, walker, tp); }
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2011-08-19 15:16:48 -07:00
|
|
|
ty_constr(sub, _) { walk_ty(cx, walker, sub); }
|
2011-07-27 14:19:39 +02:00
|
|
|
ty_var(_) {/* no-op */ }
|
2011-08-19 15:16:48 -07:00
|
|
|
ty_param(_, _) {/* no-op */ }
|
2011-09-21 18:54:54 -07:00
|
|
|
ty_uniq(tm) { walk_ty(cx, walker, tm.ty); }
|
2011-04-15 12:23:00 -07:00
|
|
|
}
|
|
|
|
walker(ty);
|
|
|
|
}
|
|
|
|
|
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),
|
|
|
|
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 {
|
2011-07-27 14:19:39 +02:00
|
|
|
let ty = ty_0;
|
2011-06-09 11:20:47 -07:00
|
|
|
// Fast paths.
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2011-07-27 14:19:39 +02:00
|
|
|
alt fld {
|
|
|
|
fm_var(_) { if !type_contains_vars(cx, ty) { ret ty; } }
|
|
|
|
fm_param(_) { if !type_contains_params(cx, ty) { ret ty; } }
|
|
|
|
fm_general(_) {/* no fast path */ }
|
|
|
|
}
|
2011-12-19 13:52:58 +01:00
|
|
|
alt interner::get(*cx.ts, ty).struct {
|
2012-01-18 22:37:22 -08:00
|
|
|
ty_nil | ty_bot | ty_bool | ty_int(_) | ty_uint(_) | ty_float(_) |
|
|
|
|
ty_str | ty_send_type | ty_type | ty_native(_) |
|
2012-01-05 16:19:12 -08:00
|
|
|
ty_opaque_closure_ptr(_) {
|
2011-12-15 11:06:48 -08:00
|
|
|
/* no-op */
|
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
ty_box(tm) {
|
2011-08-10 17:23:46 -07:00
|
|
|
ty = mk_box(cx, {ty: fold_ty(cx, fld, tm.ty), mut: tm.mut});
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2011-09-21 18:54:54 -07:00
|
|
|
ty_uniq(tm) {
|
|
|
|
ty = mk_uniq(cx, {ty: fold_ty(cx, fld, tm.ty), mut: tm.mut});
|
|
|
|
}
|
2011-12-19 13:52:58 +01:00
|
|
|
ty_named(t, nm) {
|
|
|
|
ty = mk_named(cx, fold_ty(cx, fld, t), nm);
|
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
ty_ptr(tm) {
|
2011-08-10 17:23:46 -07:00
|
|
|
ty = mk_ptr(cx, {ty: fold_ty(cx, fld, tm.ty), mut: tm.mut});
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2011-08-18 14:11:06 -07:00
|
|
|
ty_vec(tm) {
|
2011-08-18 14:32:25 -07:00
|
|
|
ty = mk_vec(cx, {ty: fold_ty(cx, fld, tm.ty), mut: tm.mut});
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
ty_tag(tid, subtys) {
|
2011-12-20 16:33:55 +01:00
|
|
|
ty = mk_tag(cx, tid, vec::map(subtys, {|t| fold_ty(cx, fld, t) }));
|
|
|
|
}
|
|
|
|
ty_iface(did, subtys) {
|
|
|
|
ty = mk_iface(cx, did, vec::map(subtys, {|t| fold_ty(cx, fld, t) }));
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
ty_rec(fields) {
|
2011-08-19 15:16:48 -07:00
|
|
|
let new_fields: [field] = [];
|
2011-08-15 21:54:52 -07:00
|
|
|
for fl: field in fields {
|
2011-07-27 14:19:39 +02:00
|
|
|
let new_ty = fold_ty(cx, fld, fl.mt.ty);
|
|
|
|
let new_mt = {ty: new_ty, mut: fl.mt.mut};
|
2011-08-19 15:16:48 -07:00
|
|
|
new_fields += [{ident: fl.ident, mt: new_mt}];
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2011-12-19 13:52:58 +01:00
|
|
|
ty = mk_rec(cx, new_fields);
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2011-08-15 12:08:05 +02:00
|
|
|
ty_tup(ts) {
|
2011-08-19 15:16:48 -07:00
|
|
|
let new_ts = [];
|
|
|
|
for tt in ts { new_ts += [fold_ty(cx, fld, tt)]; }
|
2011-12-19 13:52:58 +01:00
|
|
|
ty = mk_tup(cx, new_ts);
|
2011-08-15 11:40:26 +02:00
|
|
|
}
|
2011-12-23 16:09:52 +01:00
|
|
|
ty_fn(f) {
|
2011-08-19 15:16:48 -07:00
|
|
|
let new_args: [arg] = [];
|
2011-12-23 16:09:52 +01:00
|
|
|
for a: arg in f.inputs {
|
2011-07-27 14:19:39 +02:00
|
|
|
let new_ty = fold_ty(cx, fld, a.ty);
|
2011-08-19 15:16:48 -07:00
|
|
|
new_args += [{mode: a.mode, ty: new_ty}];
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2011-12-23 16:09:52 +01:00
|
|
|
ty = mk_fn(cx, {inputs: new_args,
|
|
|
|
output: fold_ty(cx, fld, f.output)
|
|
|
|
with f});
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2011-11-21 02:15:40 +08:00
|
|
|
ty_native_fn(args, ret_ty) {
|
2011-08-19 15:16:48 -07:00
|
|
|
let new_args: [arg] = [];
|
2011-08-15 21:54:52 -07:00
|
|
|
for a: arg in args {
|
2011-07-27 14:19:39 +02:00
|
|
|
let new_ty = fold_ty(cx, fld, a.ty);
|
2011-08-19 15:16:48 -07:00
|
|
|
new_args += [{mode: a.mode, ty: new_ty}];
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2011-12-19 13:52:58 +01:00
|
|
|
ty = mk_native_fn(cx, new_args, fold_ty(cx, fld, ret_ty));
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
ty_res(did, subty, tps) {
|
2011-08-19 15:16:48 -07:00
|
|
|
let new_tps = [];
|
|
|
|
for tp: t in tps { new_tps += [fold_ty(cx, fld, tp)]; }
|
2011-12-19 13:52:58 +01:00
|
|
|
ty = mk_res(cx, did, fold_ty(cx, fld, subty), new_tps);
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
ty_var(id) {
|
|
|
|
alt fld { fm_var(folder) { ty = folder(id); } _ {/* no-op */ } }
|
|
|
|
}
|
2011-12-29 11:23:35 +01:00
|
|
|
ty_param(id, did) {
|
2011-12-29 13:12:52 +01:00
|
|
|
alt fld { fm_param(folder) { ty = folder(id, did); } _ {} }
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2012-01-03 15:50:05 -08:00
|
|
|
ty_constr(subty, cs) {
|
|
|
|
ty = mk_constr(cx, fold_ty(cx, fld, subty), cs);
|
|
|
|
}
|
|
|
|
_ {
|
|
|
|
cx.sess.fatal("Unsupported sort of type in fold_ty");
|
|
|
|
}
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
|
2011-06-09 11:20:47 -07:00
|
|
|
// If this is a general type fold, then we need to run it now.
|
2011-07-27 14:19:39 +02:00
|
|
|
alt fld { fm_general(folder) { ret folder(ty); } _ { ret ty; } }
|
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
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn type_is_nil(cx: ctxt, ty: t) -> bool {
|
2012-01-18 22:37:22 -08:00
|
|
|
alt struct(cx, ty) { ty_nil { ret true; } _ { ret false; } }
|
2011-05-14 19:02:30 -07:00
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn type_is_bot(cx: ctxt, ty: t) -> bool {
|
2012-01-18 22:37:22 -08:00
|
|
|
alt struct(cx, ty) { ty_bot { ret true; } _ { ret false; } }
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn type_is_bool(cx: ctxt, ty: t) -> bool {
|
2012-01-18 22:37:22 -08:00
|
|
|
alt struct(cx, ty) { ty_bool { ret true; } _ { ret false; } }
|
2011-04-19 15:22:57 -07:00
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn type_is_structural(cx: ctxt, ty: t) -> bool {
|
2011-07-27 14:19:39 +02:00
|
|
|
alt struct(cx, ty) {
|
2011-12-20 16:33:55 +01:00
|
|
|
ty_rec(_) | ty_tup(_) | ty_tag(_, _) | ty_fn(_) |
|
2012-01-13 10:58:31 +01:00
|
|
|
ty_native_fn(_, _) | 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
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn type_is_sequence(cx: ctxt, ty: t) -> bool {
|
2011-07-27 14:19:39 +02:00
|
|
|
alt struct(cx, ty) {
|
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
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn type_is_str(cx: ctxt, ty: t) -> bool {
|
2012-01-18 22:37:22 -08:00
|
|
|
alt struct(cx, ty) { ty_str { ret true; } _ { ret false; } }
|
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 {
|
2011-07-27 14:19:39 +02:00
|
|
|
alt struct(cx, ty) {
|
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
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-17 10:17:40 -07:00
|
|
|
pure fn type_is_tup_like(cx: ctxt, ty: t) -> bool {
|
2011-09-24 16:33:26 -07:00
|
|
|
let sty = struct(cx, ty);
|
2011-09-17 10:17:40 -07:00
|
|
|
alt sty {
|
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
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn get_element_type(cx: ctxt, ty: t, i: uint) -> t {
|
2011-07-27 14:19:39 +02:00
|
|
|
alt struct(cx, ty) {
|
2011-08-19 15:16:48 -07:00
|
|
|
ty_rec(flds) { ret flds[i].mt.ty; }
|
|
|
|
ty_tup(ts) { ret ts[i]; }
|
2011-07-27 14:19:39 +02:00
|
|
|
_ {
|
2012-01-17 20:59:49 -08:00
|
|
|
cx.sess.bug(
|
|
|
|
#fmt["get_element_type called on invalid type %s with index %u",
|
|
|
|
ty_to_str(cx, ty), i]);
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2011-01-19 16:29:14 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-24 15:36:09 -07:00
|
|
|
pure fn type_is_box(cx: ctxt, ty: t) -> bool {
|
2011-09-21 14:00:11 -07:00
|
|
|
alt struct(cx, ty) {
|
|
|
|
ty_box(_) { ret true; }
|
|
|
|
_ { ret false; }
|
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
|
2011-09-24 15:36:09 -07:00
|
|
|
pure fn type_is_boxed(cx: ctxt, ty: t) -> bool {
|
2011-09-21 14:00:11 -07:00
|
|
|
alt struct(cx, ty) {
|
2012-01-07 22:44:14 +01:00
|
|
|
ty_box(_) | ty_iface(_, _) { ret true; }
|
2011-09-21 14:00:11 -07:00
|
|
|
_ { ret false; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-24 15:36:09 -07:00
|
|
|
pure fn type_is_unique_box(cx: ctxt, ty: t) -> bool {
|
2011-09-21 14:00:11 -07:00
|
|
|
alt struct(cx, ty) {
|
|
|
|
ty_uniq(_) { ret true; }
|
|
|
|
_ { ret false; }
|
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2011-06-18 00:41:25 -07:00
|
|
|
|
2011-11-02 12:15:50 +01:00
|
|
|
pure fn type_is_unsafe_ptr(cx: ctxt, ty: t) -> bool {
|
|
|
|
alt struct(cx, ty) {
|
|
|
|
ty_ptr(_) { ret true; }
|
|
|
|
_ { ret false; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-24 15:36:09 -07:00
|
|
|
pure fn type_is_vec(cx: ctxt, ty: t) -> bool {
|
2011-08-25 10:18:02 +02:00
|
|
|
ret alt struct(cx, ty) {
|
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
|
|
|
}
|
|
|
|
|
2011-09-24 15:36:09 -07:00
|
|
|
pure fn type_is_unique(cx: ctxt, ty: t) -> bool {
|
2011-08-25 10:18:02 +02:00
|
|
|
alt struct(cx, ty) {
|
|
|
|
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
|
|
|
}
|
|
|
|
|
2011-09-24 15:36:09 -07:00
|
|
|
pure fn type_is_scalar(cx: ctxt, ty: t) -> bool {
|
2011-07-27 14:19:39 +02:00
|
|
|
alt struct(cx, ty) {
|
2012-01-18 22:37:22 -08:00
|
|
|
ty_nil | ty_bool | ty_int(_) | ty_float(_) | ty_uint(_) |
|
|
|
|
ty_send_type | ty_type | ty_native(_) | ty_ptr(_) { 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?
|
|
|
|
fn type_is_immediate(cx: ctxt, ty: t) -> bool {
|
|
|
|
ret type_is_scalar(cx, ty) || type_is_boxed(cx, ty) ||
|
2011-10-10 13:32:50 +02:00
|
|
|
type_is_unique(cx, ty) || type_is_native(cx, 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
|
|
|
}
|
|
|
|
|
2011-11-22 13:16:23 +01:00
|
|
|
let accum = false;
|
|
|
|
let result = alt struct(cx, ty) {
|
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(_) |
|
|
|
|
ty_type | ty_native(_) | ty_ptr(_) { 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
|
|
|
}
|
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
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
ty_tag(did, tps) {
|
|
|
|
let variants = tag_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-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 {
|
|
|
|
if ty::kind_lteq(a, b) { a } else { b }
|
|
|
|
}
|
|
|
|
|
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
|
|
|
|
2011-11-15 18:15:35 +01:00
|
|
|
let result = alt struct(cx, ty) {
|
|
|
|
// 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(_) |
|
2011-12-07 21:06:12 +01:00
|
|
|
ty_native(_) | ty_ptr(_) |
|
2012-01-18 22:37:22 -08:00
|
|
|
ty_send_type | ty_str | ty_native_fn(_, _) { kind_sendable }
|
|
|
|
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.
|
2011-12-28 18:11:33 +01:00
|
|
|
ty_box(_) | ty_iface(_, _) { 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) {
|
2011-12-28 18:11:33 +01:00
|
|
|
let 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) {
|
2011-12-28 18:11:33 +01:00
|
|
|
let 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
|
|
|
}
|
2011-07-27 17:49:00 -07:00
|
|
|
// Tags lower to the lowest of their variants.
|
|
|
|
ty_tag(did, tps) {
|
2011-12-28 18:11:33 +01:00
|
|
|
let lowest = kind_sendable;
|
2011-12-15 17:14:58 -08:00
|
|
|
for variant in *tag_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) }
|
|
|
|
};
|
2011-07-27 17:49:00 -07:00
|
|
|
|
|
|
|
cx.kind_cache.insert(ty, result);
|
|
|
|
ret result;
|
|
|
|
}
|
|
|
|
|
2011-07-27 14:19:39 +02:00
|
|
|
// FIXME: should we just return true for native types in
|
|
|
|
// type_is_scalar?
|
2011-09-12 11:27:30 +02:00
|
|
|
fn type_is_native(cx: ctxt, ty: t) -> bool {
|
2011-07-27 14:19:39 +02:00
|
|
|
alt struct(cx, ty) { ty_native(_) { ret true; } _ { ret false; } }
|
|
|
|
}
|
|
|
|
|
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 {
|
2011-08-22 13:39:32 +02:00
|
|
|
let sty = struct(cx, ty);
|
|
|
|
if test(sty) { ret true; }
|
|
|
|
alt sty {
|
|
|
|
ty_tag(did, tps) {
|
2011-12-15 17:14:58 -08:00
|
|
|
for variant in *tag_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-12-29 11:23:35 +01:00
|
|
|
pure fn type_has_dynamic_size(cx: ctxt, ty: t) -> bool unchecked {
|
2011-09-12 11:27:30 +02:00
|
|
|
|
2011-09-02 18:59:22 -07:00
|
|
|
/* type_structurally_contains can't be declared pure
|
|
|
|
because it takes a function argument. But it should be
|
|
|
|
referentially transparent, since a given type's size should
|
|
|
|
never change once it's created.
|
|
|
|
(It would be interesting to think about how to make such properties
|
|
|
|
actually checkable. It seems to me like a lot of properties
|
|
|
|
that the type context tracks about types should be immutable.)
|
|
|
|
*/
|
2012-01-11 09:58:05 -08:00
|
|
|
type_structurally_contains(cx, ty) {|sty|
|
2012-01-05 10:44:59 -08:00
|
|
|
alt sty {
|
2012-01-05 16:19:12 -08:00
|
|
|
ty_param(_, _) { true }
|
2012-01-05 10:44:59 -08:00
|
|
|
_ { false }
|
|
|
|
}
|
2012-01-11 09:58:05 -08:00
|
|
|
}
|
2011-08-22 13:39:32 +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
|
|
|
|
// distinguished from the value itself. I.e. types with mutable content that's
|
|
|
|
// 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) {
|
|
|
|
mt.mut != ast::imm
|
|
|
|
}
|
|
|
|
ty_rec(fields) {
|
|
|
|
for field in fields {
|
|
|
|
if field.mt.mut !=
|
|
|
|
ast::imm {
|
|
|
|
ret true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
false
|
|
|
|
}
|
|
|
|
_ { 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|
|
2011-09-12 14:43:41 +02:00
|
|
|
ret alt sty {
|
|
|
|
ty_uniq(_) { ret true; }
|
|
|
|
ty_vec(_) { true }
|
2012-01-18 22:37:22 -08:00
|
|
|
ty_str { true }
|
2011-09-12 14:43:41 +02:00
|
|
|
_ { ret false; }
|
|
|
|
};
|
|
|
|
});
|
2011-09-12 11:27:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
fn type_is_integral(cx: ctxt, ty: t) -> bool {
|
2011-07-27 14:19:39 +02:00
|
|
|
alt struct(cx, ty) {
|
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
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn type_is_fp(cx: ctxt, ty: t) -> bool {
|
2011-07-27 14:19:39 +02:00
|
|
|
alt struct(cx, ty) {
|
2011-12-07 21:06:12 +01:00
|
|
|
ty_float(_) { true }
|
|
|
|
_ { false }
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-12-02 13:42:51 +01:00
|
|
|
fn type_is_numeric(cx: ctxt, ty: t) -> bool {
|
|
|
|
ret type_is_integral(cx, ty) || type_is_fp(cx, ty);
|
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn type_is_signed(cx: ctxt, ty: t) -> bool {
|
2011-07-27 14:19:39 +02:00
|
|
|
alt struct(cx, ty) {
|
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 {
|
2011-08-04 10:46:10 -07:00
|
|
|
let result = true;
|
|
|
|
alt struct(cx, ty) {
|
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(_) |
|
|
|
|
ty_send_type | ty_type | ty_native(_) | 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-01-13 10:58:31 +01:00
|
|
|
ty_native_fn(_, _) | ty_iface(_, _) { result = false; }
|
2011-08-19 15:16:48 -07:00
|
|
|
// Structural types
|
|
|
|
ty_tag(did, tps) {
|
|
|
|
let variants = tag_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.
|
|
|
|
tup_ty = substitute_type_params(cx, tps, tup_ty);
|
|
|
|
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_var(_) {
|
|
|
|
fail "ty_var in type_is_pod";
|
|
|
|
}
|
|
|
|
ty_param(_, _) { result = false; }
|
2011-08-04 10:46:10 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
ret result;
|
|
|
|
}
|
|
|
|
|
2012-01-10 14:55:54 -07:00
|
|
|
fn type_is_tag(cx: ctxt, ty: t) -> bool {
|
|
|
|
alt struct(cx, ty) {
|
|
|
|
ty_tag(_, _) { ret true; }
|
|
|
|
_ { 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-01-10 14:55:54 -07:00
|
|
|
alt struct(cx, ty) {
|
|
|
|
ty_tag(did, tps) {
|
|
|
|
let variants = tag_variants(cx, did);
|
|
|
|
let some_n_ary = vec::any(*variants, {|v| vec::len(v.args) > 0u});
|
|
|
|
ret !some_n_ary;
|
|
|
|
}
|
|
|
|
_ { ret false;}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn type_param(cx: ctxt, ty: t) -> option::t<uint> {
|
2011-07-27 14:19:39 +02:00
|
|
|
alt struct(cx, ty) {
|
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.
|
2011-09-12 11:27:30 +02:00
|
|
|
fn vars_in_type(cx: ctxt, ty: t) -> [int] {
|
|
|
|
fn collect_var(cx: ctxt, vars: @mutable [int], ty: t) {
|
2011-08-19 15:16:48 -07:00
|
|
|
alt struct(cx, ty) { ty_var(v) { *vars += [v]; } _ { } }
|
2011-08-03 18:06:57 -07:00
|
|
|
}
|
2011-08-19 15:16:48 -07:00
|
|
|
let rslt: @mutable [int] = @mutable [];
|
2011-08-03 18:06:57 -07:00
|
|
|
walk_ty(cx, bind collect_var(cx, rslt, _), ty);
|
|
|
|
// Works because of a "convenient" bug that lets us
|
2011-08-18 15:00:12 -07:00
|
|
|
// return a mutable vec as if it's immutable
|
2011-08-03 18:06:57 -07:00
|
|
|
ret *rslt;
|
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn type_autoderef(cx: ctxt, t: ty::t) -> ty::t {
|
2011-09-29 11:18:40 +02:00
|
|
|
let t1 = t;
|
2011-07-27 14:19:39 +02:00
|
|
|
while true {
|
|
|
|
alt struct(cx, t1) {
|
2011-09-29 11:18:40 +02:00
|
|
|
ty_box(mt) | ty_uniq(mt) { t1 = mt.ty; }
|
|
|
|
ty_res(_, inner, tps) {
|
2011-07-27 14:19:39 +02:00
|
|
|
t1 = substitute_type_params(cx, tps, inner);
|
|
|
|
}
|
2011-09-29 11:18:40 +02:00
|
|
|
ty_tag(did, tps) {
|
2011-07-27 14:19:39 +02:00
|
|
|
let variants = tag_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 {
|
2011-07-27 14:19:39 +02:00
|
|
|
fn hash_uint(id: uint, n: uint) -> uint {
|
|
|
|
let h = id;
|
2011-12-07 07:21:07 -08:00
|
|
|
h += (h << 5u) + n;
|
2011-04-20 14:34:17 -07:00
|
|
|
ret h;
|
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
fn hash_def(id: uint, did: ast::def_id) -> uint {
|
|
|
|
let h = id;
|
2011-12-07 07:21:07 -08:00
|
|
|
h += (h << 5u) + (did.crate as uint);
|
|
|
|
h += (h << 5u) + (did.node as uint);
|
2011-04-20 14:34:17 -07:00
|
|
|
ret h;
|
2011-04-19 18:46:22 -07:00
|
|
|
}
|
2011-08-22 12:45:18 +02:00
|
|
|
fn hash_subty(id: uint, subty: t) -> uint {
|
2011-07-27 14:19:39 +02:00
|
|
|
let h = id;
|
2011-12-22 14:24:36 +01:00
|
|
|
h += (h << 5u) + subty;
|
2011-04-20 14:34:17 -07:00
|
|
|
ret h;
|
|
|
|
}
|
2011-12-15 11:06:48 -08:00
|
|
|
fn hash_subtys(id: uint, subtys: [t]) -> uint {
|
|
|
|
let h = id;
|
|
|
|
vec::iter(subtys) { |subty|
|
|
|
|
h = hash_subty(h, subty);
|
|
|
|
}
|
|
|
|
ret h;
|
|
|
|
}
|
2011-09-12 11:27:30 +02:00
|
|
|
fn hash_type_constr(id: uint, c: @type_constr) -> uint {
|
2011-07-27 14:19:39 +02:00
|
|
|
let h = id;
|
2011-12-07 07:21:07 -08:00
|
|
|
h += (h << 5u) + hash_def(h, c.node.id);
|
2011-07-19 17:52:34 -07:00
|
|
|
ret hash_type_constr_args(h, c.node.args);
|
|
|
|
}
|
2011-08-04 16:20:09 -07:00
|
|
|
fn hash_type_constr_args(id: uint, args: [@ty_constr_arg]) -> uint {
|
2011-07-27 14:19:39 +02:00
|
|
|
let h = id;
|
2011-08-15 21:54:52 -07:00
|
|
|
for a: @ty_constr_arg in args {
|
2011-07-27 14:19:39 +02:00
|
|
|
alt a.node {
|
2012-01-18 22:37:22 -08:00
|
|
|
carg_base { h += h << 5u; }
|
2011-07-27 14:19:39 +02:00
|
|
|
carg_lit(_) {
|
|
|
|
// FIXME
|
|
|
|
fail "lit args not implemented yet";
|
|
|
|
}
|
|
|
|
carg_ident(p) {
|
|
|
|
// FIXME: Not sure what to do here.
|
|
|
|
h += h << 5u;
|
|
|
|
}
|
2011-07-19 17:52:34 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
ret h;
|
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn hash_fn(id: uint, args: [arg], rty: t) -> uint {
|
2011-07-27 14:19:39 +02:00
|
|
|
let h = id;
|
2011-12-22 14:24:36 +01:00
|
|
|
for a: arg in args { h += (h << 5u) + a.ty; }
|
|
|
|
h += (h << 5u) + rty;
|
2011-04-20 14:34:17 -07:00
|
|
|
ret h;
|
|
|
|
}
|
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-01-18 22:37:22 -08:00
|
|
|
ty_str { ret 17u; }
|
2011-07-27 14:19:39 +02:00
|
|
|
ty_tag(did, tys) {
|
|
|
|
let h = hash_def(18u, did);
|
2011-12-22 14:24:36 +01:00
|
|
|
for typ: t in tys { h += (h << 5u) + typ; }
|
2011-07-27 14:19:39 +02:00
|
|
|
ret h;
|
|
|
|
}
|
|
|
|
ty_box(mt) { ret hash_subty(19u, mt.ty); }
|
2011-08-18 14:11:06 -07:00
|
|
|
ty_vec(mt) { ret hash_subty(21u, mt.ty); }
|
2011-07-27 14:19:39 +02:00
|
|
|
ty_rec(fields) {
|
|
|
|
let h = 26u;
|
2011-12-22 14:24:36 +01:00
|
|
|
for f: field in fields { h += (h << 5u) + f.mt.ty; }
|
2011-07-27 14:19:39 +02:00
|
|
|
ret h;
|
|
|
|
}
|
2011-12-15 11:06:48 -08:00
|
|
|
ty_tup(ts) { ret hash_subtys(25u, ts); }
|
2011-07-27 14:19:39 +02:00
|
|
|
|
|
|
|
// ???
|
2011-12-23 16:09:52 +01:00
|
|
|
ty_fn(f) { ret hash_fn(27u, f.inputs, f.output); }
|
2011-11-21 02:15:40 +08:00
|
|
|
ty_native_fn(args, rty) { ret hash_fn(28u, args, rty); }
|
2011-07-27 14:19:39 +02:00
|
|
|
ty_var(v) { ret hash_uint(30u, v as uint); }
|
2011-08-19 15:16:48 -07:00
|
|
|
ty_param(pid, _) { ret hash_uint(31u, pid); }
|
2012-01-18 22:37:22 -08:00
|
|
|
ty_type { ret 32u; }
|
2011-07-27 14:19:39 +02:00
|
|
|
ty_native(did) { ret hash_def(33u, did); }
|
2012-01-18 22:37:22 -08:00
|
|
|
ty_bot { ret 34u; }
|
2011-07-27 14:19:39 +02:00
|
|
|
ty_ptr(mt) { ret hash_subty(35u, mt.ty); }
|
|
|
|
ty_res(did, sub, tps) {
|
|
|
|
let h = hash_subty(hash_def(18u, did), sub);
|
2011-12-15 11:06:48 -08:00
|
|
|
ret hash_subtys(h, tps);
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
ty_constr(t, cs) {
|
2011-12-19 13:52:58 +01:00
|
|
|
let h = hash_subty(36u, t);
|
2011-12-07 07:21:07 -08:00
|
|
|
for c: @type_constr in cs { h += (h << 5u) + hash_type_constr(h, c); }
|
2011-07-27 14:19:39 +02:00
|
|
|
ret h;
|
|
|
|
}
|
2011-12-15 11:06:48 -08:00
|
|
|
ty_uniq(mt) { ret hash_subty(37u, mt.ty); }
|
2012-01-18 22:37:22 -08:00
|
|
|
ty_send_type { ret 38u; }
|
2012-01-05 16:19:12 -08:00
|
|
|
ty_named(t, name) { (str::hash(*name) << 5u) + hash_subty(39u, t) }
|
2011-12-20 16:33:55 +01:00
|
|
|
ty_iface(did, tys) {
|
2012-01-05 16:19:12 -08:00
|
|
|
let h = hash_def(40u, did);
|
|
|
|
for typ: t in tys { h = hash_subty(h, typ); }
|
2011-12-20 16:33:55 +01:00
|
|
|
ret h;
|
|
|
|
}
|
2012-01-18 22:37:22 -08:00
|
|
|
ty_opaque_closure_ptr(ck_block) { ret 41u; }
|
|
|
|
ty_opaque_closure_ptr(ck_box) { ret 42u; }
|
|
|
|
ty_opaque_closure_ptr(ck_uniq) { ret 43u; }
|
2011-04-20 14:34:17 -07:00
|
|
|
}
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
|
2011-10-06 12:26:12 +02:00
|
|
|
fn hash_raw_ty(&&rt: @raw_t) -> uint { ret rt.hash; }
|
2011-05-11 17:05:39 -07: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 {
|
|
|
|
ast::carg_lit(m) { ret ast_util::lit_eq(l, m); } _ { ret false; }
|
|
|
|
}
|
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 {
|
2011-07-27 14:19:39 +02:00
|
|
|
let 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; }
|
2011-07-27 14:19:39 +02:00
|
|
|
let 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-05-13 16:40:21 -07:00
|
|
|
// Type lookups
|
2011-09-12 11:27:30 +02:00
|
|
|
fn node_id_to_ty_param_substs_opt_and_ty(cx: ctxt, id: ast::node_id) ->
|
2011-06-15 11:19:50 -07:00
|
|
|
ty_param_substs_opt_and_ty {
|
2011-06-07 13:50:30 -07:00
|
|
|
// Pull out the node type table.
|
2011-07-27 14:19:39 +02:00
|
|
|
alt smallintmap::find(*cx.node_types, id as uint) {
|
2012-01-18 22:37:22 -08:00
|
|
|
none {
|
2011-09-02 15:34:58 -07:00
|
|
|
cx.sess.bug("node_id_to_ty_param_substs_opt_and_ty() called on " +
|
2011-12-13 16:25:51 -08:00
|
|
|
"an untyped node (" + int::to_str(id, 10u) +
|
2011-09-02 15:34:58 -07:00
|
|
|
")");
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
some(tpot) { ret tpot; }
|
2011-05-13 13:05:55 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn node_id_to_type(cx: ctxt, id: ast::node_id) -> t {
|
2011-07-26 14:06:02 +02:00
|
|
|
ret node_id_to_ty_param_substs_opt_and_ty(cx, id).ty;
|
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] {
|
2011-07-27 14:19:39 +02:00
|
|
|
alt node_id_to_ty_param_substs_opt_and_ty(cx, id).substs {
|
2012-01-18 22:37:22 -08:00
|
|
|
none { ret []; }
|
2011-07-27 14:19:39 +02:00
|
|
|
some(tps) { ret tps; }
|
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 {
|
2011-08-15 16:38:23 -07:00
|
|
|
ret vec::len(node_id_to_type_params(cx, id)) > 0u;
|
2011-05-13 16:53:35 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-01-18 22:37:22 -08:00
|
|
|
// Returns a type with type parameter substitutions performed if applicable
|
2011-09-12 11:27:30 +02:00
|
|
|
fn ty_param_substs_opt_and_ty_to_monotype(cx: ctxt,
|
|
|
|
tpot: ty_param_substs_opt_and_ty) ->
|
|
|
|
t {
|
2011-07-27 14:19:39 +02:00
|
|
|
alt tpot.substs {
|
2012-01-18 22:37:22 -08:00
|
|
|
none { ret tpot.ty; }
|
2011-07-27 14:19:39 +02:00
|
|
|
some(tps) { ret substitute_type_params(cx, tps, tpot.ty); }
|
2011-05-20 18:36:35 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2011-04-08 21:27:54 -07:00
|
|
|
// Returns the type of an annotation, with type parameter substitutions
|
2012-01-18 22:37:22 -08:00
|
|
|
// performed if applicable
|
2011-09-12 11:27:30 +02:00
|
|
|
fn node_id_to_monotype(cx: ctxt, id: ast::node_id) -> t {
|
2011-07-27 14:19:39 +02:00
|
|
|
let tpot = node_id_to_ty_param_substs_opt_and_ty(cx, id);
|
2011-05-20 18:36:35 -07:00
|
|
|
ret ty_param_substs_opt_and_ty_to_monotype(cx, tpot);
|
2011-04-08 21:27:54 -07:00
|
|
|
}
|
|
|
|
|
2011-05-13 16:40:21 -07:00
|
|
|
|
2011-04-12 15:09:50 -07:00
|
|
|
// Returns the number of distinct type parameters in the given type.
|
2011-09-12 11:27:30 +02:00
|
|
|
fn count_ty_params(cx: ctxt, ty: t) -> uint {
|
|
|
|
fn counter(cx: ctxt, param_indices: @mutable [uint], ty: t) {
|
2011-07-27 14:19:39 +02:00
|
|
|
alt struct(cx, ty) {
|
2011-08-19 15:16:48 -07:00
|
|
|
ty_param(param_idx, _) {
|
2011-07-27 14:19:39 +02:00
|
|
|
let seen = false;
|
2011-08-15 21:54:52 -07:00
|
|
|
for other_param_idx: uint in *param_indices {
|
2011-07-27 14:19:39 +02:00
|
|
|
if param_idx == other_param_idx { seen = true; }
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
2011-08-19 15:16:48 -07:00
|
|
|
if !seen { *param_indices += [param_idx]; }
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
_ {/* fall through */ }
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
}
|
2011-08-19 15:16:48 -07:00
|
|
|
let param_indices: @mutable [uint] = @mutable [];
|
2011-07-27 14:19:39 +02:00
|
|
|
let f = bind counter(cx, param_indices, _);
|
2011-04-25 12:15:55 -07:00
|
|
|
walk_ty(cx, f, ty);
|
2011-08-13 00:09:25 -07:00
|
|
|
ret vec::len::<uint>(*param_indices);
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn type_contains_vars(cx: ctxt, typ: t) -> bool {
|
2011-07-21 11:43:52 -07:00
|
|
|
ret interner::get(*cx.ts, typ).has_vars;
|
2011-04-08 21:27:54 -07:00
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn type_contains_params(cx: ctxt, typ: t) -> bool {
|
2011-07-21 11:43:52 -07:00
|
|
|
ret interner::get(*cx.ts, typ).has_params;
|
2011-04-25 05:39:18 +00:00
|
|
|
}
|
|
|
|
|
2011-01-03 18:22:39 -08:00
|
|
|
|
2011-06-15 11:19:50 -07:00
|
|
|
// Type accessors for substructures of types
|
2011-09-12 11:27:30 +02:00
|
|
|
fn ty_fn_args(cx: ctxt, fty: t) -> [arg] {
|
2011-07-27 14:19:39 +02:00
|
|
|
alt struct(cx, fty) {
|
2011-12-23 16:09:52 +01:00
|
|
|
ty::ty_fn(f) { ret f.inputs; }
|
2011-11-21 02:15:40 +08:00
|
|
|
ty::ty_native_fn(a, _) { ret a; }
|
2011-09-02 15:34:58 -07:00
|
|
|
_ { cx.sess.bug("ty_fn_args() called on non-fn type"); }
|
2011-02-18 17:30:57 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn ty_fn_proto(cx: ctxt, fty: t) -> ast::proto {
|
2011-07-29 20:51:18 +02:00
|
|
|
alt struct(cx, fty) {
|
2011-12-23 16:09:52 +01:00
|
|
|
ty::ty_fn(f) { ret f.proto; }
|
2011-11-21 02:15:40 +08:00
|
|
|
ty::ty_native_fn(_, _) {
|
2011-10-18 21:18:55 -07:00
|
|
|
// FIXME: This should probably be proto_bare
|
2012-01-10 06:49:15 -08:00
|
|
|
ret ast::proto_box;
|
2011-10-18 21:18:55 -07:00
|
|
|
}
|
2011-09-02 15:34:58 -07:00
|
|
|
_ { cx.sess.bug("ty_fn_proto() called on non-fn type"); }
|
2011-07-29 20:51:18 +02:00
|
|
|
}
|
2011-01-03 18:22:39 -08:00
|
|
|
}
|
|
|
|
|
2011-09-16 13:03:11 -07:00
|
|
|
pure fn ty_fn_ret(cx: ctxt, fty: t) -> t {
|
2011-09-24 16:33:26 -07:00
|
|
|
let sty = struct(cx, fty);
|
2011-09-16 13:03:11 -07:00
|
|
|
alt sty {
|
2011-12-23 16:09:52 +01:00
|
|
|
ty::ty_fn(f) { ret f.output; }
|
2011-11-21 02:15:40 +08:00
|
|
|
ty::ty_native_fn(_, r) { ret r; }
|
2011-09-16 13:03:11 -07:00
|
|
|
_ {
|
|
|
|
// Unchecked is ok since we diverge here
|
|
|
|
// (might want to change the typechecker to allow
|
|
|
|
// it without an unchecked)
|
|
|
|
// Or, it wouldn't be necessary if we had the right
|
|
|
|
// typestate constraint on cx and t (then we could
|
|
|
|
// call unreachable() instead)
|
|
|
|
unchecked { cx.sess.bug("ty_fn_ret() called on non-fn type"); }}
|
2011-02-18 17:30:57 -08:00
|
|
|
}
|
2011-01-03 18:22:39 -08:00
|
|
|
}
|
|
|
|
|
2011-09-14 14:34:50 +02:00
|
|
|
fn ty_fn_ret_style(cx: ctxt, fty: t) -> ast::ret_style {
|
|
|
|
alt struct(cx, fty) {
|
2011-12-23 16:09:52 +01:00
|
|
|
ty::ty_fn(f) { f.ret_style }
|
2011-11-21 02:15:40 +08:00
|
|
|
ty::ty_native_fn(_, _) { ast::return_val }
|
2011-09-14 14:34:50 +02:00
|
|
|
_ { cx.sess.bug("ty_fn_ret_style() called on non-fn type"); }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn is_fn_ty(cx: ctxt, fty: t) -> bool {
|
2011-07-27 14:19:39 +02:00
|
|
|
alt struct(cx, fty) {
|
2011-12-23 16:09:52 +01:00
|
|
|
ty::ty_fn(_) { ret true; }
|
2011-11-21 02:15:40 +08:00
|
|
|
ty::ty_native_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.
|
2011-09-12 11:27:30 +02:00
|
|
|
fn is_pred_ty(cx: ctxt, fty: t) -> bool {
|
2011-08-24 17:24:58 -07:00
|
|
|
is_fn_ty(cx, fty) && type_is_bool(cx, ty_fn_ret(cx, fty))
|
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn ty_var_id(cx: ctxt, typ: t) -> int {
|
2011-07-27 14:19:39 +02:00
|
|
|
alt struct(cx, typ) {
|
|
|
|
ty::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 {
|
2011-07-04 21:53:33 +02:00
|
|
|
ret node_id_to_monotype(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 {
|
2011-06-21 22:16:40 +02:00
|
|
|
ret node_id_to_monotype(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 {
|
|
|
|
ast::expr_path(_) | ast::expr_index(_, _) |
|
2012-01-18 22:37:22 -08:00
|
|
|
ast::expr_unary(ast::deref, _) { true }
|
2012-01-13 10:58:31 +01:00
|
|
|
ast::expr_field(base, ident, _) { !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
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-12-18 19:30:40 +01:00
|
|
|
fn field_idx(id: ast::ident, fields: [field]) -> option::t<uint> {
|
|
|
|
let i = 0u;
|
|
|
|
for f in fields { if f.ident == id { ret some(i); } i += 1u; }
|
|
|
|
ret none;
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
|
2011-09-13 12:14:30 +02:00
|
|
|
fn get_field(tcx: ctxt, rec_ty: t, id: ast::ident) -> field {
|
|
|
|
alt struct(tcx, rec_ty) {
|
|
|
|
ty_rec(fields) {
|
2011-12-16 06:27:50 -08:00
|
|
|
alt vec::find(fields, {|f| str::eq(f.ident, id) }) {
|
2011-09-13 12:14:30 +02:00
|
|
|
some(f) { ret f; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-12-18 19:30:40 +01:00
|
|
|
fn method_idx(id: ast::ident, meths: [method]) -> option::t<uint> {
|
|
|
|
let i = 0u;
|
|
|
|
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 {
|
2011-09-01 17:27:58 -07:00
|
|
|
ret str::lteq(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
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn occurs_check_fails(tcx: ctxt, sp: option::t<span>, vid: int, rt: t) ->
|
2011-08-19 15:16:48 -07:00
|
|
|
bool {
|
|
|
|
if !type_contains_vars(tcx, rt) {
|
2011-08-04 15:56:40 -07:00
|
|
|
// Fast path
|
|
|
|
ret false;
|
|
|
|
}
|
2011-08-19 15:16:48 -07:00
|
|
|
|
2011-08-03 18:06:57 -07:00
|
|
|
// Occurs check!
|
2011-08-15 16:38:23 -07:00
|
|
|
if vec::member(vid, vars_in_type(tcx, rt)) {
|
2011-08-03 18:06:57 -07:00
|
|
|
alt sp {
|
2011-08-19 15:16:48 -07:00
|
|
|
some(s) {
|
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
|
|
|
|
(s, "Type inference failed because I \
|
|
|
|
could not find a type\n that's both of the form "
|
|
|
|
+ ty_to_str(tcx, ty::mk_var(tcx, vid)) +
|
|
|
|
" and of the form " + ty_to_str(tcx, rt) +
|
|
|
|
". Such a type would have to be infinitely large.");
|
2011-08-03 18:06:57 -07:00
|
|
|
}
|
|
|
|
_ { ret true; }
|
|
|
|
}
|
2011-08-19 15:16:48 -07:00
|
|
|
} else { ret false; }
|
2011-08-03 18:06:57 -07:00
|
|
|
}
|
2011-06-15 11:19:50 -07:00
|
|
|
|
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 {
|
2011-06-19 02:34:20 -07:00
|
|
|
|
|
|
|
export fixup_result;
|
|
|
|
export fixup_vars;
|
|
|
|
export fix_ok;
|
|
|
|
export fix_err;
|
|
|
|
export mk_var_bindings;
|
|
|
|
export resolve_type_structure;
|
|
|
|
export resolve_type_var;
|
|
|
|
export result;
|
|
|
|
export unify;
|
|
|
|
export ures_ok;
|
|
|
|
export ures_err;
|
|
|
|
export var_bindings;
|
2012-01-04 11:32:26 +01:00
|
|
|
export precise, in_bindings;
|
2011-06-19 02:34:20 -07:00
|
|
|
|
2012-01-19 17:56:05 -08:00
|
|
|
enum result { ures_ok(t), ures_err(type_err), }
|
|
|
|
enum union_result { unres_ok, unres_err(type_err), }
|
2012-01-19 14:24:03 -08:00
|
|
|
enum fixup_result {
|
2012-01-19 17:56:05 -08:00
|
|
|
fix_ok(t), // fixup succeeded
|
|
|
|
fix_err(int), // fixup failed because a type variable was unresolved
|
2011-05-19 16:06:23 -07:00
|
|
|
}
|
2011-06-15 11:19:50 -07:00
|
|
|
type var_bindings =
|
2011-08-12 07:15:18 -07:00
|
|
|
{sets: ufind::ufind, types: smallintmap::smallintmap<t>};
|
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,
|
|
|
|
in_bindings(@var_bindings),
|
2011-12-29 13:12:52 +01:00
|
|
|
}
|
|
|
|
type ctxt = {st: unify_style, tcx: ty_ctxt};
|
2011-05-18 11:53:26 -07:00
|
|
|
|
2011-05-20 18:36:35 -07:00
|
|
|
fn mk_var_bindings() -> @var_bindings {
|
2011-08-13 00:09:25 -07:00
|
|
|
ret @{sets: ufind::make(), types: smallintmap::mk::<t>()};
|
2011-05-18 11:53:26 -07:00
|
|
|
}
|
|
|
|
|
2011-05-20 18:36:35 -07:00
|
|
|
// Unifies two sets.
|
2011-10-17 18:12:25 -07:00
|
|
|
fn union(cx: @ctxt, set_a: uint, set_b: uint,
|
|
|
|
variance: variance) -> union_result {
|
2011-12-29 13:12:52 +01:00
|
|
|
let vb = alt cx.st {
|
|
|
|
in_bindings(vb) { vb }
|
|
|
|
};
|
2012-01-05 17:11:34 +01:00
|
|
|
ufind::grow(vb.sets, math::max(set_a, set_b) + 1u);
|
2011-12-22 14:52:30 +01:00
|
|
|
let root_a = ufind::find(vb.sets, set_a);
|
|
|
|
let root_b = ufind::find(vb.sets, set_b);
|
2011-07-27 14:19:39 +02:00
|
|
|
|
2012-01-11 09:58:05 -08:00
|
|
|
let replace_type = (
|
|
|
|
fn@(vb: @var_bindings, t: t) {
|
|
|
|
ufind::union(vb.sets, set_a, set_b);
|
|
|
|
let root_c: uint = ufind::find(vb.sets, set_a);
|
|
|
|
smallintmap::insert::<t>(vb.types, root_c, t);
|
|
|
|
}
|
|
|
|
);
|
2011-07-27 14:19:39 +02:00
|
|
|
|
|
|
|
|
2011-12-22 14:52:30 +01:00
|
|
|
alt smallintmap::find(vb.types, root_a) {
|
2012-01-18 22:37:22 -08:00
|
|
|
none {
|
2011-12-22 14:52:30 +01:00
|
|
|
alt smallintmap::find(vb.types, root_b) {
|
2012-01-18 22:37:22 -08:00
|
|
|
none { ufind::union(vb.sets, set_a, set_b); ret unres_ok; }
|
2011-12-22 14:52:30 +01:00
|
|
|
some(t_b) { replace_type(vb, t_b); ret unres_ok; }
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
some(t_a) {
|
2011-12-22 14:52:30 +01:00
|
|
|
alt smallintmap::find(vb.types, root_b) {
|
2012-01-18 22:37:22 -08:00
|
|
|
none { replace_type(vb, t_a); ret unres_ok; }
|
2011-07-27 14:19:39 +02:00
|
|
|
some(t_b) {
|
2011-10-17 18:12:25 -07:00
|
|
|
alt unify_step(cx, t_a, t_b, variance) {
|
2011-12-22 14:52:30 +01:00
|
|
|
ures_ok(t_c) { replace_type(vb, t_c); ret unres_ok; }
|
2011-07-27 14:19:39 +02:00
|
|
|
ures_err(terr) { ret unres_err(terr); }
|
2011-05-19 16:06:23 -07:00
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2011-05-19 16:06:23 -07:00
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2011-05-19 16:06:23 -07:00
|
|
|
}
|
2011-05-20 18:36:35 -07:00
|
|
|
}
|
2011-10-17 15:06:19 -07:00
|
|
|
|
|
|
|
fn record_var_binding_for_expected(
|
2011-10-17 18:12:25 -07:00
|
|
|
cx: @ctxt, key: int, typ: t, variance: variance) -> result {
|
2011-10-17 15:06:19 -07:00
|
|
|
record_var_binding(
|
2011-10-17 20:51:34 -07:00
|
|
|
cx, key, typ, variance_transform(variance, covariant))
|
2011-10-17 15:06:19 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
fn record_var_binding_for_actual(
|
2011-10-17 18:12:25 -07:00
|
|
|
cx: @ctxt, key: int, typ: t, variance: variance) -> result {
|
2011-10-17 20:51:34 -07:00
|
|
|
// Unifying in 'the other direction' so flip the variance
|
2011-10-17 15:06:19 -07:00
|
|
|
record_var_binding(
|
2011-10-17 20:51:34 -07:00
|
|
|
cx, key, typ, variance_transform(variance, contravariant))
|
2011-10-17 15:06:19 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
fn record_var_binding(
|
2011-10-17 20:51:34 -07:00
|
|
|
cx: @ctxt, key: int, typ: t, variance: variance) -> result {
|
2011-10-17 15:06:19 -07:00
|
|
|
|
2011-12-29 13:12:52 +01:00
|
|
|
let vb = alt cx.st { in_bindings(vb) { vb } };
|
2011-12-22 14:52:30 +01:00
|
|
|
ufind::grow(vb.sets, (key as uint) + 1u);
|
|
|
|
let root = ufind::find(vb.sets, key as uint);
|
2011-07-27 14:19:39 +02:00
|
|
|
let result_type = typ;
|
2011-12-22 14:52:30 +01:00
|
|
|
alt smallintmap::find(vb.types, root) {
|
2011-07-27 14:19:39 +02:00
|
|
|
some(old_type) {
|
2011-10-17 20:51:34 -07:00
|
|
|
alt unify_step(cx, old_type, typ, variance) {
|
2011-07-27 14:19:39 +02:00
|
|
|
ures_ok(unified_type) { result_type = unified_type; }
|
|
|
|
rs { ret rs; }
|
2011-05-20 18:36:35 -07:00
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2012-01-18 22:37:22 -08:00
|
|
|
none {/* fall through */ }
|
2011-05-20 18:36:35 -07:00
|
|
|
}
|
2011-12-22 14:52:30 +01:00
|
|
|
smallintmap::insert::<t>(vb.types, root, result_type);
|
2011-05-19 16:06:23 -07:00
|
|
|
ret ures_ok(typ);
|
|
|
|
}
|
|
|
|
|
2010-12-21 17:47:13 -08:00
|
|
|
// Simple structural type comparison.
|
2011-07-27 14:19:39 +02:00
|
|
|
fn struct_cmp(cx: @ctxt, expected: t, actual: t) -> result {
|
2011-12-23 01:57:35 +01:00
|
|
|
let tcx = cx.tcx;
|
2012-01-12 17:59:49 +01:00
|
|
|
let cfg = tcx.sess.targ_cfg;
|
2011-12-23 01:57:35 +01:00
|
|
|
if mach_struct(tcx, cfg, expected) == mach_struct(tcx, cfg, actual) {
|
2010-12-21 17:47:13 -08:00
|
|
|
ret ures_ok(expected);
|
|
|
|
}
|
2011-05-20 18:36:35 -07:00
|
|
|
ret ures_err(terr_mismatch);
|
2010-12-21 17:47:13 -08:00
|
|
|
}
|
|
|
|
|
2011-07-19 17:52:34 -07:00
|
|
|
// Right now this just checks that the lists of constraints are
|
|
|
|
// pairwise equal.
|
2011-08-22 12:45:18 +02:00
|
|
|
fn unify_constrs(base_t: t, expected: [@type_constr],
|
2011-09-12 11:27:30 +02:00
|
|
|
actual: [@type_constr]) -> result {
|
2011-08-15 16:38:23 -07:00
|
|
|
let expected_len = vec::len(expected);
|
|
|
|
let actual_len = vec::len(actual);
|
2011-07-19 17:52:34 -07:00
|
|
|
|
2011-07-27 14:19:39 +02:00
|
|
|
if expected_len != actual_len {
|
2011-07-19 17:52:34 -07:00
|
|
|
ret ures_err(terr_constr_len(expected_len, actual_len));
|
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
let i = 0u;
|
|
|
|
let rslt;
|
2011-08-15 21:54:52 -07:00
|
|
|
for c: @type_constr in expected {
|
2011-08-19 15:16:48 -07:00
|
|
|
rslt = unify_constr(base_t, c, actual[i]);
|
2011-07-27 14:19:39 +02:00
|
|
|
alt rslt { ures_ok(_) { } ures_err(_) { ret rslt; } }
|
2011-07-19 17:52:34 -07:00
|
|
|
i += 1u;
|
|
|
|
}
|
|
|
|
ret ures_ok(base_t);
|
|
|
|
}
|
2011-08-22 12:45:18 +02:00
|
|
|
fn unify_constr(base_t: t, expected: @type_constr,
|
2011-09-12 11:27:30 +02:00
|
|
|
actual_constr: @type_constr) -> result {
|
2011-07-27 14:19:39 +02:00
|
|
|
let ok_res = ures_ok(base_t);
|
|
|
|
let err_res = ures_err(terr_constr_mismatch(expected, actual_constr));
|
|
|
|
if expected.node.id != actual_constr.node.id { ret err_res; }
|
2011-08-15 16:38:23 -07:00
|
|
|
let expected_arg_len = vec::len(expected.node.args);
|
|
|
|
let actual_arg_len = vec::len(actual_constr.node.args);
|
2011-07-27 14:19:39 +02:00
|
|
|
if expected_arg_len != actual_arg_len { ret err_res; }
|
|
|
|
let i = 0u;
|
|
|
|
let actual;
|
2011-08-15 21:54:52 -07:00
|
|
|
for a: @ty_constr_arg in expected.node.args {
|
2011-08-19 15:16:48 -07:00
|
|
|
actual = actual_constr.node.args[i];
|
2011-07-27 14:19:39 +02:00
|
|
|
alt a.node {
|
2012-01-18 22:37:22 -08:00
|
|
|
carg_base {
|
|
|
|
alt actual.node { carg_base { } _ { ret err_res; } }
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
carg_lit(l) {
|
|
|
|
alt actual.node {
|
|
|
|
carg_lit(m) { if l != m { ret err_res; } }
|
|
|
|
_ { ret err_res; }
|
2011-07-19 17:52:34 -07:00
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
carg_ident(p) {
|
|
|
|
alt actual.node {
|
|
|
|
carg_ident(q) { if p.node != q.node { ret err_res; } }
|
|
|
|
_ { ret err_res; }
|
2011-07-19 17:52:34 -07:00
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2011-07-19 17:52:34 -07:00
|
|
|
}
|
|
|
|
i += 1u;
|
|
|
|
}
|
|
|
|
ret ok_res;
|
|
|
|
}
|
|
|
|
|
2011-03-18 11:49:06 -07:00
|
|
|
// Unifies two mutability flags.
|
2011-10-17 20:51:34 -07:00
|
|
|
fn unify_mut(expected: ast::mutability, actual: ast::mutability,
|
|
|
|
variance: variance) ->
|
|
|
|
option::t<(ast::mutability, variance)> {
|
|
|
|
|
2011-10-18 10:35:33 -07:00
|
|
|
// If you're unifying on something mutable then we have to
|
|
|
|
// be invariant on the inner type
|
|
|
|
let newvariance = alt expected {
|
2012-01-18 22:37:22 -08:00
|
|
|
ast::mut {
|
2011-10-18 10:35:33 -07:00
|
|
|
variance_transform(variance, invariant)
|
|
|
|
}
|
|
|
|
_ {
|
|
|
|
variance_transform(variance, covariant)
|
|
|
|
}
|
|
|
|
};
|
2011-10-17 20:51:34 -07:00
|
|
|
|
|
|
|
if expected == actual { ret some((expected, newvariance)); }
|
|
|
|
if variance == covariant {
|
|
|
|
if expected == ast::maybe_mut {
|
|
|
|
ret some((actual, newvariance));
|
|
|
|
}
|
|
|
|
} else if variance == contravariant {
|
|
|
|
if actual == ast::maybe_mut {
|
|
|
|
ret some((expected, newvariance));
|
|
|
|
}
|
|
|
|
}
|
2011-07-21 15:46:03 -07:00
|
|
|
ret none;
|
2011-03-18 11:49:06 -07:00
|
|
|
}
|
2011-10-14 17:34:41 -07:00
|
|
|
fn unify_fn_proto(e_proto: ast::proto, a_proto: ast::proto,
|
|
|
|
variance: variance) -> option::t<result> {
|
2011-12-16 13:50:22 -08:00
|
|
|
// Prototypes form a diamond-shaped partial order:
|
|
|
|
//
|
|
|
|
// block
|
|
|
|
// ^ ^
|
|
|
|
// shared send
|
|
|
|
// ^ ^
|
|
|
|
// bare
|
|
|
|
//
|
|
|
|
// where "^" means "subtype of" (forgive the abuse of the term
|
|
|
|
// subtype).
|
|
|
|
fn sub_proto(p_sub: ast::proto, p_sup: ast::proto) -> bool {
|
|
|
|
ret alt (p_sub, p_sup) {
|
2012-01-18 22:37:22 -08:00
|
|
|
(_, ast::proto_any) { true }
|
|
|
|
(ast::proto_bare, _) { true }
|
2011-12-16 13:50:22 -08:00
|
|
|
|
2012-01-09 16:12:37 -08:00
|
|
|
// Equal prototypes are always subprotos:
|
2011-12-16 13:50:22 -08:00
|
|
|
(_, _) { p_sub == p_sup }
|
2011-12-09 16:56:48 -08:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2011-12-16 13:50:22 -08:00
|
|
|
ret alt variance {
|
2012-01-18 22:37:22 -08:00
|
|
|
invariant if e_proto == a_proto { none }
|
|
|
|
covariant if sub_proto(a_proto, e_proto) { none }
|
|
|
|
contravariant if sub_proto(e_proto, a_proto) { none }
|
2011-12-16 13:50:22 -08:00
|
|
|
_ { some(ures_err(terr_mismatch)) }
|
|
|
|
};
|
2011-10-14 17:34:41 -07:00
|
|
|
}
|
2011-12-23 16:09:52 +01:00
|
|
|
fn unify_args(cx: @ctxt, e_args: [arg], a_args: [arg], variance: variance)
|
|
|
|
-> either::t<result, [arg]> {
|
|
|
|
if !vec::same_length(e_args, a_args) {
|
|
|
|
ret either::left(ures_err(terr_arg_count));
|
|
|
|
}
|
|
|
|
// The variance changes (flips basically) when descending
|
|
|
|
// into arguments of function types
|
|
|
|
let variance = variance_transform(variance, contravariant);
|
|
|
|
// Would use vec::map2(), but for the need to return in case of
|
|
|
|
// error:
|
|
|
|
let i = 0u, result = [];
|
|
|
|
for expected_input in e_args {
|
|
|
|
let actual_input = a_args[i];
|
|
|
|
i += 1u;
|
|
|
|
// Unify the result modes.
|
|
|
|
let result_mode = if expected_input.mode == ast::mode_infer {
|
|
|
|
actual_input.mode
|
|
|
|
} else if actual_input.mode == ast::mode_infer {
|
|
|
|
expected_input.mode
|
|
|
|
} else if expected_input.mode != actual_input.mode {
|
|
|
|
ret either::left(ures_err(terr_mode_mismatch(
|
|
|
|
expected_input.mode, actual_input.mode)));
|
|
|
|
} else { expected_input.mode };
|
2011-10-14 17:34:41 -07:00
|
|
|
|
2011-12-23 16:09:52 +01:00
|
|
|
alt unify_step(cx, expected_input.ty, actual_input.ty,
|
|
|
|
variance) {
|
|
|
|
ures_ok(rty) { result += [{mode: result_mode, ty: rty}]; }
|
|
|
|
err { ret either::left(err); }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
either::right(result)
|
|
|
|
}
|
|
|
|
fn unify_fn(cx: @ctxt, e_f: fn_ty, a_f: fn_ty, variance: variance)
|
|
|
|
-> result {
|
|
|
|
alt unify_fn_proto(e_f.proto, a_f.proto, variance) {
|
2011-10-14 17:34:41 -07:00
|
|
|
some(err) { ret err; }
|
2012-01-18 22:37:22 -08:00
|
|
|
none { /* fall through */ }
|
2011-10-14 17:34:41 -07:00
|
|
|
}
|
|
|
|
|
2011-12-23 16:09:52 +01:00
|
|
|
if a_f.ret_style != ast::noreturn && a_f.ret_style != e_f.ret_style {
|
2011-09-14 11:01:42 +02:00
|
|
|
/* even though typestate checking is mostly
|
|
|
|
responsible for checking control flow annotations,
|
|
|
|
this check is necessary to ensure that the
|
|
|
|
annotation in an object method matches the
|
|
|
|
declared object type */
|
2011-12-23 16:09:52 +01:00
|
|
|
ret ures_err(terr_ret_style_mismatch(e_f.ret_style,
|
|
|
|
a_f.ret_style));
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2011-12-23 16:09:52 +01:00
|
|
|
let result_ins = alt unify_args(cx, e_f.inputs, a_f.inputs,
|
|
|
|
variance) {
|
|
|
|
either::left(err) { ret err; }
|
|
|
|
either::right(ts) { ts }
|
|
|
|
};
|
|
|
|
|
|
|
|
// Check the output.
|
|
|
|
alt unify_step(cx, e_f.output, a_f.output, variance) {
|
|
|
|
ures_ok(rty) {
|
|
|
|
ures_ok(mk_fn(cx.tcx, {proto: e_f.proto,
|
|
|
|
inputs: result_ins,
|
|
|
|
output: rty
|
|
|
|
with a_f}))
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2011-12-23 16:09:52 +01:00
|
|
|
x { x }
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
}
|
2011-12-23 16:09:52 +01:00
|
|
|
fn unify_native_fn(cx: @ctxt, expected_inputs: [arg], expected_output: t,
|
2011-10-17 18:32:49 -07:00
|
|
|
actual_inputs: [arg], actual_output: t,
|
|
|
|
variance: variance) -> result {
|
2011-12-23 16:09:52 +01:00
|
|
|
let result_ins = alt unify_args(cx, expected_inputs,
|
|
|
|
actual_inputs, variance) {
|
|
|
|
either::left(err) { ret err; }
|
|
|
|
either::right(ts) { ts }
|
|
|
|
};
|
|
|
|
alt unify_step(cx, expected_output, actual_output, variance) {
|
|
|
|
ures_ok(out) { ures_ok(mk_native_fn(cx.tcx, result_ins, out)) }
|
|
|
|
err { err }
|
2011-02-23 11:59:07 -05:00
|
|
|
}
|
2010-12-31 09:48:54 -08:00
|
|
|
}
|
2010-12-31 10:26:29 -08:00
|
|
|
|
2011-06-08 18:39:34 -07:00
|
|
|
// If the given type is a variable, returns the structure of that type.
|
2011-09-12 11:27:30 +02:00
|
|
|
fn resolve_type_structure(tcx: ty_ctxt, vb: @var_bindings, typ: t) ->
|
2011-06-15 11:19:50 -07:00
|
|
|
fixup_result {
|
2011-07-27 14:19:39 +02:00
|
|
|
alt struct(tcx, typ) {
|
|
|
|
ty_var(vid) {
|
2011-08-19 15:16:48 -07:00
|
|
|
if vid as uint >= ufind::set_count(vb.sets) { ret fix_err(vid); }
|
2011-08-15 16:21:55 -07:00
|
|
|
let root_id = ufind::find(vb.sets, vid as uint);
|
2011-08-13 00:09:25 -07:00
|
|
|
alt smallintmap::find::<t>(vb.types, root_id) {
|
2012-01-18 22:37:22 -08:00
|
|
|
none { ret fix_err(vid); }
|
2011-07-27 14:19:39 +02:00
|
|
|
some(rt) { ret fix_ok(rt); }
|
2011-06-08 18:39:34 -07:00
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
_ { ret fix_ok(typ); }
|
2011-06-08 18:39:34 -07:00
|
|
|
}
|
|
|
|
}
|
2011-10-17 18:12:25 -07:00
|
|
|
|
|
|
|
// Specifies the allowable subtyping between expected and actual types
|
2012-01-19 14:24:03 -08:00
|
|
|
enum variance {
|
2011-10-17 18:12:25 -07:00
|
|
|
// Actual may be a subtype of expected
|
2012-01-19 17:56:05 -08:00
|
|
|
covariant,
|
2011-10-17 18:12:25 -07:00
|
|
|
// Actual may be a supertype of expected
|
2012-01-19 17:56:05 -08:00
|
|
|
contravariant,
|
2011-10-17 18:12:25 -07:00
|
|
|
// Actual must be the same type as expected
|
2012-01-19 17:56:05 -08:00
|
|
|
invariant,
|
2011-10-17 18:12:25 -07:00
|
|
|
}
|
|
|
|
|
2011-10-17 18:32:49 -07:00
|
|
|
// The calculation for recursive variance
|
|
|
|
// "Taming the Wildcards: Combining Definition- and Use-Site Variance"
|
|
|
|
// by John Altidor, et. al.
|
|
|
|
//
|
|
|
|
// I'm just copying the table from figure 1 - haven't actually
|
|
|
|
// read the paper (yet).
|
|
|
|
fn variance_transform(a: variance, b: variance) -> variance {
|
|
|
|
alt a {
|
2012-01-18 22:37:22 -08:00
|
|
|
covariant {
|
2011-10-17 18:32:49 -07:00
|
|
|
alt b {
|
2012-01-18 22:37:22 -08:00
|
|
|
covariant { covariant }
|
|
|
|
contravariant { contravariant }
|
|
|
|
invariant { invariant }
|
2011-10-17 18:32:49 -07:00
|
|
|
}
|
|
|
|
}
|
2012-01-18 22:37:22 -08:00
|
|
|
contravariant {
|
2011-10-17 18:32:49 -07:00
|
|
|
alt b {
|
2012-01-18 22:37:22 -08:00
|
|
|
covariant { contravariant }
|
|
|
|
contravariant { covariant }
|
|
|
|
invariant { invariant }
|
2011-10-17 18:32:49 -07:00
|
|
|
}
|
|
|
|
}
|
2012-01-18 22:37:22 -08:00
|
|
|
invariant {
|
2011-10-17 18:32:49 -07:00
|
|
|
alt b {
|
2012-01-18 22:37:22 -08:00
|
|
|
covariant { invariant }
|
|
|
|
contravariant { invariant }
|
|
|
|
invariant { invariant }
|
2011-10-17 18:32:49 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-12-20 16:33:55 +01:00
|
|
|
fn unify_tps(cx: @ctxt, expected_tps: [t], actual_tps: [t],
|
2012-01-23 14:59:00 -08:00
|
|
|
variance: variance, finish: fn([t]) -> result) -> result {
|
2011-12-20 16:33:55 +01:00
|
|
|
let result_tps = [], i = 0u;
|
|
|
|
for exp in expected_tps {
|
|
|
|
let act = actual_tps[i];
|
|
|
|
i += 1u;
|
|
|
|
let result = unify_step(cx, exp, act, variance);
|
|
|
|
alt result {
|
|
|
|
ures_ok(rty) { result_tps += [rty]; }
|
|
|
|
_ { ret result; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
finish(result_tps)
|
|
|
|
}
|
2011-10-17 18:12:25 -07:00
|
|
|
fn unify_step(cx: @ctxt, expected: t, actual: t,
|
|
|
|
variance: variance) -> result {
|
2011-12-22 14:24:36 +01:00
|
|
|
// FIXME: rewrite this using tuple pattern matching when available, to
|
2010-12-21 17:47:13 -08:00
|
|
|
// avoid all this rightward drift and spikiness.
|
2011-12-22 14:24:36 +01:00
|
|
|
// NOTE: we have tuple matching now, but that involves copying the
|
|
|
|
// matched elements into a tuple first, which is expensive, since sty
|
|
|
|
// holds vectors, which are currently unique
|
2010-12-21 17:47:13 -08:00
|
|
|
|
2011-04-22 10:51:32 -07:00
|
|
|
// Fast path.
|
2011-12-22 14:24:36 +01:00
|
|
|
if expected == actual { ret ures_ok(expected); }
|
2011-04-22 10:51:32 -07:00
|
|
|
|
2011-05-20 18:36:35 -07:00
|
|
|
// Stage 1: Handle the cases in which one side or another is a type
|
2012-01-18 22:37:22 -08:00
|
|
|
// variable
|
2011-05-26 20:14:00 -07:00
|
|
|
|
2011-07-27 14:19:39 +02:00
|
|
|
alt struct(cx.tcx, actual) {
|
|
|
|
// If the RHS is a variable type, then just do the
|
|
|
|
// appropriate binding.
|
|
|
|
ty::ty_var(actual_id) {
|
|
|
|
let actual_n = actual_id as uint;
|
|
|
|
alt struct(cx.tcx, expected) {
|
|
|
|
ty::ty_var(expected_id) {
|
|
|
|
let expected_n = expected_id as uint;
|
2011-10-17 18:32:49 -07:00
|
|
|
alt union(cx, expected_n, actual_n, variance) {
|
2012-01-18 22:37:22 -08:00
|
|
|
unres_ok {/* fall through */ }
|
2011-07-27 14:19:39 +02:00
|
|
|
unres_err(t_e) { ret ures_err(t_e); }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_ {
|
|
|
|
// Just bind the type variable to the expected type.
|
2011-10-17 18:32:49 -07:00
|
|
|
alt record_var_binding_for_actual(
|
|
|
|
cx, actual_id, expected, variance) {
|
2011-07-27 14:19:39 +02:00
|
|
|
ures_ok(_) {/* fall through */ }
|
|
|
|
rs { ret rs; }
|
2011-04-08 14:53:16 -07:00
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2010-12-21 17:47:13 -08:00
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
ret ures_ok(mk_var(cx.tcx, actual_id));
|
|
|
|
}
|
|
|
|
_ {/* empty */ }
|
2011-05-20 18:36:35 -07:00
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
alt struct(cx.tcx, expected) {
|
|
|
|
ty::ty_var(expected_id) {
|
|
|
|
// Add a binding. (`actual` can't actually be a var here.)
|
2011-10-17 18:32:49 -07:00
|
|
|
alt record_var_binding_for_expected(
|
|
|
|
cx, expected_id, actual,
|
|
|
|
variance) {
|
2011-07-27 14:19:39 +02:00
|
|
|
ures_ok(_) {/* fall through */ }
|
|
|
|
rs { ret rs; }
|
2011-01-04 16:52:06 -08:00
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
ret ures_ok(mk_var(cx.tcx, expected_id));
|
|
|
|
}
|
|
|
|
_ {/* fall through */ }
|
2011-05-20 18:36:35 -07:00
|
|
|
}
|
|
|
|
// Stage 2: Handle all other cases.
|
|
|
|
|
2011-07-27 14:19:39 +02:00
|
|
|
alt struct(cx.tcx, actual) {
|
2012-01-18 22:37:22 -08:00
|
|
|
ty::ty_bot { ret ures_ok(expected); }
|
2011-07-27 14:19:39 +02:00
|
|
|
_ {/* fall through */ }
|
|
|
|
}
|
|
|
|
alt struct(cx.tcx, expected) {
|
2012-01-18 22:37:22 -08:00
|
|
|
ty::ty_nil { ret struct_cmp(cx, expected, actual); }
|
2011-07-27 14:19:39 +02:00
|
|
|
// _|_ unifies with anything
|
2012-01-18 22:37:22 -08:00
|
|
|
ty::ty_bot {
|
2011-07-27 14:19:39 +02:00
|
|
|
ret ures_ok(actual);
|
|
|
|
}
|
2012-01-18 22:37:22 -08:00
|
|
|
ty::ty_bool | ty::ty_int(_) | ty_uint(_) | ty_float(_) |
|
|
|
|
ty::ty_str | ty::ty_type | ty::ty_send_type {
|
2011-12-07 21:06:12 +01:00
|
|
|
ret struct_cmp(cx, expected, actual);
|
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
ty::ty_native(ex_id) {
|
|
|
|
alt struct(cx.tcx, actual) {
|
|
|
|
ty_native(act_id) {
|
|
|
|
if ex_id.crate == act_id.crate && ex_id.node == act_id.node {
|
|
|
|
ret ures_ok(actual);
|
|
|
|
} else { ret ures_err(terr_mismatch); }
|
|
|
|
}
|
|
|
|
_ { ret ures_err(terr_mismatch); }
|
|
|
|
}
|
|
|
|
}
|
2011-12-29 11:23:35 +01:00
|
|
|
ty::ty_param(expected_n, _) {
|
|
|
|
alt struct(cx.tcx, actual) {
|
2012-01-09 19:15:17 -06:00
|
|
|
ty::ty_param(actual_n, _) if expected_n == actual_n {
|
2011-12-29 11:23:35 +01:00
|
|
|
ret ures_ok(expected);
|
|
|
|
}
|
|
|
|
_ { ret ures_err(terr_mismatch); }
|
|
|
|
}
|
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
ty::ty_tag(expected_id, expected_tps) {
|
|
|
|
alt struct(cx.tcx, actual) {
|
|
|
|
ty::ty_tag(actual_id, actual_tps) {
|
2011-12-20 16:33:55 +01:00
|
|
|
if expected_id != actual_id {
|
2011-07-27 14:19:39 +02:00
|
|
|
ret ures_err(terr_mismatch);
|
2011-07-01 18:39:24 +02:00
|
|
|
}
|
2011-12-20 16:33:55 +01:00
|
|
|
ret unify_tps(cx, expected_tps, actual_tps, variance, {|tps|
|
|
|
|
ures_ok(mk_tag(cx.tcx, expected_id, tps))
|
|
|
|
});
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
_ {/* fall through */ }
|
|
|
|
}
|
|
|
|
ret ures_err(terr_mismatch);
|
|
|
|
}
|
2011-12-20 16:33:55 +01:00
|
|
|
ty_iface(expected_id, expected_tps) {
|
|
|
|
alt struct(cx.tcx, actual) {
|
|
|
|
ty::ty_iface(actual_id, actual_tps) {
|
|
|
|
if expected_id != actual_id {
|
|
|
|
ret ures_err(terr_mismatch);
|
|
|
|
}
|
|
|
|
ret unify_tps(cx, expected_tps, actual_tps, variance, {|tps|
|
|
|
|
ures_ok(mk_iface(cx.tcx, expected_id, tps))
|
|
|
|
});
|
|
|
|
}
|
|
|
|
_ {}
|
|
|
|
}
|
|
|
|
ret ures_err(terr_mismatch);
|
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
ty::ty_box(expected_mt) {
|
|
|
|
alt struct(cx.tcx, actual) {
|
|
|
|
ty::ty_box(actual_mt) {
|
2012-01-14 16:05:07 -08:00
|
|
|
let (mutt, var) = alt unify_mut(
|
2011-10-17 20:51:34 -07:00
|
|
|
expected_mt.mut, actual_mt.mut, variance) {
|
2012-01-18 22:37:22 -08:00
|
|
|
none { ret ures_err(terr_box_mutability); }
|
2011-10-17 20:51:34 -07:00
|
|
|
some(mv) { mv }
|
|
|
|
};
|
2011-10-17 18:12:25 -07:00
|
|
|
let result = unify_step(
|
2011-10-17 20:51:34 -07:00
|
|
|
cx, expected_mt.ty, actual_mt.ty, var);
|
2011-07-27 14:19:39 +02:00
|
|
|
alt result {
|
|
|
|
ures_ok(result_sub) {
|
2012-01-14 16:05:07 -08:00
|
|
|
let mt = {ty: result_sub, mut: mutt};
|
2011-07-27 14:19:39 +02:00
|
|
|
ret ures_ok(mk_box(cx.tcx, mt));
|
|
|
|
}
|
|
|
|
_ { ret result; }
|
2010-12-21 17:47:13 -08:00
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
_ { ret ures_err(terr_mismatch); }
|
|
|
|
}
|
|
|
|
}
|
2011-09-21 18:54:54 -07:00
|
|
|
ty::ty_uniq(expected_mt) {
|
2011-08-10 17:23:46 -07:00
|
|
|
alt struct(cx.tcx, actual) {
|
2011-09-21 18:54:54 -07:00
|
|
|
ty::ty_uniq(actual_mt) {
|
2012-01-14 16:05:07 -08:00
|
|
|
let (mutt, var) = alt unify_mut(
|
2011-10-17 20:51:34 -07:00
|
|
|
expected_mt.mut, actual_mt.mut, variance) {
|
2012-01-18 22:37:22 -08:00
|
|
|
none { ret ures_err(terr_box_mutability); }
|
2011-10-17 20:51:34 -07:00
|
|
|
some(mv) { mv }
|
|
|
|
};
|
2011-10-17 18:12:25 -07:00
|
|
|
let result = unify_step(
|
2011-10-17 20:51:34 -07:00
|
|
|
cx, expected_mt.ty, actual_mt.ty, var);
|
2011-08-10 17:23:46 -07:00
|
|
|
alt result {
|
2011-09-21 18:54:54 -07:00
|
|
|
ures_ok(result_mt) {
|
2012-01-14 16:05:07 -08:00
|
|
|
let mt = {ty: result_mt, mut: mutt};
|
2011-09-21 18:54:54 -07:00
|
|
|
ret ures_ok(mk_uniq(cx.tcx, mt));
|
2011-08-10 17:23:46 -07:00
|
|
|
}
|
|
|
|
_ { ret result; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_ { ret ures_err(terr_mismatch); }
|
|
|
|
}
|
|
|
|
}
|
2011-08-18 14:11:06 -07:00
|
|
|
ty::ty_vec(expected_mt) {
|
2011-07-27 14:19:39 +02:00
|
|
|
alt struct(cx.tcx, actual) {
|
2011-08-18 14:11:06 -07:00
|
|
|
ty::ty_vec(actual_mt) {
|
2012-01-14 16:05:07 -08:00
|
|
|
let (mutt, var) = alt unify_mut(
|
2011-10-17 20:51:34 -07:00
|
|
|
expected_mt.mut, actual_mt.mut, variance) {
|
2012-01-18 22:37:22 -08:00
|
|
|
none { ret ures_err(terr_vec_mutability); }
|
2011-10-17 20:51:34 -07:00
|
|
|
some(mv) { mv }
|
|
|
|
};
|
2011-10-17 18:12:25 -07:00
|
|
|
let result = unify_step(
|
2011-10-17 20:51:34 -07:00
|
|
|
cx, expected_mt.ty, actual_mt.ty, var);
|
2011-07-27 14:19:39 +02:00
|
|
|
alt result {
|
|
|
|
ures_ok(result_sub) {
|
2012-01-14 16:05:07 -08:00
|
|
|
let mt = {ty: result_sub, mut: mutt};
|
2011-08-18 14:32:25 -07:00
|
|
|
ret ures_ok(mk_vec(cx.tcx, mt));
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
_ { ret result; }
|
2011-06-28 16:54:47 +02:00
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
_ { ret ures_err(terr_mismatch); }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ty::ty_ptr(expected_mt) {
|
|
|
|
alt struct(cx.tcx, actual) {
|
|
|
|
ty::ty_ptr(actual_mt) {
|
2012-01-14 16:05:07 -08:00
|
|
|
let (mutt, var) = alt unify_mut(
|
2011-10-17 20:51:34 -07:00
|
|
|
expected_mt.mut, actual_mt.mut, variance) {
|
2012-01-18 22:37:22 -08:00
|
|
|
none { ret ures_err(terr_vec_mutability); }
|
2011-10-17 20:51:34 -07:00
|
|
|
some(mv) { mv }
|
|
|
|
};
|
2011-10-17 18:12:25 -07:00
|
|
|
let result = unify_step(
|
2011-10-17 20:51:34 -07:00
|
|
|
cx, expected_mt.ty, actual_mt.ty, var);
|
2011-07-27 14:19:39 +02:00
|
|
|
alt result {
|
|
|
|
ures_ok(result_sub) {
|
2012-01-14 16:05:07 -08:00
|
|
|
let mt = {ty: result_sub, mut: mutt};
|
2011-07-27 14:19:39 +02:00
|
|
|
ret ures_ok(mk_ptr(cx.tcx, mt));
|
|
|
|
}
|
|
|
|
_ { ret result; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_ { ret ures_err(terr_mismatch); }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ty::ty_res(ex_id, ex_inner, ex_tps) {
|
|
|
|
alt struct(cx.tcx, actual) {
|
|
|
|
ty::ty_res(act_id, act_inner, act_tps) {
|
|
|
|
if ex_id.crate != act_id.crate || ex_id.node != act_id.node {
|
|
|
|
ret ures_err(terr_mismatch);
|
|
|
|
}
|
2011-10-17 18:12:25 -07:00
|
|
|
let result = unify_step(
|
|
|
|
cx, ex_inner, act_inner, variance);
|
2011-07-27 14:19:39 +02:00
|
|
|
alt result {
|
|
|
|
ures_ok(res_inner) {
|
|
|
|
let i = 0u;
|
2011-08-19 15:16:48 -07:00
|
|
|
let res_tps = [];
|
2011-08-15 21:54:52 -07:00
|
|
|
for ex_tp: t in ex_tps {
|
2011-10-17 18:12:25 -07:00
|
|
|
let result = unify_step(
|
|
|
|
cx, ex_tp, act_tps[i], variance);
|
2011-07-27 14:19:39 +02:00
|
|
|
alt result {
|
2011-08-19 15:16:48 -07:00
|
|
|
ures_ok(rty) { res_tps += [rty]; }
|
2011-07-27 14:19:39 +02:00
|
|
|
_ { ret result; }
|
2010-12-21 17:47:13 -08:00
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
i += 1u;
|
2010-12-21 17:47:13 -08:00
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
ret ures_ok(mk_res(cx.tcx, act_id, res_inner, res_tps));
|
|
|
|
}
|
|
|
|
_ { ret result; }
|
2010-12-21 17:47:13 -08:00
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
_ { ret ures_err(terr_mismatch); }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ty::ty_rec(expected_fields) {
|
|
|
|
alt struct(cx.tcx, actual) {
|
|
|
|
ty::ty_rec(actual_fields) {
|
2011-08-13 00:09:25 -07:00
|
|
|
let expected_len = vec::len::<field>(expected_fields);
|
|
|
|
let actual_len = vec::len::<field>(actual_fields);
|
2011-07-27 14:19:39 +02:00
|
|
|
if expected_len != actual_len {
|
|
|
|
let err = terr_record_size(expected_len, actual_len);
|
|
|
|
ret ures_err(err);
|
2011-02-23 11:59:07 -05:00
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
// TODO: implement an iterator that can iterate over
|
|
|
|
// two arrays simultaneously.
|
|
|
|
|
2011-08-19 15:16:48 -07:00
|
|
|
let result_fields: [field] = [];
|
2011-07-27 14:19:39 +02:00
|
|
|
let i = 0u;
|
|
|
|
while i < expected_len {
|
2011-08-19 15:16:48 -07:00
|
|
|
let expected_field = expected_fields[i];
|
|
|
|
let actual_field = actual_fields[i];
|
2012-01-14 16:05:07 -08:00
|
|
|
let (mutt, var) = alt unify_mut(
|
2011-10-17 20:51:34 -07:00
|
|
|
expected_field.mt.mut, actual_field.mt.mut, variance)
|
2011-07-27 14:19:39 +02:00
|
|
|
{
|
2012-01-18 22:37:22 -08:00
|
|
|
none { ret ures_err(terr_record_mutability); }
|
2011-10-17 20:51:34 -07:00
|
|
|
some(mv) { mv }
|
|
|
|
};
|
2011-09-01 17:27:58 -07:00
|
|
|
if !str::eq(expected_field.ident, actual_field.ident) {
|
2011-07-27 14:19:39 +02:00
|
|
|
let err =
|
|
|
|
terr_record_fields(expected_field.ident,
|
|
|
|
actual_field.ident);
|
|
|
|
ret ures_err(err);
|
2011-07-19 17:52:34 -07:00
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
let result =
|
|
|
|
unify_step(cx, expected_field.mt.ty,
|
2011-10-17 20:51:34 -07:00
|
|
|
actual_field.mt.ty, var);
|
2011-07-27 14:19:39 +02:00
|
|
|
alt result {
|
|
|
|
ures_ok(rty) {
|
2012-01-14 16:05:07 -08:00
|
|
|
let mt = {ty: rty, mut: mutt};
|
2011-08-19 15:16:48 -07:00
|
|
|
result_fields += [{mt: mt with expected_field}];
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
_ { ret result; }
|
2011-07-19 17:52:34 -07:00
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
i += 1u;
|
2011-07-19 17:52:34 -07:00
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
ret ures_ok(mk_rec(cx.tcx, result_fields));
|
|
|
|
}
|
|
|
|
_ { ret ures_err(terr_mismatch); }
|
|
|
|
}
|
|
|
|
}
|
2011-08-15 11:40:26 +02:00
|
|
|
ty::ty_tup(expected_elems) {
|
|
|
|
alt struct(cx.tcx, actual) {
|
|
|
|
ty::ty_tup(actual_elems) {
|
2011-08-15 16:38:23 -07:00
|
|
|
let expected_len = vec::len(expected_elems);
|
|
|
|
let actual_len = vec::len(actual_elems);
|
2011-08-19 15:16:48 -07:00
|
|
|
if expected_len != actual_len {
|
2011-08-15 11:40:26 +02:00
|
|
|
let err = terr_tuple_size(expected_len, actual_len);
|
|
|
|
ret ures_err(err);
|
|
|
|
}
|
|
|
|
// TODO: implement an iterator that can iterate over
|
|
|
|
// two arrays simultaneously.
|
|
|
|
|
2011-08-19 15:16:48 -07:00
|
|
|
let result_elems = [];
|
2011-08-15 11:40:26 +02:00
|
|
|
let i = 0u;
|
|
|
|
while i < expected_len {
|
2011-08-19 15:16:48 -07:00
|
|
|
let expected_elem = expected_elems[i];
|
|
|
|
let actual_elem = actual_elems[i];
|
2011-10-17 18:12:25 -07:00
|
|
|
let result = unify_step(
|
|
|
|
cx, expected_elem, actual_elem, variance);
|
2011-08-15 11:40:26 +02:00
|
|
|
alt result {
|
2011-08-19 15:16:48 -07:00
|
|
|
ures_ok(rty) { result_elems += [rty]; }
|
2011-08-15 11:40:26 +02:00
|
|
|
_ { ret result; }
|
|
|
|
}
|
|
|
|
i += 1u;
|
|
|
|
}
|
|
|
|
ret ures_ok(mk_tup(cx.tcx, result_elems));
|
|
|
|
}
|
|
|
|
_ { ret ures_err(terr_mismatch); }
|
|
|
|
}
|
|
|
|
}
|
2011-12-23 16:09:52 +01:00
|
|
|
ty::ty_fn(expected_f) {
|
2011-07-27 14:19:39 +02:00
|
|
|
alt struct(cx.tcx, actual) {
|
2011-12-23 16:09:52 +01:00
|
|
|
ty::ty_fn(actual_f) {
|
|
|
|
ret unify_fn(cx, expected_f, actual_f, variance);
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
_ { ret ures_err(terr_mismatch); }
|
|
|
|
}
|
|
|
|
}
|
2011-11-21 02:15:40 +08:00
|
|
|
ty::ty_native_fn(expected_inputs, expected_output) {
|
2011-07-27 14:19:39 +02:00
|
|
|
alt struct(cx.tcx, actual) {
|
2011-11-21 02:15:40 +08:00
|
|
|
ty::ty_native_fn(actual_inputs, actual_output) {
|
2011-12-23 16:09:52 +01:00
|
|
|
ret unify_native_fn(cx, expected_inputs, expected_output,
|
2011-10-17 18:32:49 -07:00
|
|
|
actual_inputs, actual_output, variance);
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
_ { ret ures_err(terr_mismatch); }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ty::ty_constr(expected_t, expected_constrs) {
|
|
|
|
|
|
|
|
// unify the base types...
|
|
|
|
alt struct(cx.tcx, actual) {
|
|
|
|
ty::ty_constr(actual_t, actual_constrs) {
|
2011-10-17 18:12:25 -07:00
|
|
|
let rslt = unify_step(
|
|
|
|
cx, expected_t, actual_t, variance);
|
2011-07-27 14:19:39 +02:00
|
|
|
alt rslt {
|
|
|
|
ures_ok(rty) {
|
|
|
|
// FIXME: probably too restrictive --
|
|
|
|
// requires the constraints to be
|
|
|
|
// syntactically equal
|
|
|
|
ret unify_constrs(expected, expected_constrs,
|
|
|
|
actual_constrs);
|
|
|
|
}
|
|
|
|
_ { ret rslt; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_ {
|
|
|
|
// If the actual type is *not* a constrained type,
|
|
|
|
// then we go ahead and just ignore the constraints on
|
|
|
|
// the expected type. typestate handles the rest.
|
2011-10-17 18:12:25 -07:00
|
|
|
ret unify_step(
|
|
|
|
cx, expected_t, actual, variance);
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2011-07-19 17:52:34 -07:00
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2011-05-20 18:36:35 -07:00
|
|
|
}
|
|
|
|
}
|
2011-12-29 13:12:52 +01:00
|
|
|
fn unify(expected: t, actual: t, st: unify_style,
|
2011-12-22 14:52:30 +01:00
|
|
|
tcx: ty_ctxt) -> result {
|
2011-12-29 13:12:52 +01:00
|
|
|
let cx = @{st: st, tcx: tcx};
|
2011-10-17 18:12:25 -07:00
|
|
|
ret unify_step(cx, expected, actual, covariant);
|
2011-05-20 18:36:35 -07:00
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
fn dump_var_bindings(tcx: ty_ctxt, vb: @var_bindings) {
|
|
|
|
let i = 0u;
|
2011-08-13 00:09:25 -07:00
|
|
|
while i < vec::len::<ufind::node>(vb.sets.nodes) {
|
2011-09-02 15:34:58 -07:00
|
|
|
let sets = "";
|
2011-07-27 14:19:39 +02:00
|
|
|
let j = 0u;
|
2011-08-13 00:09:25 -07:00
|
|
|
while j < vec::len::<option::t<uint>>(vb.sets.nodes) {
|
2011-09-01 18:49:10 -07:00
|
|
|
if ufind::find(vb.sets, j) == i { sets += #fmt[" %u", j]; }
|
2011-05-20 18:36:35 -07:00
|
|
|
j += 1u;
|
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
let typespec;
|
2011-08-13 00:09:25 -07:00
|
|
|
alt smallintmap::find::<t>(vb.types, i) {
|
2012-01-18 22:37:22 -08:00
|
|
|
none { typespec = ""; }
|
2011-09-02 15:34:58 -07:00
|
|
|
some(typ) { typespec = " =" + ty_to_str(tcx, typ); }
|
2010-12-21 17:47:13 -08:00
|
|
|
}
|
2011-12-22 14:42:52 -08:00
|
|
|
#error("set %u:%s%s", i, typespec, sets);
|
2011-05-20 18:36:35 -07:00
|
|
|
i += 1u;
|
2010-12-21 17:47:13 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-05-20 18:36:35 -07:00
|
|
|
// Fixups and substitutions
|
2011-08-03 18:06:57 -07:00
|
|
|
// Takes an optional span - complain about occurs check violations
|
|
|
|
// iff the span is present (so that if we already know we're going
|
|
|
|
// to error anyway, we don't complain)
|
2011-09-12 11:27:30 +02:00
|
|
|
fn fixup_vars(tcx: ty_ctxt, sp: option::t<span>, vb: @var_bindings,
|
2011-08-19 15:16:48 -07:00
|
|
|
typ: t) -> fixup_result {
|
2011-09-12 11:27:30 +02:00
|
|
|
fn subst_vars(tcx: ty_ctxt, sp: option::t<span>, vb: @var_bindings,
|
2011-08-12 07:15:18 -07:00
|
|
|
unresolved: @mutable option::t<int>, vid: int) -> t {
|
2011-08-03 18:06:57 -07:00
|
|
|
// Should really return a fixup_result instead of a t, but fold_ty
|
|
|
|
// doesn't allow returning anything but a t.
|
2011-08-15 16:21:55 -07:00
|
|
|
if vid as uint >= ufind::set_count(vb.sets) {
|
2011-07-21 15:46:03 -07:00
|
|
|
*unresolved = some(vid);
|
2011-06-09 11:20:47 -07:00
|
|
|
ret ty::mk_var(tcx, vid);
|
|
|
|
}
|
2011-08-15 16:21:55 -07:00
|
|
|
let root_id = ufind::find(vb.sets, vid as uint);
|
2011-08-13 00:09:25 -07:00
|
|
|
alt smallintmap::find::<t>(vb.types, root_id) {
|
2012-01-18 22:37:22 -08:00
|
|
|
none { *unresolved = some(vid); ret ty::mk_var(tcx, vid); }
|
2011-07-27 14:19:39 +02:00
|
|
|
some(rt) {
|
2011-08-03 18:06:57 -07:00
|
|
|
if occurs_check_fails(tcx, sp, vid, rt) {
|
2011-08-19 15:16:48 -07:00
|
|
|
// Return the type unchanged, so we can error out
|
|
|
|
// downstream
|
2011-08-03 18:06:57 -07:00
|
|
|
ret rt;
|
|
|
|
}
|
2011-07-27 14:19:39 +02:00
|
|
|
ret fold_ty(tcx,
|
2011-08-19 15:16:48 -07:00
|
|
|
fm_var(bind subst_vars(tcx, sp, vb, unresolved,
|
|
|
|
_)), rt);
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
2011-04-08 14:53:16 -07:00
|
|
|
}
|
|
|
|
}
|
2011-08-13 00:09:25 -07:00
|
|
|
let unresolved = @mutable none::<int>;
|
2011-07-27 14:19:39 +02:00
|
|
|
let rty =
|
2011-08-03 18:06:57 -07:00
|
|
|
fold_ty(tcx, fm_var(bind subst_vars(tcx, sp, vb, unresolved, _)),
|
2011-06-15 11:19:50 -07:00
|
|
|
typ);
|
2011-07-27 14:19:39 +02:00
|
|
|
let ur = *unresolved;
|
|
|
|
alt ur {
|
2012-01-18 22:37:22 -08:00
|
|
|
none { ret fix_ok(rty); }
|
2011-07-27 14:19:39 +02:00
|
|
|
some(var_id) { ret fix_err(var_id); }
|
2011-06-07 23:03:41 -07:00
|
|
|
}
|
2011-04-08 14:53:16 -07:00
|
|
|
}
|
2011-09-12 11:27:30 +02:00
|
|
|
fn resolve_type_var(tcx: ty_ctxt, sp: option::t<span>, vb: @var_bindings,
|
|
|
|
vid: int) -> fixup_result {
|
2011-08-15 16:21:55 -07:00
|
|
|
if vid as uint >= ufind::set_count(vb.sets) { ret fix_err(vid); }
|
|
|
|
let root_id = ufind::find(vb.sets, vid as uint);
|
2011-08-13 00:09:25 -07:00
|
|
|
alt smallintmap::find::<t>(vb.types, root_id) {
|
2012-01-18 22:37:22 -08:00
|
|
|
none { ret fix_err(vid); }
|
2011-08-03 18:06:57 -07:00
|
|
|
some(rt) { ret fixup_vars(tcx, sp, vb, rt); }
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-12-22 14:52:30 +01:00
|
|
|
fn same_type(cx: ctxt, a: t, b: t) -> bool {
|
2011-12-29 13:12:52 +01:00
|
|
|
alt unify::unify(a, b, unify::precise, cx) {
|
2011-12-22 14:52:30 +01:00
|
|
|
unify::ures_ok(_) { true }
|
|
|
|
_ { false }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn type_err_to_str(err: ty::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-01-18 22:37:22 -08:00
|
|
|
terr_box_mutability { ret "boxed values differ in mutability"; }
|
|
|
|
terr_vec_mutability { ret "vectors differ in mutability"; }
|
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-01-04 11:32:26 +01:00
|
|
|
ret "expected argument mode " + mode_str(e_mode) + " but found " +
|
|
|
|
mode_str(a_mode);
|
2011-07-27 14:19:39 +02:00
|
|
|
}
|
|
|
|
terr_constr_len(e_len, a_len) {
|
2011-09-02 15:34:58 -07:00
|
|
|
ret "Expected a type with " + uint::str(e_len) +
|
|
|
|
" 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) {
|
2011-09-02 15:34:58 -07:00
|
|
|
ret "Expected a type with constraint " + ty_constr_to_str(e_constr) +
|
|
|
|
" but found one with constraint " +
|
|
|
|
ty_constr_to_str(a_constr);
|
2011-07-27 14:19:39 +02: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-01-14 16:05:07 -08:00
|
|
|
|
|
|
|
if !type_contains_params(cx, typ) { ret typ; }
|
|
|
|
// Precondition? idx < vec::len(substs)
|
2011-12-29 11:23:35 +01:00
|
|
|
fn substituter(_cx: ctxt, substs: @[ty::t], idx: uint, _did: def_id)
|
|
|
|
-> t {
|
2012-01-14 16:05:07 -08:00
|
|
|
if idx < vec::len(*substs) {
|
|
|
|
ret substs[idx];
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
fail #fmt("Internal error in substituter (substitute_type_params)\
|
|
|
|
%u %u", vec::len(*substs), idx);
|
|
|
|
}
|
2011-04-08 21:27:54 -07:00
|
|
|
}
|
2011-07-28 13:29:29 -07:00
|
|
|
ret fold_ty(cx, fm_param(bind substituter(cx, @substs, _, _)), 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-13 10:58:31 +01:00
|
|
|
ast::def_mod(_) | ast::def_const(_) |
|
2011-12-13 13:19:56 +01:00
|
|
|
ast::def_arg(_, _) | ast::def_local(_, _) | ast::def_upvar(_, _, _) |
|
|
|
|
ast::def_ty_param(_, _) | ast::def_binding(_) | ast::def_use(_) |
|
|
|
|
ast::def_native_ty(_) | ast::def_self(_) | ast::def_ty(_) { false }
|
|
|
|
ast::def_fn(_, _) | ast::def_variant(_, _) |
|
|
|
|
ast::def_native_fn(_, _) { true }
|
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-05 10:57:19 +01:00
|
|
|
fn impl_iface(cx: ctxt, id: ast::def_id) -> option::t<t> {
|
|
|
|
if id.crate == ast::local_crate {
|
|
|
|
option::map(cx.tcache.find(id), {|it| it.ty})
|
|
|
|
} else {
|
|
|
|
csearch::get_impl_iface(cx, id)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-05-19 17:21:21 -07:00
|
|
|
// Tag information
|
2012-01-15 21:42:10 -08:00
|
|
|
type variant_info = @{args: [ty::t], ctor_ty: ty::t, name: str,
|
|
|
|
id: ast::def_id, disr_val: int};
|
2011-07-27 14:19:39 +02:00
|
|
|
|
2011-12-22 16:23:49 +01:00
|
|
|
fn tag_variants(cx: ctxt, id: ast::def_id) -> @[variant_info] {
|
|
|
|
alt cx.tag_var_cache.find(id) {
|
|
|
|
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 {
|
|
|
|
@csearch::get_tag_variants(cx, id)
|
|
|
|
} 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) {
|
2011-12-22 08:45:18 +01:00
|
|
|
ast_map::node_item(@{node: ast::item_tag(variants, _), _}) {
|
2012-01-16 02:36:47 -07:00
|
|
|
let disr_val = -1;
|
2011-12-22 08:45:18 +01:00
|
|
|
@vec::map(variants, {|variant|
|
|
|
|
let ctor_ty = node_id_to_monotype(cx, variant.node.id);
|
|
|
|
let arg_tys = if vec::len(variant.node.args) > 0u {
|
|
|
|
vec::map(ty_fn_args(cx, ctor_ty), {|a| a.ty})
|
|
|
|
} else { [] };
|
2012-01-16 02:36:47 -07:00
|
|
|
alt variant.node.disr_expr {
|
|
|
|
some (ex) {
|
|
|
|
// FIXME: issue #1417
|
|
|
|
disr_val = alt syntax::ast_util::eval_const_expr(ex) {
|
|
|
|
ast_util::const_int(val) {val as int}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_ {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
|
|
|
}
|
2011-05-19 17:21:21 -07:00
|
|
|
}
|
2011-12-22 16:23:49 +01:00
|
|
|
};
|
|
|
|
cx.tag_var_cache.insert(id, result);
|
|
|
|
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:
|
2011-09-12 11:27:30 +02:00
|
|
|
fn tag_variant_with_id(cx: ctxt, tag_id: ast::def_id, variant_id: ast::def_id)
|
|
|
|
-> variant_info {
|
2011-07-27 14:19:39 +02:00
|
|
|
let variants = tag_variants(cx, tag_id);
|
|
|
|
let 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;
|
|
|
|
}
|
2011-09-02 15:34:58 -07:00
|
|
|
cx.sess.bug("tag_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 {
|
2011-07-27 14:19:39 +02:00
|
|
|
if did.crate == ast::local_crate {
|
2011-03-30 17:23:25 -07:00
|
|
|
// The item is in this crate. The caller should have added it to the
|
|
|
|
// type cache already; we simply return it.
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2011-05-19 15:47:15 -07:00
|
|
|
ret cx.tcache.get(did);
|
2011-03-30 17:23:25 -07:00
|
|
|
}
|
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 {
|
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
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-07-27 14:19:39 +02:00
|
|
|
fn ret_ty_of_fn(cx: ctxt, id: ast::node_id) -> t {
|
2011-09-16 13:03:11 -07:00
|
|
|
ty_fn_ret(cx, node_id_to_type(cx, id))
|
2011-05-14 19:02:30 -07:00
|
|
|
}
|
2011-03-30 17:23:25 -07:00
|
|
|
|
2011-09-12 11:27:30 +02: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
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn tycat(cx: ctxt, ty: t) -> int {
|
2011-07-27 14:19:39 +02:00
|
|
|
alt struct(cx, ty) {
|
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 }
|
2011-07-27 14:19:39 +02:00
|
|
|
ty_tag(_, _) { 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*/
|
|
|
|
|
|
|
|
ret tbl[tycat(cx, ty)][opcat(op)];
|
2011-06-24 14:19:58 -07:00
|
|
|
}
|
|
|
|
|
2011-09-12 11:27:30 +02:00
|
|
|
fn ast_constr_to_constr<T>(tcx: ty::ctxt, c: @ast::constr_general<T>) ->
|
2011-08-12 07:15:18 -07:00
|
|
|
@ty::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,
|
|
|
|
"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
|
|
|
}
|
|
|
|
}
|
|
|
|
|
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:
|