2011-05-19 16:06:23 -07:00
|
|
|
import std::int;
|
2011-05-17 20:41:41 +02:00
|
|
|
import std::str;
|
|
|
|
import std::uint;
|
|
|
|
import std::vec;
|
2011-05-12 17:24:54 +02:00
|
|
|
import std::box;
|
|
|
|
import std::ufind;
|
|
|
|
import std::map;
|
|
|
|
import std::map::hashmap;
|
|
|
|
import std::option;
|
|
|
|
import std::option::none;
|
|
|
|
import std::option::some;
|
|
|
|
|
|
|
|
import driver::session;
|
|
|
|
import front::ast;
|
|
|
|
import front::ast::mutability;
|
2011-05-20 14:15:39 -07:00
|
|
|
import front::ast::controlflow;
|
2011-05-12 17:24:54 +02:00
|
|
|
import front::creader;
|
|
|
|
import middle::metadata;
|
|
|
|
import util::common;
|
|
|
|
|
|
|
|
import util::common::ty_u8;
|
|
|
|
import util::common::ty_u16;
|
|
|
|
import util::common::ty_u32;
|
|
|
|
import util::common::ty_u64;
|
|
|
|
|
|
|
|
import util::common::ty_i8;
|
|
|
|
import util::common::ty_i16;
|
|
|
|
import util::common::ty_i32;
|
|
|
|
import util::common::ty_i64;
|
|
|
|
|
|
|
|
import util::common::ty_f32;
|
|
|
|
import util::common::ty_f64;
|
|
|
|
|
|
|
|
import util::common::new_def_hash;
|
|
|
|
import util::common::span;
|
2010-12-21 12:13:51 -08:00
|
|
|
|
2011-05-20 13:57:09 -07:00
|
|
|
import util::data::interner;
|
2011-05-16 13:58:13 -07:00
|
|
|
|
2010-12-21 12:13:51 -08:00
|
|
|
// Data types
|
|
|
|
|
2011-05-09 12:27:03 -07:00
|
|
|
tag mode {
|
|
|
|
mo_val;
|
|
|
|
mo_alias;
|
|
|
|
mo_either;
|
|
|
|
}
|
|
|
|
|
|
|
|
type arg = rec(mode mode, t ty);
|
2011-05-12 17:24:54 +02:00
|
|
|
type field = rec(ast::ident ident, mt mt);
|
|
|
|
type method = rec(ast::proto proto,
|
|
|
|
ast::ident ident,
|
2011-02-18 17:30:57 -08:00
|
|
|
vec[arg] inputs,
|
2011-05-20 14:15:39 -07:00
|
|
|
t output,
|
|
|
|
controlflow cf);
|
2010-12-21 12:13:51 -08:00
|
|
|
|
2011-05-19 17:21:21 -07:00
|
|
|
tag any_item {
|
|
|
|
any_item_rust(@ast::item);
|
|
|
|
any_item_native(@ast::native_item, ast::native_abi);
|
|
|
|
}
|
|
|
|
|
|
|
|
type item_table = hashmap[ast::def_id,any_item];
|
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
type mt = rec(t ty, ast::mutability mut);
|
2011-03-17 17:39:47 -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-05-12 17:24:54 +02:00
|
|
|
type creader_cache = hashmap[tup(int,uint,uint),ty::t];
|
2011-04-29 15:26:28 +00:00
|
|
|
type ctxt = rec(@type_store ts,
|
2011-05-12 17:24:54 +02:00
|
|
|
session::session sess,
|
|
|
|
resolve::def_map def_map,
|
2011-05-19 15:47:15 -07:00
|
|
|
node_type_table node_types,
|
2011-05-19 17:21:21 -07:00
|
|
|
item_table items,
|
2011-05-19 15:47:15 -07:00
|
|
|
type_cache tcache,
|
2011-05-10 14:47:28 -07:00
|
|
|
creader_cache rcache,
|
|
|
|
hashmap[t,str] short_names_cache);
|
2011-05-14 21:13:47 -04:00
|
|
|
type ty_ctxt = ctxt; // Needed for disambiguation from unify::ctxt.
|
2011-04-25 12:15:55 -07:00
|
|
|
|
2011-04-05 16:17:47 -07:00
|
|
|
// Convert from method type to function type. Pretty easy; we just drop
|
|
|
|
// 'ident'.
|
2011-05-19 15:47:15 -07:00
|
|
|
fn method_ty_to_fn_ty(&ctxt cx, method m) -> t {
|
2011-05-20 14:15:39 -07:00
|
|
|
ret mk_fn(cx, m.proto, m.inputs, m.output, m.cf);
|
2011-04-05 16:17:47 -07:00
|
|
|
}
|
|
|
|
|
2011-05-11 17:05:39 -07:00
|
|
|
// Never construct these manually. These are interned.
|
2011-04-20 10:51:41 -07:00
|
|
|
//
|
|
|
|
// TODO: It'd be really nice to be able to hide this definition from the
|
2011-04-22 12:08:23 -07:00
|
|
|
// outside world, to enforce the above invariants.
|
2011-04-25 16:17:14 -07:00
|
|
|
type raw_t = rec(sty struct,
|
2011-05-12 17:24:54 +02:00
|
|
|
option::t[str] cname,
|
2011-04-25 16:17:14 -07:00
|
|
|
uint hash,
|
|
|
|
bool has_params,
|
|
|
|
bool has_bound_params,
|
|
|
|
bool has_vars,
|
2011-05-19 18:28:09 -07:00
|
|
|
bool has_locals);
|
2011-05-11 17:05:39 -07:00
|
|
|
|
|
|
|
type t = uint;
|
2011-04-20 10:51:41 -07:00
|
|
|
|
2010-12-21 12:13:51 -08:00
|
|
|
// NB: If you change this, you'll probably want to change the corresponding
|
2011-05-12 17:24:54 +02:00
|
|
|
// AST structure in front/ast::rs as well.
|
2010-12-21 12:13:51 -08:00
|
|
|
tag sty {
|
|
|
|
ty_nil;
|
2011-05-14 19:02:30 -07:00
|
|
|
ty_bot;
|
2010-12-21 12:13:51 -08:00
|
|
|
ty_bool;
|
|
|
|
ty_int;
|
2011-03-21 17:12:05 -07:00
|
|
|
ty_float;
|
2010-12-21 12:13:51 -08:00
|
|
|
ty_uint;
|
2011-05-12 17:24:54 +02:00
|
|
|
ty_machine(util::common::ty_mach);
|
2010-12-21 12:13:51 -08:00
|
|
|
ty_char;
|
|
|
|
ty_str;
|
2011-05-12 17:24:54 +02:00
|
|
|
ty_tag(ast::def_id, vec[t]);
|
2011-03-17 17:39:47 -07:00
|
|
|
ty_box(mt);
|
|
|
|
ty_vec(mt);
|
2011-04-22 12:27:28 -07:00
|
|
|
ty_port(t);
|
|
|
|
ty_chan(t);
|
2011-03-26 00:53:57 -04:00
|
|
|
ty_task;
|
2011-03-17 17:39:47 -07:00
|
|
|
ty_tup(vec[mt]);
|
2010-12-21 12:13:51 -08:00
|
|
|
ty_rec(vec[field]);
|
2011-05-20 14:15:39 -07:00
|
|
|
ty_fn(ast::proto, vec[arg], t, controlflow);
|
2011-05-12 17:24:54 +02:00
|
|
|
ty_native_fn(ast::native_abi, vec[arg], t);
|
2010-12-21 12:13:51 -08:00
|
|
|
ty_obj(vec[method]);
|
|
|
|
ty_var(int); // ephemeral type var
|
2011-05-12 17:24:54 +02:00
|
|
|
ty_local(ast::def_id); // type of a local var
|
2011-04-12 15:09:50 -07:00
|
|
|
ty_param(uint); // fn/tag type param
|
2011-04-08 21:27:54 -07:00
|
|
|
ty_bound_param(uint); // bound param, only paths
|
2011-02-01 14:56:21 -08:00
|
|
|
ty_type;
|
2011-02-10 18:39:55 -05:00
|
|
|
ty_native;
|
2011-04-22 12:27:28 -07:00
|
|
|
// TODO: ty_fn_arg(t), for a possibly-aliased function argument
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
|
2010-12-21 17:47:13 -08:00
|
|
|
// Data structures used in type unification
|
|
|
|
|
|
|
|
type unify_handler = obj {
|
2011-05-12 17:24:54 +02:00
|
|
|
fn resolve_local(ast::def_id id) -> option::t[t];
|
2011-05-14 21:13:47 -04:00
|
|
|
fn record_local(ast::def_id id, t ty); // TODO: -> unify::result
|
|
|
|
fn record_param(uint index, t binding) -> unify::result;
|
2010-12-21 17:47:13 -08:00
|
|
|
};
|
|
|
|
|
|
|
|
tag type_err {
|
|
|
|
terr_mismatch;
|
2011-05-20 14:15:39 -07:00
|
|
|
terr_controlflow_mismatch;
|
2011-03-17 17:39:47 -07:00
|
|
|
terr_box_mutability;
|
|
|
|
terr_vec_mutability;
|
2010-12-21 17:47:13 -08:00
|
|
|
terr_tuple_size(uint, uint);
|
|
|
|
terr_tuple_mutability;
|
|
|
|
terr_record_size(uint, uint);
|
|
|
|
terr_record_mutability;
|
2011-05-12 17:24:54 +02:00
|
|
|
terr_record_fields(ast::ident,ast::ident);
|
2010-12-31 10:26:29 -08:00
|
|
|
terr_meth_count;
|
2011-05-12 17:24:54 +02:00
|
|
|
terr_obj_meths(ast::ident,ast::ident);
|
2010-12-21 17:47:13 -08:00
|
|
|
terr_arg_count;
|
|
|
|
}
|
|
|
|
|
2011-03-30 17:23:25 -07:00
|
|
|
|
2011-04-22 12:27:28 -07:00
|
|
|
type ty_param_count_and_ty = tup(uint, t);
|
2011-05-12 17:24:54 +02:00
|
|
|
type type_cache = hashmap[ast::def_id,ty_param_count_and_ty];
|
2011-03-30 17:23:25 -07:00
|
|
|
|
2011-05-11 17:05:39 -07:00
|
|
|
const uint idx_nil = 0u;
|
|
|
|
const uint idx_bool = 1u;
|
|
|
|
const uint idx_int = 2u;
|
|
|
|
const uint idx_float = 3u;
|
|
|
|
const uint idx_uint = 4u;
|
|
|
|
const uint idx_i8 = 5u;
|
|
|
|
const uint idx_i16 = 6u;
|
|
|
|
const uint idx_i32 = 7u;
|
|
|
|
const uint idx_i64 = 8u;
|
|
|
|
const uint idx_u8 = 9u;
|
|
|
|
const uint idx_u16 = 10u;
|
|
|
|
const uint idx_u32 = 11u;
|
|
|
|
const uint idx_u64 = 12u;
|
|
|
|
const uint idx_f32 = 13u;
|
|
|
|
const uint idx_f64 = 14u;
|
|
|
|
const uint idx_char = 15u;
|
|
|
|
const uint idx_str = 16u;
|
|
|
|
const uint idx_task = 17u;
|
|
|
|
const uint idx_native = 18u;
|
|
|
|
const uint idx_type = 19u;
|
2011-05-14 19:02:30 -07:00
|
|
|
const uint idx_bot = 20u;
|
|
|
|
const uint idx_first_others = 21u;
|
2011-05-11 17:05:39 -07:00
|
|
|
|
2011-05-16 13:58:13 -07:00
|
|
|
type type_store = interner::interner[raw_t];
|
2011-04-20 18:52:04 -07:00
|
|
|
|
2011-05-13 12:12:54 -07:00
|
|
|
type ty_param_substs_opt_and_ty = tup(option::t[vec[ty::t]], ty::t);
|
2011-05-13 13:05:55 -07:00
|
|
|
type node_type_table =
|
|
|
|
@mutable vec[mutable option::t[ty::ty_param_substs_opt_and_ty]];
|
2011-05-12 16:36:47 -07:00
|
|
|
|
2011-05-19 17:21:21 -07:00
|
|
|
fn populate_type_store(&ctxt cx) {
|
|
|
|
|
|
|
|
intern(cx, ty_nil, none[str]);
|
|
|
|
intern(cx, ty_bool, none[str]);
|
|
|
|
intern(cx, ty_int, none[str]);
|
|
|
|
intern(cx, ty_float, none[str]);
|
|
|
|
intern(cx, ty_uint, none[str]);
|
|
|
|
intern(cx, ty_machine(ty_i8), none[str]);
|
|
|
|
intern(cx, ty_machine(ty_i16), none[str]);
|
|
|
|
intern(cx, ty_machine(ty_i32), none[str]);
|
|
|
|
intern(cx, ty_machine(ty_i64), none[str]);
|
|
|
|
intern(cx, ty_machine(ty_u8), none[str]);
|
|
|
|
intern(cx, ty_machine(ty_u16), none[str]);
|
|
|
|
intern(cx, ty_machine(ty_u32), none[str]);
|
|
|
|
intern(cx, ty_machine(ty_u64), none[str]);
|
|
|
|
intern(cx, ty_machine(ty_f32), none[str]);
|
|
|
|
intern(cx, ty_machine(ty_f64), none[str]);
|
|
|
|
intern(cx, ty_char, none[str]);
|
|
|
|
intern(cx, ty_str, none[str]);
|
|
|
|
intern(cx, ty_task, none[str]);
|
|
|
|
intern(cx, ty_native, none[str]);
|
|
|
|
intern(cx, ty_type, none[str]);
|
|
|
|
intern(cx, ty_bot, none[str]);
|
|
|
|
|
|
|
|
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 {
|
|
|
|
fn hash_cache_entry(&tup(int,uint,uint) k) -> uint {
|
|
|
|
ret (k._0 as uint) + k._1 + k._2;
|
|
|
|
}
|
|
|
|
fn eq_cache_entries(&tup(int,uint,uint) a,
|
|
|
|
&tup(int,uint,uint) b) -> bool {
|
|
|
|
ret a._0 == b._0 &&
|
|
|
|
a._1 == b._1 &&
|
|
|
|
a._2 == b._2;
|
|
|
|
}
|
|
|
|
auto h = hash_cache_entry;
|
|
|
|
auto e = eq_cache_entries;
|
2011-05-12 17:24:54 +02:00
|
|
|
ret map::mk_hashmap[tup(int,uint,uint),t](h, e);
|
2011-04-29 15:26:28 +00:00
|
|
|
}
|
2011-04-25 12:15:55 -07:00
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
fn mk_ctxt(session::session s, resolve::def_map dm) -> ctxt {
|
2011-05-19 15:47:15 -07:00
|
|
|
|
|
|
|
let vec[mutable option::t[ty::ty_param_substs_opt_and_ty]] ntt_sub =
|
|
|
|
[mutable];
|
|
|
|
let node_type_table ntt = @mutable ntt_sub;
|
|
|
|
|
|
|
|
auto tcache =
|
|
|
|
common::new_def_hash[ty::ty_param_count_and_ty]();
|
|
|
|
|
2011-05-19 17:21:21 -07:00
|
|
|
auto items = common::new_def_hash[any_item]();
|
2011-05-20 13:57:09 -07:00
|
|
|
auto ts = @interner::mk[raw_t](hash_raw_ty, eq_raw_ty);
|
2011-05-19 17:21:21 -07:00
|
|
|
|
|
|
|
auto cx =
|
|
|
|
rec(ts = ts,
|
2011-04-29 15:26:28 +00:00
|
|
|
sess = s,
|
2011-05-12 13:25:18 +02:00
|
|
|
def_map = dm,
|
2011-05-19 15:47:15 -07:00
|
|
|
node_types = ntt,
|
2011-05-19 17:21:21 -07:00
|
|
|
items = items,
|
2011-05-19 15:47:15 -07:00
|
|
|
tcache = tcache,
|
2011-05-10 14:47:28 -07:00
|
|
|
rcache = mk_rcache(),
|
|
|
|
short_names_cache =
|
2011-05-19 17:21:21 -07:00
|
|
|
map::mk_hashmap[ty::t,str](ty::hash_ty, ty::eq_ty));
|
|
|
|
|
|
|
|
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-05-19 17:21:21 -07:00
|
|
|
fn mk_raw_ty(&ctxt cx, &sty st, &option::t[str] cname) -> raw_t {
|
2011-04-22 19:26:00 -07:00
|
|
|
auto h = hash_type_info(st, cname);
|
2011-04-25 16:17:14 -07:00
|
|
|
|
|
|
|
let bool has_params = false;
|
|
|
|
let bool has_bound_params = false;
|
|
|
|
let bool has_vars = false;
|
|
|
|
let bool has_locals = false;
|
|
|
|
|
2011-05-19 17:21:21 -07:00
|
|
|
fn derive_flags_t(&ctxt cx,
|
2011-05-11 17:05:39 -07:00
|
|
|
&mutable bool has_params,
|
2011-04-25 16:17:14 -07:00
|
|
|
&mutable bool has_bound_params,
|
|
|
|
&mutable bool has_vars,
|
|
|
|
&mutable bool has_locals,
|
|
|
|
&t tt) {
|
2011-05-19 17:21:21 -07:00
|
|
|
auto rt = interner::get[raw_t](*cx.ts, tt);
|
2011-05-11 17:05:39 -07:00
|
|
|
has_params = has_params || rt.has_params;
|
|
|
|
has_bound_params = has_bound_params || rt.has_bound_params;
|
|
|
|
has_vars = has_vars || rt.has_vars;
|
|
|
|
has_locals = has_locals || rt.has_locals;
|
2011-04-25 16:17:14 -07:00
|
|
|
}
|
|
|
|
|
2011-05-19 17:21:21 -07:00
|
|
|
fn derive_flags_mt(&ctxt cx,
|
2011-05-11 17:05:39 -07:00
|
|
|
&mutable bool has_params,
|
2011-04-25 16:17:14 -07:00
|
|
|
&mutable bool has_bound_params,
|
|
|
|
&mutable bool has_vars,
|
|
|
|
&mutable bool has_locals,
|
|
|
|
&mt m) {
|
2011-05-19 17:21:21 -07:00
|
|
|
derive_flags_t(cx, has_params, has_bound_params,
|
2011-05-19 18:28:09 -07:00
|
|
|
has_vars, has_locals, m.ty);
|
2011-04-25 16:17:14 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-05-19 17:21:21 -07:00
|
|
|
fn derive_flags_arg(&ctxt cx,
|
2011-05-11 17:05:39 -07:00
|
|
|
&mutable bool has_params,
|
2011-04-25 16:17:14 -07:00
|
|
|
&mutable bool has_bound_params,
|
|
|
|
&mutable bool has_vars,
|
|
|
|
&mutable bool has_locals,
|
|
|
|
&arg a) {
|
2011-05-19 17:21:21 -07:00
|
|
|
derive_flags_t(cx, has_params, has_bound_params,
|
2011-05-19 18:28:09 -07:00
|
|
|
has_vars, has_locals, a.ty);
|
2011-04-25 16:17:14 -07:00
|
|
|
}
|
|
|
|
|
2011-05-19 17:21:21 -07:00
|
|
|
fn derive_flags_sig(&ctxt cx,
|
2011-05-11 17:05:39 -07:00
|
|
|
&mutable bool has_params,
|
2011-04-25 16:17:14 -07:00
|
|
|
&mutable bool has_bound_params,
|
|
|
|
&mutable bool has_vars,
|
|
|
|
&mutable bool has_locals,
|
|
|
|
&vec[arg] args,
|
|
|
|
&t tt) {
|
|
|
|
for (arg a in args) {
|
2011-05-19 17:21:21 -07:00
|
|
|
derive_flags_arg(cx, has_params, has_bound_params,
|
2011-05-19 18:28:09 -07:00
|
|
|
has_vars, has_locals, a);
|
2011-04-25 16:17:14 -07:00
|
|
|
}
|
2011-05-19 17:21:21 -07:00
|
|
|
derive_flags_t(cx, has_params, has_bound_params,
|
2011-05-19 18:28:09 -07:00
|
|
|
has_vars, has_locals, tt);
|
2011-04-25 16:17:14 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
alt (st) {
|
2011-05-19 17:21:21 -07:00
|
|
|
case (ty_param(_)) {
|
|
|
|
has_params = true;
|
|
|
|
}
|
|
|
|
case (ty_bound_param(_)) {
|
|
|
|
has_bound_params = true;
|
|
|
|
}
|
2011-04-25 16:17:14 -07:00
|
|
|
case (ty_var(_)) { has_vars = true; }
|
|
|
|
case (ty_local(_)) { has_locals = true; }
|
2011-05-19 18:28:09 -07:00
|
|
|
case (ty_tag(_, ?tys)) {
|
2011-04-25 16:17:14 -07:00
|
|
|
for (t tt in tys) {
|
2011-05-19 17:21:21 -07:00
|
|
|
derive_flags_t(cx, has_params, has_bound_params,
|
2011-05-19 18:28:09 -07:00
|
|
|
has_vars, has_locals, tt);
|
2011-04-25 16:17:14 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
case (ty_box(?m)) {
|
2011-05-19 17:21:21 -07:00
|
|
|
derive_flags_mt(cx, has_params, has_bound_params,
|
2011-05-19 18:28:09 -07:00
|
|
|
has_vars, has_locals, m);
|
2011-04-25 16:17:14 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
case (ty_vec(?m)) {
|
2011-05-19 17:21:21 -07:00
|
|
|
derive_flags_mt(cx, has_params, has_bound_params,
|
2011-05-19 18:28:09 -07:00
|
|
|
has_vars, has_locals, m);
|
2011-04-25 16:17:14 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
case (ty_port(?tt)) {
|
2011-05-19 17:21:21 -07:00
|
|
|
derive_flags_t(cx, has_params, has_bound_params,
|
2011-05-19 18:28:09 -07:00
|
|
|
has_vars, has_locals, tt);
|
2011-04-25 16:17:14 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
case (ty_chan(?tt)) {
|
2011-05-19 17:21:21 -07:00
|
|
|
derive_flags_t(cx, has_params, has_bound_params,
|
2011-05-19 18:28:09 -07:00
|
|
|
has_vars, has_locals, tt);
|
2011-04-25 16:17:14 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
case (ty_tup(?mts)) {
|
|
|
|
for (mt m in mts) {
|
2011-05-19 17:21:21 -07:00
|
|
|
derive_flags_mt(cx, has_params, has_bound_params,
|
2011-05-19 18:28:09 -07:00
|
|
|
has_vars, has_locals, m);
|
2011-04-25 16:17:14 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
case (ty_rec(?flds)) {
|
|
|
|
for (field f in flds) {
|
2011-05-19 17:21:21 -07:00
|
|
|
derive_flags_mt(cx, has_params, has_bound_params,
|
2011-05-19 18:28:09 -07:00
|
|
|
has_vars, has_locals, f.mt);
|
2011-04-25 16:17:14 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-05-20 14:15:39 -07:00
|
|
|
case (ty_fn(_, ?args, ?tt, _)) {
|
2011-05-19 17:21:21 -07:00
|
|
|
derive_flags_sig(cx, has_params, has_bound_params,
|
2011-05-19 18:28:09 -07:00
|
|
|
has_vars, has_locals, args, tt);
|
2011-04-25 16:17:14 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
case (ty_native_fn(_, ?args, ?tt)) {
|
2011-05-19 17:21:21 -07:00
|
|
|
derive_flags_sig(cx, has_params, has_bound_params,
|
2011-05-19 18:28:09 -07:00
|
|
|
has_vars, has_locals, args, tt);
|
2011-04-25 16:17:14 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
case (ty_obj(?meths)) {
|
|
|
|
for (method m in meths) {
|
2011-05-19 17:21:21 -07:00
|
|
|
derive_flags_sig(cx, has_params, has_bound_params,
|
2011-05-19 18:28:09 -07:00
|
|
|
has_vars, has_locals, m.inputs, m.output);
|
2011-04-25 16:17:14 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
case (_) { }
|
|
|
|
}
|
|
|
|
|
2011-05-11 17:05:39 -07:00
|
|
|
ret rec(struct=st, cname=cname, hash=h,
|
|
|
|
has_params = has_params,
|
|
|
|
has_bound_params = has_bound_params,
|
|
|
|
has_vars = has_vars,
|
2011-05-19 18:28:09 -07:00
|
|
|
has_locals = has_locals);
|
2011-05-11 17:05:39 -07:00
|
|
|
}
|
|
|
|
|
2011-05-19 17:21:21 -07:00
|
|
|
fn intern(&ctxt cx, &sty st, &option::t[str] cname) {
|
|
|
|
interner::intern[raw_t](*cx.ts, mk_raw_ty(cx, st, cname));
|
2011-04-25 09:49:08 -07:00
|
|
|
}
|
2011-04-21 19:30:53 -07:00
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
fn gen_ty_full(&ctxt cx, &sty st, &option::t[str] cname) -> t {
|
2011-05-19 17:21:21 -07:00
|
|
|
auto raw_type = mk_raw_ty(cx, st, cname);
|
2011-05-19 18:28:09 -07:00
|
|
|
ret interner::intern[raw_t](*cx.ts, raw_type);
|
2011-04-20 12:22:28 -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-05-09 13:09:20 -07:00
|
|
|
fn gen_ty(&ctxt cx, &sty st) -> t {
|
2011-04-25 16:17:14 -07:00
|
|
|
ret gen_ty_full(cx, st, none[str]);
|
|
|
|
}
|
|
|
|
|
2011-05-11 17:05:39 -07:00
|
|
|
fn mk_nil(&ctxt cx) -> t { ret idx_nil; }
|
2011-05-14 19:02:30 -07:00
|
|
|
fn mk_bot(&ctxt cx) -> t { ret idx_bot; }
|
2011-05-11 17:05:39 -07:00
|
|
|
fn mk_bool(&ctxt cx) -> t { ret idx_bool; }
|
|
|
|
fn mk_int(&ctxt cx) -> t { ret idx_int; }
|
|
|
|
fn mk_float(&ctxt cx) -> t { ret idx_float; }
|
|
|
|
fn mk_uint(&ctxt cx) -> t { ret idx_uint; }
|
2011-04-20 10:51:41 -07:00
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
fn mk_mach(&ctxt cx, &util::common::ty_mach tm) -> t {
|
2011-04-25 09:49:08 -07:00
|
|
|
alt (tm) {
|
2011-05-11 17:05:39 -07:00
|
|
|
case (ty_u8) { ret idx_u8; }
|
|
|
|
case (ty_u16) { ret idx_u16; }
|
|
|
|
case (ty_u32) { ret idx_u32; }
|
|
|
|
case (ty_u64) { ret idx_u64; }
|
|
|
|
|
|
|
|
case (ty_i8) { ret idx_i8; }
|
|
|
|
case (ty_i16) { ret idx_i16; }
|
|
|
|
case (ty_i32) { ret idx_i32; }
|
|
|
|
case (ty_i64) { ret idx_i64; }
|
|
|
|
|
|
|
|
case (ty_f32) { ret idx_f32; }
|
|
|
|
case (ty_f64) { ret idx_f64; }
|
2011-04-25 09:49:08 -07:00
|
|
|
}
|
|
|
|
fail;
|
2011-04-20 10:51:41 -07:00
|
|
|
}
|
|
|
|
|
2011-05-11 17:05:39 -07:00
|
|
|
fn mk_char(&ctxt cx) -> t { ret idx_char; }
|
|
|
|
fn mk_str(&ctxt cx) -> t { ret idx_str; }
|
2011-04-20 11:23:36 -07:00
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
fn mk_tag(&ctxt cx, &ast::def_id did, &vec[t] tys) -> 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-05-09 13:09:20 -07:00
|
|
|
fn mk_box(&ctxt cx, &mt tm) -> t {
|
2011-04-25 12:15:55 -07:00
|
|
|
ret gen_ty(cx, ty_box(tm));
|
2011-04-20 18:52:04 -07:00
|
|
|
}
|
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn mk_imm_box(&ctxt cx, &t ty) -> t {
|
2011-05-12 17:24:54 +02:00
|
|
|
ret mk_box(cx, rec(ty=ty, mut=ast::imm));
|
2011-04-20 18:52:04 -07:00
|
|
|
}
|
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn mk_vec(&ctxt cx, &mt tm) -> t { ret gen_ty(cx, ty_vec(tm)); }
|
2011-05-12 18:37:28 -07:00
|
|
|
|
|
|
|
fn mk_imm_vec(&ctxt cx, &t typ) -> t {
|
|
|
|
ret gen_ty(cx, ty_vec(rec(ty=typ, mut=ast::imm)));
|
|
|
|
}
|
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn mk_port(&ctxt cx, &t ty) -> t { ret gen_ty(cx, ty_port(ty)); }
|
|
|
|
fn mk_chan(&ctxt cx, &t ty) -> t { ret gen_ty(cx, ty_chan(ty)); }
|
|
|
|
fn mk_task(&ctxt cx) -> t { ret gen_ty(cx, ty_task); }
|
2011-04-20 18:52:04 -07:00
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn mk_tup(&ctxt cx, &vec[mt] tms) -> t { ret gen_ty(cx, ty_tup(tms)); }
|
2011-04-20 18:52:04 -07:00
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn mk_imm_tup(&ctxt cx, &vec[t] tys) -> t {
|
2011-04-20 11:23:36 -07:00
|
|
|
// TODO: map
|
2011-05-16 18:21:22 -07:00
|
|
|
let vec[ty::mt] mts = [];
|
2011-04-22 12:27:28 -07:00
|
|
|
for (t typ in tys) {
|
2011-05-16 18:21:22 -07:00
|
|
|
mts += [rec(ty=typ, mut=ast::imm)];
|
2011-04-20 11:23:36 -07:00
|
|
|
}
|
2011-04-25 12:15:55 -07:00
|
|
|
ret mk_tup(cx, mts);
|
2011-04-20 18:52:04 -07:00
|
|
|
}
|
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn mk_rec(&ctxt cx, &vec[field] fs) -> t { ret gen_ty(cx, ty_rec(fs)); }
|
2011-04-20 11:23:36 -07:00
|
|
|
|
2011-05-20 14:15:39 -07:00
|
|
|
fn mk_fn(&ctxt cx, &ast::proto proto, &vec[arg] args, &t ty,
|
|
|
|
&controlflow cf) -> t {
|
|
|
|
ret gen_ty(cx, ty_fn(proto, args, ty, cf));
|
2011-04-20 18:52:04 -07:00
|
|
|
}
|
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
fn mk_native_fn(&ctxt cx, &ast::native_abi abi, &vec[arg] args, &t ty) -> t {
|
2011-04-25 12:15:55 -07:00
|
|
|
ret gen_ty(cx, ty_native_fn(abi, args, ty));
|
2011-04-20 18:52:04 -07:00
|
|
|
}
|
2011-04-20 10:51:41 -07:00
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn mk_obj(&ctxt cx, &vec[method] meths) -> t {
|
2011-04-25 12:15:55 -07:00
|
|
|
ret gen_ty(cx, ty_obj(meths));
|
2011-04-20 10:51:41 -07:00
|
|
|
}
|
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn mk_var(&ctxt cx, int v) -> t {
|
2011-05-11 17:05:39 -07:00
|
|
|
ret gen_ty(cx, ty_var(v));
|
2011-04-25 09:49:08 -07:00
|
|
|
}
|
2011-04-20 18:52:04 -07:00
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
fn mk_local(&ctxt cx, ast::def_id did) -> t {
|
2011-05-11 17:05:39 -07:00
|
|
|
ret gen_ty(cx, ty_local(did));
|
2011-04-20 10:51:41 -07:00
|
|
|
}
|
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn mk_param(&ctxt cx, uint n) -> t {
|
2011-05-11 17:05:39 -07:00
|
|
|
ret gen_ty(cx, ty_param(n));
|
2011-04-20 18:52:04 -07:00
|
|
|
}
|
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn mk_bound_param(&ctxt cx, uint n) -> t {
|
2011-05-11 17:05:39 -07:00
|
|
|
ret gen_ty(cx, ty_bound_param(n));
|
2011-04-20 18:52:04 -07:00
|
|
|
}
|
|
|
|
|
2011-05-11 17:05:39 -07:00
|
|
|
fn mk_type(&ctxt cx) -> t { ret idx_type; }
|
|
|
|
fn mk_native(&ctxt cx) -> t { ret idx_native; }
|
2011-04-20 10:51:41 -07:00
|
|
|
|
|
|
|
|
2011-04-22 12:08:23 -07:00
|
|
|
// Returns the one-level-deep type structure of the given type.
|
2011-05-16 13:58:13 -07:00
|
|
|
fn struct(&ctxt cx, &t typ) -> sty {
|
|
|
|
ret interner::get[raw_t](*cx.ts, typ).struct;
|
|
|
|
}
|
2011-04-22 12:08:23 -07:00
|
|
|
|
2011-04-22 13:45:21 -07:00
|
|
|
// Returns the canonical name of the given type.
|
2011-05-16 13:58:13 -07:00
|
|
|
fn cname(&ctxt cx, &t typ) -> option::t[str] {
|
|
|
|
ret interner::get[raw_t](*cx.ts, typ).cname;
|
|
|
|
}
|
2011-04-22 13:45:21 -07:00
|
|
|
|
2011-04-22 12:08:23 -07:00
|
|
|
|
2010-12-21 12:13:51 -08:00
|
|
|
// Stringification
|
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
fn path_to_str(&ast::path pth) -> str {
|
2011-05-17 20:41:41 +02:00
|
|
|
auto result = str::connect(pth.node.idents, "::");
|
|
|
|
if (vec::len[@ast::ty](pth.node.types) > 0u) {
|
2011-05-12 17:24:54 +02:00
|
|
|
auto f = pretty::pprust::ty_to_str;
|
2010-12-21 12:13:51 -08:00
|
|
|
result += "[";
|
2011-05-17 20:41:41 +02:00
|
|
|
result += str::connect(vec::map(f, pth.node.types), ",");
|
2010-12-21 12:13:51 -08:00
|
|
|
result += "]";
|
|
|
|
}
|
|
|
|
ret result;
|
|
|
|
}
|
|
|
|
|
2011-05-19 15:47:15 -07:00
|
|
|
fn ty_to_str(&ctxt cx, &t typ) -> str {
|
2010-12-21 12:13:51 -08:00
|
|
|
|
2011-05-19 15:47:15 -07:00
|
|
|
fn fn_input_to_str(&ctxt cx, &rec(mode mode, t ty) input) -> str {
|
2010-12-21 12:13:51 -08:00
|
|
|
auto s;
|
2011-05-09 12:27:03 -07:00
|
|
|
alt (input.mode) {
|
|
|
|
case (mo_val) { s = ""; }
|
|
|
|
case (mo_alias) { s = "&"; }
|
|
|
|
case (mo_either) { s = "?"; }
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
|
2011-04-25 12:15:55 -07:00
|
|
|
ret s + ty_to_str(cx, input.ty);
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
|
2011-05-19 15:47:15 -07:00
|
|
|
fn fn_to_str(&ctxt cx,
|
2011-05-12 17:24:54 +02:00
|
|
|
ast::proto proto,
|
|
|
|
option::t[ast::ident] ident,
|
2011-04-22 12:27:28 -07:00
|
|
|
vec[arg] inputs, t output) -> str {
|
2011-04-25 12:15:55 -07:00
|
|
|
auto f = bind fn_input_to_str(cx, _);
|
2011-03-22 13:24:29 -07:00
|
|
|
|
|
|
|
auto s;
|
|
|
|
alt (proto) {
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ast::proto_iter) {
|
2011-03-22 13:24:29 -07:00
|
|
|
s = "iter";
|
|
|
|
}
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ast::proto_fn) {
|
2011-03-22 13:24:29 -07:00
|
|
|
s = "fn";
|
|
|
|
}
|
2011-02-18 17:30:57 -08:00
|
|
|
}
|
2011-03-22 13:24:29 -07:00
|
|
|
|
2010-12-21 12:13:51 -08:00
|
|
|
alt (ident) {
|
2011-05-12 17:24:54 +02:00
|
|
|
case (some[ast::ident](?i)) {
|
2010-12-21 12:13:51 -08:00
|
|
|
s += " ";
|
|
|
|
s += i;
|
|
|
|
}
|
|
|
|
case (_) { }
|
|
|
|
}
|
|
|
|
|
|
|
|
s += "(";
|
2011-05-17 20:41:41 +02:00
|
|
|
s += str::connect(vec::map[arg,str](f, inputs), ", ");
|
2010-12-21 12:13:51 -08:00
|
|
|
s += ")";
|
|
|
|
|
2011-04-25 12:15:55 -07:00
|
|
|
if (struct(cx, output) != ty_nil) {
|
|
|
|
s += " -> " + ty_to_str(cx, output);
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
ret s;
|
|
|
|
}
|
|
|
|
|
2011-05-19 15:47:15 -07:00
|
|
|
fn method_to_str(&ctxt cx, &method m) -> str {
|
2011-05-12 17:24:54 +02:00
|
|
|
ret fn_to_str(cx, m.proto, some[ast::ident](m.ident),
|
2011-02-18 17:30:57 -08:00
|
|
|
m.inputs, m.output) + ";";
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
|
2011-05-19 15:47:15 -07:00
|
|
|
fn field_to_str(&ctxt cx, &field f) -> str {
|
2011-04-25 12:15:55 -07:00
|
|
|
ret mt_to_str(cx, f.mt) + " " + f.ident;
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
|
2011-05-19 15:47:15 -07:00
|
|
|
fn mt_to_str(&ctxt cx, &mt m) -> str {
|
2011-03-17 17:39:47 -07:00
|
|
|
auto mstr;
|
|
|
|
alt (m.mut) {
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ast::mut) { mstr = "mutable "; }
|
|
|
|
case (ast::imm) { mstr = ""; }
|
|
|
|
case (ast::maybe_mut) { mstr = "mutable? "; }
|
2011-03-17 17:39:47 -07:00
|
|
|
}
|
|
|
|
|
2011-04-25 12:15:55 -07:00
|
|
|
ret mstr + ty_to_str(cx, m.ty);
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
|
2011-04-27 13:36:39 -07:00
|
|
|
alt (cname(cx, typ)) {
|
|
|
|
case (some[str](?cs)) {
|
|
|
|
ret cs;
|
|
|
|
}
|
|
|
|
case (_) { }
|
|
|
|
}
|
|
|
|
|
2011-03-17 17:39:47 -07:00
|
|
|
auto s = "";
|
2011-04-25 16:17:14 -07:00
|
|
|
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx, typ)) {
|
2011-04-22 17:00:46 -07:00
|
|
|
case (ty_native) { s += "native"; }
|
|
|
|
case (ty_nil) { s += "()"; }
|
2011-05-14 19:02:30 -07:00
|
|
|
case (ty_bot) { s += "_|_"; }
|
2011-04-22 17:00:46 -07:00
|
|
|
case (ty_bool) { s += "bool"; }
|
|
|
|
case (ty_int) { s += "int"; }
|
|
|
|
case (ty_float) { s += "float"; }
|
|
|
|
case (ty_uint) { s += "uint"; }
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ty_machine(?tm)) { s += common::ty_mach_to_str(tm); }
|
2011-04-22 17:00:46 -07:00
|
|
|
case (ty_char) { s += "char"; }
|
|
|
|
case (ty_str) { s += "str"; }
|
2011-04-25 12:15:55 -07:00
|
|
|
case (ty_box(?tm)) { s += "@" + mt_to_str(cx, tm); }
|
|
|
|
case (ty_vec(?tm)) { s += "vec[" + mt_to_str(cx, tm) + "]"; }
|
|
|
|
case (ty_port(?t)) { s += "port[" + ty_to_str(cx, t) + "]"; }
|
|
|
|
case (ty_chan(?t)) { s += "chan[" + ty_to_str(cx, t) + "]"; }
|
2011-04-22 17:00:46 -07:00
|
|
|
case (ty_type) { s += "type"; }
|
2011-05-14 19:02:30 -07:00
|
|
|
case (ty_task) { s += "task"; }
|
2010-12-21 12:13:51 -08:00
|
|
|
|
|
|
|
case (ty_tup(?elems)) {
|
2011-04-25 12:15:55 -07:00
|
|
|
auto f = bind mt_to_str(cx, _);
|
2011-05-17 20:41:41 +02:00
|
|
|
auto strs = vec::map[mt,str](f, elems);
|
|
|
|
s += "tup(" + str::connect(strs, ",") + ")";
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
case (ty_rec(?elems)) {
|
2011-04-25 12:15:55 -07:00
|
|
|
auto f = bind field_to_str(cx, _);
|
2011-05-17 20:41:41 +02:00
|
|
|
auto strs = vec::map[field,str](f, elems);
|
|
|
|
s += "rec(" + str::connect(strs, ",") + ")";
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
|
2011-02-25 19:42:26 -08:00
|
|
|
case (ty_tag(?id, ?tps)) {
|
2010-12-21 12:13:51 -08:00
|
|
|
// The user should never see this if the cname is set properly!
|
2011-05-12 17:24:54 +02:00
|
|
|
s += "<tag#" + util::common::istr(id._0) + ":" +
|
|
|
|
util::common::istr(id._1) + ">";
|
2011-05-17 20:41:41 +02:00
|
|
|
if (vec::len[t](tps) > 0u) {
|
2011-04-25 12:15:55 -07:00
|
|
|
auto f = bind ty_to_str(cx, _);
|
2011-05-17 20:41:41 +02:00
|
|
|
auto strs = vec::map[t,str](f, tps);
|
|
|
|
s += "[" + str::connect(strs, ",") + "]";
|
2011-02-17 16:23:31 -08:00
|
|
|
}
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
|
2011-05-20 14:15:39 -07:00
|
|
|
case (ty_fn(?proto, ?inputs, ?output, _)) {
|
2011-05-12 17:24:54 +02:00
|
|
|
s += fn_to_str(cx, proto, none[ast::ident], inputs, output);
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
|
2011-02-25 15:58:08 -05:00
|
|
|
case (ty_native_fn(_, ?inputs, ?output)) {
|
2011-05-12 17:24:54 +02:00
|
|
|
s += fn_to_str(cx, ast::proto_fn, none[ast::ident],
|
|
|
|
inputs, output);
|
2011-02-16 16:16:11 -05:00
|
|
|
}
|
|
|
|
|
2010-12-21 12:13:51 -08:00
|
|
|
case (ty_obj(?meths)) {
|
2011-04-27 13:36:39 -07:00
|
|
|
auto f = bind method_to_str(cx, _);
|
2011-05-17 20:41:41 +02:00
|
|
|
auto m = vec::map[method,str](f, meths);
|
|
|
|
s += "obj {\n\t" + str::connect(m, "\n\t") + "\n}";
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
case (ty_var(?v)) {
|
2011-05-12 17:24:54 +02:00
|
|
|
s += "<T" + util::common::istr(v) + ">";
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
|
2011-01-17 17:24:47 -08:00
|
|
|
case (ty_local(?id)) {
|
2011-05-12 17:24:54 +02:00
|
|
|
s += "<L" + util::common::istr(id._0) + ":" +
|
|
|
|
util::common::istr(id._1) + ">";
|
2011-01-17 17:24:47 -08:00
|
|
|
}
|
|
|
|
|
2010-12-21 12:13:51 -08:00
|
|
|
case (ty_param(?id)) {
|
2011-05-17 20:41:41 +02:00
|
|
|
s += "'" + str::unsafe_from_bytes([('a' as u8) + (id as u8)]);
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
2011-04-08 21:27:54 -07:00
|
|
|
|
|
|
|
case (ty_bound_param(?id)) {
|
2011-05-17 20:41:41 +02:00
|
|
|
s += "''" + str::unsafe_from_bytes([('a' as u8) +
|
2011-05-16 18:21:22 -07:00
|
|
|
(id as u8)]);
|
2011-04-08 21:27:54 -07:00
|
|
|
}
|
2011-05-14 19:02:30 -07:00
|
|
|
|
|
|
|
case (_) {
|
|
|
|
s += ty_to_short_str(cx, typ);
|
|
|
|
}
|
|
|
|
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
ret s;
|
|
|
|
}
|
|
|
|
|
2011-05-19 15:47:15 -07:00
|
|
|
fn ty_to_short_str(&ctxt cx, t typ) -> str {
|
2011-04-25 16:59:49 -07:00
|
|
|
auto f = def_to_str;
|
2011-05-12 17:24:54 +02:00
|
|
|
auto ecx = @rec(ds=f, tcx=cx, abbrevs=metadata::ac_no_abbrevs);
|
|
|
|
auto s = metadata::Encode::ty_str(ecx, typ);
|
2011-05-17 20:41:41 +02:00
|
|
|
if (str::byte_len(s) >= 32u) { s = str::substr(s, 0u, 32u); }
|
2011-04-26 13:07:51 -07:00
|
|
|
ret s;
|
2011-04-25 16:59:49 -07:00
|
|
|
}
|
|
|
|
|
2010-12-21 12:13:51 -08:00
|
|
|
// Type folds
|
|
|
|
|
2011-04-22 12:27:28 -07:00
|
|
|
type ty_walk = fn(t);
|
2011-04-15 12:23:00 -07:00
|
|
|
|
2011-05-19 15:47:15 -07:00
|
|
|
fn walk_ty(&ctxt cx, ty_walk walker, t ty) {
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx, ty)) {
|
2011-04-15 12:23:00 -07:00
|
|
|
case (ty_nil) { /* no-op */ }
|
2011-05-14 19:02:30 -07:00
|
|
|
case (ty_bot) { /* no-op */ }
|
2011-04-15 12:23:00 -07:00
|
|
|
case (ty_bool) { /* no-op */ }
|
|
|
|
case (ty_int) { /* no-op */ }
|
|
|
|
case (ty_uint) { /* no-op */ }
|
|
|
|
case (ty_float) { /* no-op */ }
|
|
|
|
case (ty_machine(_)) { /* no-op */ }
|
|
|
|
case (ty_char) { /* no-op */ }
|
|
|
|
case (ty_str) { /* no-op */ }
|
|
|
|
case (ty_type) { /* no-op */ }
|
|
|
|
case (ty_native) { /* no-op */ }
|
2011-04-25 12:15:55 -07:00
|
|
|
case (ty_box(?tm)) { walk_ty(cx, walker, tm.ty); }
|
|
|
|
case (ty_vec(?tm)) { walk_ty(cx, walker, tm.ty); }
|
|
|
|
case (ty_port(?subty)) { walk_ty(cx, walker, subty); }
|
|
|
|
case (ty_chan(?subty)) { walk_ty(cx, walker, subty); }
|
2011-04-15 12:23:00 -07:00
|
|
|
case (ty_tag(?tid, ?subtys)) {
|
2011-04-22 12:27:28 -07:00
|
|
|
for (t subty in subtys) {
|
2011-04-25 12:15:55 -07:00
|
|
|
walk_ty(cx, walker, subty);
|
2011-04-15 12:23:00 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
case (ty_tup(?mts)) {
|
|
|
|
for (mt tm in mts) {
|
2011-04-25 12:15:55 -07:00
|
|
|
walk_ty(cx, walker, tm.ty);
|
2011-04-15 12:23:00 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
case (ty_rec(?fields)) {
|
|
|
|
for (field fl in fields) {
|
2011-04-25 12:15:55 -07:00
|
|
|
walk_ty(cx, walker, fl.mt.ty);
|
2011-04-15 12:23:00 -07:00
|
|
|
}
|
|
|
|
}
|
2011-05-20 14:15:39 -07:00
|
|
|
case (ty_fn(?proto, ?args, ?ret_ty, _)) {
|
2011-04-15 12:23:00 -07:00
|
|
|
for (arg a in args) {
|
2011-04-25 12:15:55 -07:00
|
|
|
walk_ty(cx, walker, a.ty);
|
2011-04-15 12:23:00 -07:00
|
|
|
}
|
2011-04-25 12:15:55 -07:00
|
|
|
walk_ty(cx, walker, ret_ty);
|
2011-04-15 12:23:00 -07:00
|
|
|
}
|
|
|
|
case (ty_native_fn(?abi, ?args, ?ret_ty)) {
|
|
|
|
for (arg a in args) {
|
2011-04-25 12:15:55 -07:00
|
|
|
walk_ty(cx, walker, a.ty);
|
2011-04-15 12:23:00 -07:00
|
|
|
}
|
2011-04-25 12:15:55 -07:00
|
|
|
walk_ty(cx, walker, ret_ty);
|
2011-04-15 12:23:00 -07:00
|
|
|
}
|
|
|
|
case (ty_obj(?methods)) {
|
2011-05-16 18:21:22 -07:00
|
|
|
let vec[method] new_methods = [];
|
2011-04-15 12:23:00 -07:00
|
|
|
for (method m in methods) {
|
|
|
|
for (arg a in m.inputs) {
|
2011-04-25 12:15:55 -07:00
|
|
|
walk_ty(cx, walker, a.ty);
|
2011-04-15 12:23:00 -07:00
|
|
|
}
|
2011-04-25 12:15:55 -07:00
|
|
|
walk_ty(cx, walker, m.output);
|
2011-04-15 12:23:00 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
case (ty_var(_)) { /* no-op */ }
|
|
|
|
case (ty_local(_)) { /* no-op */ }
|
|
|
|
case (ty_param(_)) { /* no-op */ }
|
|
|
|
case (ty_bound_param(_)) { /* no-op */ }
|
|
|
|
}
|
2010-12-21 12:13:51 -08:00
|
|
|
|
2011-04-15 12:23:00 -07:00
|
|
|
walker(ty);
|
|
|
|
}
|
|
|
|
|
2011-04-22 12:27:28 -07:00
|
|
|
type ty_fold = fn(t) -> t;
|
2011-04-15 12:23:00 -07:00
|
|
|
|
2011-05-19 15:47:15 -07:00
|
|
|
fn fold_ty(&ctxt cx, ty_fold fld, t ty_0) -> t {
|
2011-04-15 12:23:00 -07:00
|
|
|
auto ty = ty_0;
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx, ty)) {
|
2011-04-15 12:23:00 -07:00
|
|
|
case (ty_nil) { /* no-op */ }
|
2011-05-14 19:02:30 -07:00
|
|
|
case (ty_bot) { /* no-op */ }
|
2011-04-15 12:23:00 -07:00
|
|
|
case (ty_bool) { /* no-op */ }
|
|
|
|
case (ty_int) { /* no-op */ }
|
|
|
|
case (ty_uint) { /* no-op */ }
|
|
|
|
case (ty_float) { /* no-op */ }
|
|
|
|
case (ty_machine(_)) { /* no-op */ }
|
|
|
|
case (ty_char) { /* no-op */ }
|
|
|
|
case (ty_str) { /* no-op */ }
|
|
|
|
case (ty_type) { /* no-op */ }
|
|
|
|
case (ty_native) { /* no-op */ }
|
2011-03-17 17:39:47 -07:00
|
|
|
case (ty_box(?tm)) {
|
2011-04-25 12:15:55 -07:00
|
|
|
ty = copy_cname(cx, mk_box(cx, rec(ty=fold_ty(cx, fld, tm.ty),
|
|
|
|
mut=tm.mut)), ty);
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
2011-03-17 17:39:47 -07:00
|
|
|
case (ty_vec(?tm)) {
|
2011-04-25 12:15:55 -07:00
|
|
|
ty = copy_cname(cx, mk_vec(cx, rec(ty=fold_ty(cx, fld, tm.ty),
|
|
|
|
mut=tm.mut)), ty);
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
2011-03-10 22:58:55 -05:00
|
|
|
case (ty_port(?subty)) {
|
2011-04-25 12:15:55 -07:00
|
|
|
ty = copy_cname(cx, mk_port(cx, fold_ty(cx, fld, subty)), ty);
|
2011-03-10 22:58:55 -05:00
|
|
|
}
|
|
|
|
case (ty_chan(?subty)) {
|
2011-04-25 12:15:55 -07:00
|
|
|
ty = copy_cname(cx, mk_chan(cx, fold_ty(cx, fld, subty)), ty);
|
2011-03-10 22:58:55 -05:00
|
|
|
}
|
2011-02-17 16:23:31 -08:00
|
|
|
case (ty_tag(?tid, ?subtys)) {
|
2011-05-16 18:21:22 -07:00
|
|
|
let vec[t] new_subtys = [];
|
2011-04-22 12:27:28 -07:00
|
|
|
for (t subty in subtys) {
|
2011-05-16 18:21:22 -07:00
|
|
|
new_subtys += [fold_ty(cx, fld, subty)];
|
2011-02-17 16:23:31 -08:00
|
|
|
}
|
2011-04-25 12:15:55 -07:00
|
|
|
ty = copy_cname(cx, mk_tag(cx, tid, new_subtys), ty);
|
2011-02-17 16:23:31 -08:00
|
|
|
}
|
2011-03-17 17:39:47 -07:00
|
|
|
case (ty_tup(?mts)) {
|
2011-05-16 18:21:22 -07:00
|
|
|
let vec[mt] new_mts = [];
|
2011-03-17 17:39:47 -07:00
|
|
|
for (mt tm in mts) {
|
2011-04-25 12:15:55 -07:00
|
|
|
auto new_subty = fold_ty(cx, fld, tm.ty);
|
2011-05-16 18:21:22 -07:00
|
|
|
new_mts += [rec(ty=new_subty, mut=tm.mut)];
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
2011-04-25 12:15:55 -07:00
|
|
|
ty = copy_cname(cx, mk_tup(cx, new_mts), ty);
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
case (ty_rec(?fields)) {
|
2011-05-16 18:21:22 -07:00
|
|
|
let vec[field] new_fields = [];
|
2010-12-21 12:13:51 -08:00
|
|
|
for (field fl in fields) {
|
2011-04-25 12:15:55 -07:00
|
|
|
auto new_ty = fold_ty(cx, fld, fl.mt.ty);
|
2011-03-17 17:39:47 -07:00
|
|
|
auto new_mt = rec(ty=new_ty, mut=fl.mt.mut);
|
2011-05-16 18:21:22 -07:00
|
|
|
new_fields += [rec(ident=fl.ident, mt=new_mt)];
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
2011-04-25 12:15:55 -07:00
|
|
|
ty = copy_cname(cx, mk_rec(cx, new_fields), ty);
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
2011-05-20 14:15:39 -07:00
|
|
|
case (ty_fn(?proto, ?args, ?ret_ty, ?cf)) {
|
2011-05-16 18:21:22 -07:00
|
|
|
let vec[arg] new_args = [];
|
2010-12-21 12:13:51 -08:00
|
|
|
for (arg a in args) {
|
2011-04-25 12:15:55 -07:00
|
|
|
auto new_ty = fold_ty(cx, fld, a.ty);
|
2011-05-16 18:21:22 -07:00
|
|
|
new_args += [rec(mode=a.mode, ty=new_ty)];
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
2011-04-25 12:15:55 -07:00
|
|
|
ty = copy_cname(cx, mk_fn(cx, proto, new_args,
|
2011-05-20 14:15:39 -07:00
|
|
|
fold_ty(cx, fld, ret_ty), cf), ty);
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
2011-02-25 15:58:08 -05:00
|
|
|
case (ty_native_fn(?abi, ?args, ?ret_ty)) {
|
2011-05-16 18:21:22 -07:00
|
|
|
let vec[arg] new_args = [];
|
2011-02-23 11:59:07 -05:00
|
|
|
for (arg a in args) {
|
2011-04-25 12:15:55 -07:00
|
|
|
auto new_ty = fold_ty(cx, fld, a.ty);
|
2011-05-16 18:21:22 -07:00
|
|
|
new_args += [rec(mode=a.mode, ty=new_ty)];
|
2011-02-23 11:59:07 -05:00
|
|
|
}
|
2011-04-25 12:15:55 -07:00
|
|
|
ty = copy_cname(cx, mk_native_fn(cx, abi, new_args,
|
|
|
|
fold_ty(cx, fld, ret_ty)), ty);
|
2011-02-23 11:59:07 -05:00
|
|
|
}
|
2010-12-21 12:13:51 -08:00
|
|
|
case (ty_obj(?methods)) {
|
2011-05-16 18:21:22 -07:00
|
|
|
let vec[method] new_methods = [];
|
2010-12-21 12:13:51 -08:00
|
|
|
for (method m in methods) {
|
2011-05-16 18:21:22 -07:00
|
|
|
let vec[arg] new_args = [];
|
2010-12-21 12:13:51 -08:00
|
|
|
for (arg a in m.inputs) {
|
2011-05-16 18:21:22 -07:00
|
|
|
new_args += [rec(mode=a.mode,
|
|
|
|
ty=fold_ty(cx, fld, a.ty))];
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
2011-05-16 18:21:22 -07:00
|
|
|
new_methods += [rec(proto=m.proto, ident=m.ident,
|
2011-05-20 14:15:39 -07:00
|
|
|
inputs=new_args,
|
|
|
|
output=fold_ty(cx, fld, m.output), cf=m.cf)];
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
2011-04-25 12:15:55 -07:00
|
|
|
ty = copy_cname(cx, mk_obj(cx, new_methods), ty);
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
2011-04-15 12:23:00 -07:00
|
|
|
case (ty_var(_)) { /* no-op */ }
|
|
|
|
case (ty_local(_)) { /* no-op */ }
|
|
|
|
case (ty_param(_)) { /* no-op */ }
|
|
|
|
case (ty_bound_param(_)) { /* no-op */ }
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
|
2011-04-15 12:23:00 -07:00
|
|
|
ret fld(ty);
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Type utilities
|
|
|
|
|
2011-05-19 15:47:15 -07:00
|
|
|
fn rename(&ctxt cx, t typ, str new_cname) -> t {
|
2011-04-25 12:15:55 -07:00
|
|
|
ret gen_ty_full(cx, struct(cx, typ), some[str](new_cname));
|
2011-04-20 11:59:10 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// Returns a type with the structural part taken from `struct_ty` and the
|
|
|
|
// canonical name from `cname_ty`.
|
2011-05-19 15:47:15 -07:00
|
|
|
fn copy_cname(&ctxt cx, t struct_ty, t cname_ty) -> t {
|
2011-05-11 17:05:39 -07:00
|
|
|
ret gen_ty_full(cx, struct(cx, struct_ty), cname(cx, cname_ty));
|
2011-04-20 11:59:10 -07:00
|
|
|
}
|
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn type_is_nil(&ctxt cx, &t ty) -> bool {
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx, ty)) {
|
2010-12-21 12:13:51 -08:00
|
|
|
case (ty_nil) { ret true; }
|
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
2011-05-14 19:02:30 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
fn type_is_bot(&ctxt cx, &t ty) -> bool {
|
|
|
|
alt (struct(cx, ty)) {
|
|
|
|
case (ty_bot) { ret true; }
|
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn type_is_bool(&ctxt cx, &t ty) -> bool {
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx, ty)) {
|
2011-04-19 15:22:57 -07:00
|
|
|
case (ty_bool) { ret true; }
|
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-03-02 16:42:09 -08:00
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn type_is_structural(&ctxt cx, &t ty) -> bool {
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx, ty)) {
|
2011-02-17 16:23:31 -08:00
|
|
|
case (ty_tup(_)) { ret true; }
|
|
|
|
case (ty_rec(_)) { ret true; }
|
|
|
|
case (ty_tag(_,_)) { ret true; }
|
2011-05-20 14:15:39 -07:00
|
|
|
case (ty_fn(_,_,_,_)) { ret true; }
|
2011-02-17 16:23:31 -08:00
|
|
|
case (ty_obj(_)) { ret true; }
|
|
|
|
case (_) { ret false; }
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
fail;
|
|
|
|
}
|
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn type_is_sequence(&ctxt cx, &t ty) -> bool {
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx, ty)) {
|
2011-03-02 16:42:09 -08:00
|
|
|
case (ty_str) { ret true; }
|
|
|
|
case (ty_vec(_)) { ret true; }
|
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
fail;
|
|
|
|
}
|
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn sequence_element_type(&ctxt cx, &t ty) -> t {
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx, ty)) {
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ty_str) { ret mk_mach(cx, common::ty_u8); }
|
2011-03-17 17:39:47 -07:00
|
|
|
case (ty_vec(?mt)) { ret mt.ty; }
|
2011-03-03 18:18:51 -08:00
|
|
|
}
|
|
|
|
fail;
|
|
|
|
}
|
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn type_is_tup_like(&ctxt cx, &t ty) -> bool {
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx, ty)) {
|
2011-02-17 16:23:31 -08:00
|
|
|
case (ty_box(_)) { ret true; }
|
|
|
|
case (ty_tup(_)) { ret true; }
|
|
|
|
case (ty_rec(_)) { ret true; }
|
|
|
|
case (ty_tag(_,_)) { ret true; }
|
|
|
|
case (_) { ret false; }
|
2011-01-19 16:29:14 -08:00
|
|
|
}
|
|
|
|
fail;
|
|
|
|
}
|
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn get_element_type(&ctxt cx, &t ty, uint i) -> t {
|
2011-05-02 17:47:24 -07:00
|
|
|
assert (type_is_tup_like(cx, ty));
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx, ty)) {
|
2011-03-17 17:39:47 -07:00
|
|
|
case (ty_tup(?mts)) {
|
|
|
|
ret mts.(i).ty;
|
2011-01-19 16:29:14 -08:00
|
|
|
}
|
|
|
|
case (ty_rec(?flds)) {
|
2011-03-17 17:39:47 -07:00
|
|
|
ret flds.(i).mt.ty;
|
2011-01-19 16:29:14 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
fail;
|
|
|
|
}
|
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn type_is_box(&ctxt cx, &t ty) -> bool {
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx, ty)) {
|
2011-03-09 20:14:19 -08:00
|
|
|
case (ty_box(_)) { ret true; }
|
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
fail;
|
|
|
|
}
|
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn type_is_boxed(&ctxt cx, &t ty) -> bool {
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx, ty)) {
|
2010-12-21 12:13:51 -08:00
|
|
|
case (ty_str) { ret true; }
|
|
|
|
case (ty_vec(_)) { ret true; }
|
|
|
|
case (ty_box(_)) { ret true; }
|
2011-03-16 21:49:15 -04:00
|
|
|
case (ty_port(_)) { ret true; }
|
|
|
|
case (ty_chan(_)) { ret true; }
|
2010-12-21 12:13:51 -08:00
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
fail;
|
|
|
|
}
|
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn type_is_scalar(&ctxt cx, &t ty) -> bool {
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx, ty)) {
|
2010-12-21 12:13:51 -08:00
|
|
|
case (ty_nil) { ret true; }
|
|
|
|
case (ty_bool) { ret true; }
|
|
|
|
case (ty_int) { ret true; }
|
2011-03-21 17:12:05 -07:00
|
|
|
case (ty_float) { ret true; }
|
2010-12-21 12:13:51 -08:00
|
|
|
case (ty_uint) { ret true; }
|
|
|
|
case (ty_machine(_)) { ret true; }
|
|
|
|
case (ty_char) { ret true; }
|
2011-02-01 14:56:21 -08:00
|
|
|
case (ty_type) { ret true; }
|
2011-03-17 11:46:21 -07:00
|
|
|
case (ty_native) { ret true; }
|
2010-12-21 12:13:51 -08:00
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
fail;
|
|
|
|
}
|
|
|
|
|
2011-05-19 17:21:21 -07:00
|
|
|
|
2011-05-19 18:28:09 -07:00
|
|
|
fn type_has_pointers(&ctxt cx, &t ty) -> bool {
|
|
|
|
alt (struct(cx, ty)) {
|
|
|
|
// scalar types
|
|
|
|
case (ty_nil) { ret false; }
|
|
|
|
case (ty_bool) { ret false; }
|
|
|
|
case (ty_int) { ret false; }
|
|
|
|
case (ty_float) { ret false; }
|
|
|
|
case (ty_uint) { ret false; }
|
|
|
|
case (ty_machine(_)) { ret false; }
|
|
|
|
case (ty_char) { ret false; }
|
|
|
|
case (ty_type) { ret false; }
|
|
|
|
case (ty_native) { ret false; }
|
|
|
|
|
|
|
|
case (ty_tup(?elts)) {
|
|
|
|
for (mt m in elts) {
|
|
|
|
if (type_has_pointers(cx, m.ty)) { ret true; }
|
|
|
|
}
|
|
|
|
ret false;
|
|
|
|
}
|
|
|
|
case (ty_rec(?flds)) {
|
|
|
|
for (field f in flds) {
|
|
|
|
if (type_has_pointers(cx, f.mt.ty)) { ret true; }
|
|
|
|
}
|
|
|
|
ret false;
|
|
|
|
}
|
|
|
|
|
|
|
|
case (ty_tag(?did,?tps)) {
|
|
|
|
auto variants = tag_variants(cx, did);
|
|
|
|
for (variant_info variant in variants) {
|
|
|
|
auto tup_ty = mk_imm_tup(cx, variant.args);
|
|
|
|
// Perform any type parameter substitutions.
|
|
|
|
tup_ty = bind_params_in_type(cx, tup_ty);
|
|
|
|
tup_ty = substitute_type_params(cx, tps, tup_ty);
|
|
|
|
if (type_has_pointers(cx, tup_ty)) { ret true; }
|
|
|
|
}
|
|
|
|
ret false;
|
|
|
|
}
|
|
|
|
case (_) { ret true; }
|
|
|
|
}
|
|
|
|
fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-03-03 16:43:47 -05:00
|
|
|
// FIXME: should we just return true for native types in
|
|
|
|
// type_is_scalar?
|
2011-05-09 13:09:20 -07:00
|
|
|
fn type_is_native(&ctxt cx, &t ty) -> bool {
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx, ty)) {
|
2011-03-03 16:43:47 -05:00
|
|
|
case (ty_native) { ret true; }
|
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
fail;
|
|
|
|
}
|
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn type_has_dynamic_size(&ctxt cx, &t ty) -> bool {
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx, ty)) {
|
2011-03-17 17:39:47 -07:00
|
|
|
case (ty_tup(?mts)) {
|
2010-12-22 18:02:16 -08:00
|
|
|
auto i = 0u;
|
2011-05-17 20:41:41 +02:00
|
|
|
while (i < vec::len[mt](mts)) {
|
2011-04-25 12:15:55 -07:00
|
|
|
if (type_has_dynamic_size(cx, mts.(i).ty)) { ret true; }
|
2010-12-22 18:02:16 -08:00
|
|
|
i += 1u;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case (ty_rec(?fields)) {
|
|
|
|
auto i = 0u;
|
2011-05-17 20:41:41 +02:00
|
|
|
while (i < vec::len[field](fields)) {
|
2011-04-25 12:15:55 -07:00
|
|
|
if (type_has_dynamic_size(cx, fields.(i).mt.ty)) {
|
2011-04-22 17:00:46 -07:00
|
|
|
ret true;
|
|
|
|
}
|
2010-12-22 18:02:16 -08:00
|
|
|
i += 1u;
|
|
|
|
}
|
|
|
|
}
|
2011-02-28 19:07:26 -08:00
|
|
|
case (ty_tag(_, ?subtys)) {
|
|
|
|
auto i = 0u;
|
2011-05-17 20:41:41 +02:00
|
|
|
while (i < vec::len[t](subtys)) {
|
2011-04-25 12:15:55 -07:00
|
|
|
if (type_has_dynamic_size(cx, subtys.(i))) { ret true; }
|
2011-02-28 19:07:26 -08:00
|
|
|
i += 1u;
|
|
|
|
}
|
|
|
|
}
|
2010-12-22 18:02:16 -08:00
|
|
|
case (ty_param(_)) { ret true; }
|
|
|
|
case (_) { /* fall through */ }
|
|
|
|
}
|
|
|
|
ret false;
|
|
|
|
}
|
2010-12-21 12:13:51 -08:00
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn type_is_integral(&ctxt cx, &t ty) -> bool {
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx, ty)) {
|
2010-12-21 12:13:51 -08:00
|
|
|
case (ty_int) { ret true; }
|
|
|
|
case (ty_uint) { ret true; }
|
|
|
|
case (ty_machine(?m)) {
|
|
|
|
alt (m) {
|
2011-05-12 17:24:54 +02:00
|
|
|
case (common::ty_i8) { ret true; }
|
|
|
|
case (common::ty_i16) { ret true; }
|
|
|
|
case (common::ty_i32) { ret true; }
|
|
|
|
case (common::ty_i64) { ret true; }
|
|
|
|
|
|
|
|
case (common::ty_u8) { ret true; }
|
|
|
|
case (common::ty_u16) { ret true; }
|
|
|
|
case (common::ty_u32) { ret true; }
|
|
|
|
case (common::ty_u64) { ret true; }
|
2010-12-21 12:13:51 -08:00
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case (ty_char) { ret true; }
|
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
fail;
|
|
|
|
}
|
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn type_is_fp(&ctxt cx, &t ty) -> bool {
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx, ty)) {
|
2010-12-21 12:13:51 -08:00
|
|
|
case (ty_machine(?tm)) {
|
|
|
|
alt (tm) {
|
2011-05-12 17:24:54 +02:00
|
|
|
case (common::ty_f32) { ret true; }
|
|
|
|
case (common::ty_f64) { ret true; }
|
2010-12-21 12:13:51 -08:00
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
}
|
2011-03-21 16:21:54 -07:00
|
|
|
case (ty_float) {
|
|
|
|
ret true;
|
|
|
|
}
|
2010-12-21 12:13:51 -08:00
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
fail;
|
|
|
|
}
|
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn type_is_signed(&ctxt cx, &t ty) -> bool {
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx, ty)) {
|
2010-12-21 12:13:51 -08:00
|
|
|
case (ty_int) { ret true; }
|
|
|
|
case (ty_machine(?tm)) {
|
|
|
|
alt (tm) {
|
2011-05-12 17:24:54 +02:00
|
|
|
case (common::ty_i8) { ret true; }
|
|
|
|
case (common::ty_i16) { ret true; }
|
|
|
|
case (common::ty_i32) { ret true; }
|
|
|
|
case (common::ty_i64) { ret true; }
|
2010-12-21 12:13:51 -08:00
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
fail;
|
|
|
|
}
|
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
fn type_param(&ctxt cx, &t ty) -> option::t[uint] {
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx, ty)) {
|
2011-04-12 15:09:50 -07:00
|
|
|
case (ty_param(?id)) { ret some[uint](id); }
|
|
|
|
case (_) { /* fall through */ }
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
2011-04-12 15:09:50 -07:00
|
|
|
ret none[uint];
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
fn def_to_str(&ast::def_id did) -> str {
|
2011-04-19 16:40:18 -07:00
|
|
|
ret #fmt("%d:%d", did._0, did._1);
|
|
|
|
}
|
|
|
|
|
2011-04-22 19:52:30 -07:00
|
|
|
|
2011-04-20 12:22:28 -07:00
|
|
|
// Type hashing. This function is private to this module (and slow); external
|
|
|
|
// users should use `hash_ty()` instead.
|
|
|
|
fn hash_type_structure(&sty st) -> uint {
|
2011-04-20 14:34:17 -07:00
|
|
|
fn hash_uint(uint id, uint n) -> uint {
|
|
|
|
auto h = id;
|
|
|
|
h += h << 5u + n;
|
|
|
|
ret h;
|
|
|
|
}
|
2011-04-20 12:22:28 -07:00
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
fn hash_def(uint id, ast::def_id did) -> uint {
|
2011-04-20 14:34:17 -07:00
|
|
|
auto h = id;
|
|
|
|
h += h << 5u + (did._0 as uint);
|
|
|
|
h += h << 5u + (did._1 as uint);
|
|
|
|
ret h;
|
2011-04-19 18:46:22 -07:00
|
|
|
}
|
2011-04-20 12:22:28 -07:00
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn hash_subty(uint id, &t subty) -> uint {
|
2011-04-20 14:34:17 -07:00
|
|
|
auto h = id;
|
|
|
|
h += h << 5u + hash_ty(subty);
|
|
|
|
ret h;
|
|
|
|
}
|
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn hash_fn(uint id, &vec[arg] args, &t rty) -> uint {
|
2011-04-20 14:34:17 -07:00
|
|
|
auto h = id;
|
|
|
|
for (arg a in args) {
|
|
|
|
h += h << 5u + hash_ty(a.ty);
|
|
|
|
}
|
|
|
|
h += h << 5u + hash_ty(rty);
|
|
|
|
ret h;
|
|
|
|
}
|
|
|
|
|
|
|
|
alt (st) {
|
|
|
|
case (ty_nil) { ret 0u; }
|
|
|
|
case (ty_bool) { ret 1u; }
|
|
|
|
case (ty_int) { ret 2u; }
|
|
|
|
case (ty_float) { ret 3u; }
|
|
|
|
case (ty_uint) { ret 4u; }
|
|
|
|
case (ty_machine(?tm)) {
|
|
|
|
alt (tm) {
|
2011-05-12 17:24:54 +02:00
|
|
|
case (common::ty_i8) { ret 5u; }
|
|
|
|
case (common::ty_i16) { ret 6u; }
|
|
|
|
case (common::ty_i32) { ret 7u; }
|
|
|
|
case (common::ty_i64) { ret 8u; }
|
2011-04-20 14:34:17 -07:00
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
case (common::ty_u8) { ret 9u; }
|
|
|
|
case (common::ty_u16) { ret 10u; }
|
|
|
|
case (common::ty_u32) { ret 11u; }
|
|
|
|
case (common::ty_u64) { ret 12u; }
|
2011-04-20 14:34:17 -07:00
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
case (common::ty_f32) { ret 13u; }
|
|
|
|
case (common::ty_f64) { ret 14u; }
|
2011-04-20 14:34:17 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
case (ty_char) { ret 15u; }
|
|
|
|
case (ty_str) { ret 16u; }
|
|
|
|
case (ty_tag(?did, ?tys)) {
|
|
|
|
auto h = hash_def(17u, did);
|
2011-04-22 12:27:28 -07:00
|
|
|
for (t typ in tys) {
|
2011-04-20 14:34:17 -07:00
|
|
|
h += h << 5u + hash_ty(typ);
|
|
|
|
}
|
|
|
|
ret h;
|
|
|
|
}
|
|
|
|
case (ty_box(?mt)) { ret hash_subty(18u, mt.ty); }
|
|
|
|
case (ty_vec(?mt)) { ret hash_subty(19u, mt.ty); }
|
|
|
|
case (ty_port(?typ)) { ret hash_subty(20u, typ); }
|
|
|
|
case (ty_chan(?typ)) { ret hash_subty(21u, typ); }
|
|
|
|
case (ty_task) { ret 22u; }
|
|
|
|
case (ty_tup(?mts)) {
|
|
|
|
auto h = 23u;
|
|
|
|
for (mt tm in mts) {
|
|
|
|
h += h << 5u + hash_ty(tm.ty);
|
|
|
|
}
|
|
|
|
ret h;
|
|
|
|
}
|
|
|
|
case (ty_rec(?fields)) {
|
|
|
|
auto h = 24u;
|
|
|
|
for (field f in fields) {
|
|
|
|
h += h << 5u + hash_ty(f.mt.ty);
|
|
|
|
}
|
|
|
|
ret h;
|
|
|
|
}
|
2011-05-20 14:15:39 -07:00
|
|
|
case (ty_fn(_, ?args, ?rty, _)) { ret hash_fn(25u, args, rty); }
|
2011-04-20 14:34:17 -07:00
|
|
|
case (ty_native_fn(_, ?args, ?rty)) { ret hash_fn(26u, args, rty); }
|
|
|
|
case (ty_obj(?methods)) {
|
|
|
|
auto h = 27u;
|
|
|
|
for (method m in methods) {
|
2011-05-17 20:41:41 +02:00
|
|
|
h += h << 5u + str::hash(m.ident);
|
2011-04-20 14:34:17 -07:00
|
|
|
}
|
|
|
|
ret h;
|
|
|
|
}
|
|
|
|
case (ty_var(?v)) { ret hash_uint(28u, v as uint); }
|
|
|
|
case (ty_local(?did)) { ret hash_def(29u, did); }
|
|
|
|
case (ty_param(?pid)) { ret hash_uint(30u, pid); }
|
|
|
|
case (ty_bound_param(?pid)) { ret hash_uint(31u, pid); }
|
|
|
|
case (ty_type) { ret 32u; }
|
|
|
|
case (ty_native) { ret 33u; }
|
2011-05-14 19:02:30 -07:00
|
|
|
case (ty_bot) { ret 34u; }
|
2011-04-20 14:34:17 -07:00
|
|
|
}
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
fn hash_type_info(&sty st, &option::t[str] cname_opt) -> uint {
|
2011-04-22 19:26:00 -07:00
|
|
|
auto h = hash_type_structure(st);
|
|
|
|
alt (cname_opt) {
|
|
|
|
case (none[str]) { /* no-op */ }
|
2011-05-17 20:41:41 +02:00
|
|
|
case (some[str](?s)) { h += h << 5u + str::hash(s); }
|
2011-04-22 19:26:00 -07:00
|
|
|
}
|
|
|
|
ret h;
|
|
|
|
}
|
|
|
|
|
2011-05-11 17:05:39 -07:00
|
|
|
fn hash_raw_ty(&raw_t rt) -> uint { ret rt.hash; }
|
|
|
|
|
|
|
|
fn hash_ty(&t typ) -> uint { ret typ; }
|
2011-04-20 12:22:28 -07:00
|
|
|
|
2011-04-21 05:30:48 +00:00
|
|
|
|
2011-04-21 17:06:01 -07:00
|
|
|
// Type equality. This function is private to this module (and slow); external
|
|
|
|
// users should use `eq_ty()` instead.
|
|
|
|
fn equal_type_structures(&sty a, &sty b) -> bool {
|
|
|
|
fn equal_mt(&mt a, &mt b) -> bool {
|
2011-05-11 16:18:02 -07:00
|
|
|
ret a.mut == b.mut && eq_ty(a.ty, b.ty);
|
2011-04-21 17:06:01 -07:00
|
|
|
}
|
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn equal_fn(&vec[arg] args_a, &t rty_a,
|
|
|
|
&vec[arg] args_b, &t rty_b) -> bool {
|
2011-04-25 16:17:14 -07:00
|
|
|
if (!eq_ty(rty_a, rty_b)) { ret false; }
|
2011-04-21 17:06:01 -07:00
|
|
|
|
2011-05-17 20:41:41 +02:00
|
|
|
auto len = vec::len[arg](args_a);
|
|
|
|
if (len != vec::len[arg](args_b)) { ret false; }
|
2011-04-25 16:17:14 -07:00
|
|
|
|
2011-04-21 17:06:01 -07:00
|
|
|
auto i = 0u;
|
|
|
|
while (i < len) {
|
|
|
|
auto arg_a = args_a.(i); auto arg_b = args_b.(i);
|
2011-05-09 12:27:03 -07:00
|
|
|
if (arg_a.mode != arg_b.mode) { ret false; }
|
2011-04-25 16:17:14 -07:00
|
|
|
if (!eq_ty(arg_a.ty, arg_b.ty)) { ret false; }
|
2011-04-21 17:06:01 -07:00
|
|
|
i += 1u;
|
|
|
|
}
|
|
|
|
ret true;
|
|
|
|
}
|
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
fn equal_def(&ast::def_id did_a, &ast::def_id did_b) -> bool {
|
2011-04-21 17:06:01 -07:00
|
|
|
ret did_a._0 == did_b._0 && did_a._1 == did_b._1;
|
|
|
|
}
|
|
|
|
|
|
|
|
alt (a) {
|
|
|
|
case (ty_nil) {
|
|
|
|
alt (b) {
|
|
|
|
case (ty_nil) { ret true; }
|
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
}
|
2011-05-14 19:02:30 -07:00
|
|
|
case (ty_bot) {
|
|
|
|
alt(b) {
|
|
|
|
case (ty_bot) { ret true; }
|
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
}
|
2011-04-21 17:06:01 -07:00
|
|
|
case (ty_bool) {
|
|
|
|
alt (b) {
|
|
|
|
case (ty_bool) { ret true; }
|
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case (ty_int) {
|
|
|
|
alt (b) {
|
|
|
|
case (ty_int) { ret true; }
|
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case (ty_float) {
|
|
|
|
alt (b) {
|
|
|
|
case (ty_float) { ret true; }
|
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case (ty_uint) {
|
|
|
|
alt (b) {
|
|
|
|
case (ty_uint) { ret true; }
|
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case (ty_machine(?tm_a)) {
|
|
|
|
alt (b) {
|
|
|
|
case (ty_machine(?tm_b)) {
|
|
|
|
ret hash_type_structure(a) == hash_type_structure(b);
|
|
|
|
}
|
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case (ty_char) {
|
|
|
|
alt (b) {
|
|
|
|
case (ty_char) { ret true; }
|
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case (ty_str) {
|
|
|
|
alt (b) {
|
|
|
|
case (ty_str) { ret true; }
|
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case (ty_tag(?id_a, ?tys_a)) {
|
|
|
|
alt (b) {
|
|
|
|
case (ty_tag(?id_b, ?tys_b)) {
|
2011-04-21 19:30:53 -07:00
|
|
|
if (!equal_def(id_a, id_b)) { ret false; }
|
2011-04-21 17:06:01 -07:00
|
|
|
|
2011-05-17 20:41:41 +02:00
|
|
|
auto len = vec::len[t](tys_a);
|
|
|
|
if (len != vec::len[t](tys_b)) { ret false; }
|
2011-04-21 17:06:01 -07:00
|
|
|
auto i = 0u;
|
|
|
|
while (i < len) {
|
2011-04-25 16:17:14 -07:00
|
|
|
if (!eq_ty(tys_a.(i), tys_b.(i))) { ret false; }
|
2011-04-21 17:06:01 -07:00
|
|
|
i += 1u;
|
|
|
|
}
|
|
|
|
ret true;
|
|
|
|
}
|
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case (ty_box(?mt_a)) {
|
|
|
|
alt (b) {
|
|
|
|
case (ty_box(?mt_b)) { ret equal_mt(mt_a, mt_b); }
|
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case (ty_vec(?mt_a)) {
|
|
|
|
alt (b) {
|
|
|
|
case (ty_vec(?mt_b)) { ret equal_mt(mt_a, mt_b); }
|
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case (ty_port(?t_a)) {
|
|
|
|
alt (b) {
|
2011-04-25 16:17:14 -07:00
|
|
|
case (ty_port(?t_b)) { ret eq_ty(t_a, t_b); }
|
2011-04-21 17:06:01 -07:00
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case (ty_chan(?t_a)) {
|
|
|
|
alt (b) {
|
2011-04-25 16:17:14 -07:00
|
|
|
case (ty_chan(?t_b)) { ret eq_ty(t_a, t_b); }
|
2011-04-21 17:06:01 -07:00
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case (ty_task) {
|
|
|
|
alt (b) {
|
|
|
|
case (ty_task) { ret true; }
|
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case (ty_tup(?mts_a)) {
|
|
|
|
alt (b) {
|
|
|
|
case (ty_tup(?mts_b)) {
|
2011-05-17 20:41:41 +02:00
|
|
|
auto len = vec::len[mt](mts_a);
|
|
|
|
if (len != vec::len[mt](mts_b)) { ret false; }
|
2011-04-21 17:06:01 -07:00
|
|
|
auto i = 0u;
|
|
|
|
while (i < len) {
|
|
|
|
if (!equal_mt(mts_a.(i), mts_b.(i))) { ret false; }
|
|
|
|
i += 1u;
|
|
|
|
}
|
|
|
|
ret true;
|
|
|
|
}
|
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case (ty_rec(?flds_a)) {
|
|
|
|
alt (b) {
|
|
|
|
case (ty_rec(?flds_b)) {
|
2011-05-17 20:41:41 +02:00
|
|
|
auto len = vec::len[field](flds_a);
|
|
|
|
if (len != vec::len[field](flds_b)) { ret false; }
|
2011-04-21 17:06:01 -07:00
|
|
|
auto i = 0u;
|
|
|
|
while (i < len) {
|
|
|
|
auto fld_a = flds_a.(i); auto fld_b = flds_b.(i);
|
2011-05-17 20:41:41 +02:00
|
|
|
if (!str::eq(fld_a.ident, fld_b.ident) ||
|
2011-04-21 17:06:01 -07:00
|
|
|
!equal_mt(fld_a.mt, fld_b.mt)) {
|
|
|
|
ret false;
|
|
|
|
}
|
|
|
|
i += 1u;
|
|
|
|
}
|
|
|
|
ret true;
|
|
|
|
}
|
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
}
|
2011-05-20 14:15:39 -07:00
|
|
|
case (ty_fn(?p_a, ?args_a, ?rty_a, ?cf_a)) {
|
2011-04-21 17:06:01 -07:00
|
|
|
alt (b) {
|
2011-05-20 14:15:39 -07:00
|
|
|
case (ty_fn(?p_b, ?args_b, ?rty_b, ?cf_b)) {
|
2011-05-11 16:18:02 -07:00
|
|
|
ret p_a == p_b &&
|
2011-05-20 14:15:39 -07:00
|
|
|
cf_a == cf_b &&
|
2011-04-21 17:06:01 -07:00
|
|
|
equal_fn(args_a, rty_a, args_b, rty_b);
|
|
|
|
}
|
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case (ty_native_fn(?abi_a, ?args_a, ?rty_a)) {
|
|
|
|
alt (b) {
|
|
|
|
case (ty_native_fn(?abi_b, ?args_b, ?rty_b)) {
|
2011-05-11 16:18:02 -07:00
|
|
|
ret abi_a == abi_b &&
|
2011-04-21 17:06:01 -07:00
|
|
|
equal_fn(args_a, rty_a, args_b, rty_b);
|
|
|
|
}
|
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case (ty_obj(?methods_a)) {
|
|
|
|
alt (b) {
|
|
|
|
case (ty_obj(?methods_b)) {
|
2011-05-17 20:41:41 +02:00
|
|
|
auto len = vec::len[method](methods_a);
|
|
|
|
if (len != vec::len[method](methods_b)) { ret false; }
|
2011-04-21 17:06:01 -07:00
|
|
|
auto i = 0u;
|
|
|
|
while (i < len) {
|
|
|
|
auto m_a = methods_a.(i); auto m_b = methods_b.(i);
|
2011-05-11 16:18:02 -07:00
|
|
|
if (m_a.proto != m_b.proto ||
|
2011-05-17 20:41:41 +02:00
|
|
|
!str::eq(m_a.ident, m_b.ident) ||
|
2011-04-21 19:30:53 -07:00
|
|
|
!equal_fn(m_a.inputs, m_a.output,
|
|
|
|
m_b.inputs, m_b.output)) {
|
2011-04-21 17:06:01 -07:00
|
|
|
ret false;
|
|
|
|
}
|
2011-04-21 19:30:53 -07:00
|
|
|
i += 1u;
|
2011-04-21 17:06:01 -07:00
|
|
|
}
|
|
|
|
ret true;
|
|
|
|
}
|
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case (ty_var(?v_a)) {
|
|
|
|
alt (b) {
|
|
|
|
case (ty_var(?v_b)) { ret v_a == v_b; }
|
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case (ty_local(?did_a)) {
|
|
|
|
alt (b) {
|
|
|
|
case (ty_local(?did_b)) { ret equal_def(did_a, did_b); }
|
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case (ty_param(?pid_a)) {
|
|
|
|
alt (b) {
|
|
|
|
case (ty_param(?pid_b)) { ret pid_a == pid_b; }
|
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case (ty_bound_param(?pid_a)) {
|
|
|
|
alt (b) {
|
|
|
|
case (ty_bound_param(?pid_b)) { ret pid_a == pid_b; }
|
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case (ty_type) {
|
|
|
|
alt (b) {
|
|
|
|
case (ty_type) { ret true; }
|
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case (ty_native) {
|
|
|
|
alt (b) {
|
|
|
|
case (ty_native) { ret true; }
|
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-04-21 19:30:53 -07:00
|
|
|
// An expensive type equality function. This function is private to this
|
|
|
|
// module.
|
2011-05-11 16:10:17 -07:00
|
|
|
//
|
|
|
|
// FIXME: Use structural comparison, but this loops forever and segfaults.
|
2011-05-11 17:05:39 -07:00
|
|
|
fn eq_raw_ty(&raw_t a, &raw_t b) -> bool {
|
2011-04-21 19:30:53 -07:00
|
|
|
// Check hashes (fast path).
|
2011-04-21 05:30:48 +00:00
|
|
|
if (a.hash != b.hash) {
|
|
|
|
ret false;
|
|
|
|
}
|
|
|
|
|
2011-04-21 19:30:53 -07:00
|
|
|
// Check canonical names.
|
2011-04-22 17:00:46 -07:00
|
|
|
alt (a.cname) {
|
2011-04-21 19:30:53 -07:00
|
|
|
case (none[str]) {
|
2011-04-22 17:00:46 -07:00
|
|
|
alt (b.cname) {
|
2011-04-21 19:30:53 -07:00
|
|
|
case (none[str]) { /* ok */ }
|
2011-04-21 05:30:48 +00:00
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
}
|
2011-04-21 19:30:53 -07:00
|
|
|
case (some[str](?s_a)) {
|
2011-04-22 17:00:46 -07:00
|
|
|
alt (b.cname) {
|
2011-04-21 19:30:53 -07:00
|
|
|
case (some[str](?s_b)) {
|
2011-05-17 20:41:41 +02:00
|
|
|
if (!str::eq(s_a, s_b)) { ret false; }
|
2011-04-21 05:30:48 +00:00
|
|
|
}
|
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
}
|
2011-04-21 19:30:53 -07:00
|
|
|
}
|
2011-04-21 05:30:48 +00:00
|
|
|
|
2011-04-21 19:30:53 -07:00
|
|
|
// Check structures.
|
2011-04-22 17:00:46 -07:00
|
|
|
ret equal_type_structures(a.struct, b.struct);
|
2011-04-21 19:30:53 -07:00
|
|
|
}
|
2011-04-21 05:30:48 +00:00
|
|
|
|
2011-04-21 19:30:53 -07:00
|
|
|
// This is the equality function the public should use. It works as long as
|
|
|
|
// the types are interned.
|
2011-05-11 17:05:39 -07:00
|
|
|
fn eq_ty(&t a, &t b) -> bool { ret a == b; }
|
2011-04-21 05:30:48 +00:00
|
|
|
|
2010-12-21 12:13:51 -08:00
|
|
|
|
2011-05-13 16:40:21 -07:00
|
|
|
// Type lookups
|
2011-04-08 21:27:54 -07:00
|
|
|
|
2011-05-13 13:05:55 -07:00
|
|
|
fn ann_to_ty_param_substs_opt_and_ty(&node_type_table ntt, &ast::ann ann)
|
|
|
|
-> ty_param_substs_opt_and_ty {
|
2011-05-17 11:30:11 -07:00
|
|
|
alt (ntt.(ann.id)) {
|
2011-05-13 18:46:16 -07:00
|
|
|
case (none[ty::ty_param_substs_opt_and_ty]) {
|
|
|
|
log_err "ann_to_ty_param_substs_opt_and_ty() called on an " +
|
|
|
|
"untyped node";
|
2011-05-13 13:05:55 -07:00
|
|
|
fail;
|
|
|
|
}
|
2011-05-13 18:46:16 -07:00
|
|
|
case (some[ty::ty_param_substs_opt_and_ty](?tpot)) { ret tpot; }
|
2011-05-13 13:05:55 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-05-13 16:40:21 -07:00
|
|
|
fn ann_to_type(&node_type_table ntt, &ast::ann ann) -> t {
|
|
|
|
ret ann_to_ty_param_substs_opt_and_ty(ntt, ann)._1;
|
|
|
|
}
|
|
|
|
|
|
|
|
fn ann_to_type_params(&node_type_table ntt, &ast::ann ann) -> vec[t] {
|
|
|
|
alt (ann_to_ty_param_substs_opt_and_ty(ntt, ann)._0) {
|
|
|
|
case (none[vec[t]]) {
|
2011-05-16 18:21:22 -07:00
|
|
|
let vec[t] result = [];
|
2011-05-13 16:40:21 -07:00
|
|
|
ret result;
|
|
|
|
}
|
|
|
|
case (some[vec[t]](?tps)) { ret tps; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-05-13 16:53:35 -07:00
|
|
|
fn ann_has_type_params(&node_type_table ntt, &ast::ann ann) -> bool {
|
|
|
|
auto tpt = ann_to_ty_param_substs_opt_and_ty(ntt, ann);
|
|
|
|
ret !option::is_none[vec[t]](tpt._0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-04-08 21:27:54 -07:00
|
|
|
// Returns the type of an annotation, with type parameter substitutions
|
|
|
|
// performed if applicable.
|
2011-05-19 15:47:15 -07:00
|
|
|
fn ann_to_monotype(&ctxt cx, ast::ann a) -> t {
|
|
|
|
auto tpot = ann_to_ty_param_substs_opt_and_ty(cx.node_types, a);
|
2011-05-13 16:40:21 -07:00
|
|
|
alt (tpot._0) {
|
|
|
|
case (none[vec[t]]) { ret tpot._1; }
|
|
|
|
case (some[vec[t]](?tps)) {
|
|
|
|
ret substitute_type_params(cx, tps, tpot._1);
|
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-05-19 15:47:15 -07:00
|
|
|
fn count_ty_params(&ctxt cx, t ty) -> uint {
|
|
|
|
fn counter(&ctxt cx, @mutable vec[uint] param_indices, t ty) {
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx, ty)) {
|
2011-04-15 12:23:00 -07:00
|
|
|
case (ty_param(?param_idx)) {
|
|
|
|
auto seen = false;
|
|
|
|
for (uint other_param_idx in *param_indices) {
|
|
|
|
if (param_idx == other_param_idx) {
|
|
|
|
seen = true;
|
2011-04-12 15:09:50 -07:00
|
|
|
}
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
2011-04-15 12:23:00 -07:00
|
|
|
if (!seen) {
|
2011-05-16 18:21:22 -07:00
|
|
|
*param_indices += [param_idx];
|
2011-04-15 12:23:00 -07:00
|
|
|
}
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
2011-04-15 12:23:00 -07:00
|
|
|
case (_) { /* fall through */ }
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-05-16 18:21:22 -07:00
|
|
|
let vec[uint] v = []; // FIXME: typechecker botch
|
2011-04-12 15:09:50 -07:00
|
|
|
let @mutable vec[uint] param_indices = @mutable v;
|
2011-04-25 12:15:55 -07:00
|
|
|
auto f = bind counter(cx, param_indices, _);
|
|
|
|
walk_ty(cx, f, ty);
|
2011-05-17 20:41:41 +02:00
|
|
|
ret vec::len[uint](*param_indices);
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn type_contains_vars(&ctxt cx, &t typ) -> bool {
|
2011-05-16 13:58:13 -07:00
|
|
|
ret interner::get[raw_t](*cx.ts, typ).has_vars;
|
2011-04-08 21:27:54 -07:00
|
|
|
}
|
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn type_contains_locals(&ctxt cx, &t typ) -> bool {
|
2011-05-16 13:58:13 -07:00
|
|
|
ret interner::get[raw_t](*cx.ts, typ).has_locals;
|
2011-04-25 18:05:09 -07:00
|
|
|
}
|
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn type_contains_params(&ctxt cx, &t typ) -> bool {
|
2011-05-16 13:58:13 -07:00
|
|
|
ret interner::get[raw_t](*cx.ts, typ).has_params;
|
2011-04-25 05:39:18 +00:00
|
|
|
}
|
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn type_contains_bound_params(&ctxt cx, &t typ) -> bool {
|
2011-05-16 13:58:13 -07:00
|
|
|
ret interner::get[raw_t](*cx.ts, typ).has_bound_params;
|
2011-04-25 05:39:18 +00:00
|
|
|
}
|
|
|
|
|
2011-01-03 18:22:39 -08:00
|
|
|
// Type accessors for substructures of types
|
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn ty_fn_args(&ctxt cx, &t fty) -> vec[arg] {
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx, fty)) {
|
2011-05-20 14:15:39 -07:00
|
|
|
case (ty::ty_fn(_, ?a, _, _)) { ret a; }
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ty::ty_native_fn(_, ?a, _)) { ret a; }
|
2011-02-18 17:30:57 -08:00
|
|
|
}
|
2011-02-23 10:58:43 -08:00
|
|
|
fail;
|
2011-02-18 17:30:57 -08:00
|
|
|
}
|
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
fn ty_fn_proto(&ctxt cx, &t fty) -> ast::proto {
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx, fty)) {
|
2011-05-20 14:15:39 -07:00
|
|
|
case (ty::ty_fn(?p, _, _, _)) { ret p; }
|
2011-02-18 17:30:57 -08:00
|
|
|
}
|
2011-02-23 10:58:43 -08:00
|
|
|
fail;
|
2011-01-03 18:22:39 -08:00
|
|
|
}
|
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
fn ty_fn_abi(&ctxt cx, &t fty) -> ast::native_abi {
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx, fty)) {
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ty::ty_native_fn(?a, _, _)) { ret a; }
|
2011-03-07 15:37:40 -05:00
|
|
|
}
|
|
|
|
fail;
|
|
|
|
}
|
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn ty_fn_ret(&ctxt cx, &t fty) -> t {
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx, fty)) {
|
2011-05-20 14:15:39 -07:00
|
|
|
case (ty::ty_fn(_, _, ?r, _)) { ret r; }
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ty::ty_native_fn(_, _, ?r)) { ret r; }
|
2011-02-18 17:30:57 -08:00
|
|
|
}
|
2011-02-23 10:58:43 -08:00
|
|
|
fail;
|
2011-01-03 18:22:39 -08:00
|
|
|
}
|
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn is_fn_ty(&ctxt cx, &t fty) -> bool {
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx, fty)) {
|
2011-05-20 14:15:39 -07:00
|
|
|
case (ty::ty_fn(_, _, _, _)) { ret true; }
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ty::ty_native_fn(_, _, _)) { ret true; }
|
2011-02-18 17:30:57 -08:00
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
ret false;
|
2011-01-03 18:22:39 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-12-21 12:13:51 -08:00
|
|
|
// Type accessors for AST nodes
|
|
|
|
|
2011-04-12 15:09:50 -07:00
|
|
|
// Given an item, returns the associated type as well as the number of type
|
|
|
|
// parameters it has.
|
2011-05-12 16:36:47 -07:00
|
|
|
fn native_item_ty(&node_type_table ntt, &@ast::native_item it)
|
|
|
|
-> ty_param_count_and_ty {
|
2011-04-12 15:09:50 -07:00
|
|
|
auto ty_param_count;
|
2011-02-16 14:02:02 -05:00
|
|
|
auto result_ty;
|
|
|
|
alt (it.node) {
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ast::native_item_fn(_, _, _, ?tps, _, ?ann)) {
|
2011-05-17 20:41:41 +02:00
|
|
|
ty_param_count = vec::len[ast::ty_param](tps);
|
2011-05-12 16:36:47 -07:00
|
|
|
result_ty = ann_to_type(ntt, ann);
|
2011-02-16 14:02:02 -05:00
|
|
|
}
|
|
|
|
}
|
2011-04-12 15:09:50 -07:00
|
|
|
ret tup(ty_param_count, result_ty);
|
2011-02-16 14:02:02 -05:00
|
|
|
}
|
|
|
|
|
2011-05-12 16:36:47 -07:00
|
|
|
fn item_ty(&node_type_table ntt, &@ast::item it) -> ty_param_count_and_ty {
|
2011-04-12 15:09:50 -07:00
|
|
|
auto ty_param_count;
|
2011-01-04 12:46:11 -08:00
|
|
|
auto result_ty;
|
|
|
|
alt (it.node) {
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ast::item_const(_, _, _, _, ?ann)) {
|
2011-04-12 15:09:50 -07:00
|
|
|
ty_param_count = 0u;
|
2011-05-12 16:36:47 -07:00
|
|
|
result_ty = ann_to_type(ntt, ann);
|
2011-01-04 12:46:11 -08:00
|
|
|
}
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ast::item_fn(_, _, ?tps, _, ?ann)) {
|
2011-05-17 20:41:41 +02:00
|
|
|
ty_param_count = vec::len[ast::ty_param](tps);
|
2011-05-12 16:36:47 -07:00
|
|
|
result_ty = ann_to_type(ntt, ann);
|
2011-01-04 12:46:11 -08:00
|
|
|
}
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ast::item_mod(_, _, _)) {
|
2011-01-04 12:46:11 -08:00
|
|
|
fail; // modules are typeless
|
|
|
|
}
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ast::item_ty(_, _, ?tps, _, ?ann)) {
|
2011-05-17 20:41:41 +02:00
|
|
|
ty_param_count = vec::len[ast::ty_param](tps);
|
2011-05-12 16:36:47 -07:00
|
|
|
result_ty = ann_to_type(ntt, ann);
|
2011-01-04 12:46:11 -08:00
|
|
|
}
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ast::item_tag(_, _, ?tps, ?did, ?ann)) {
|
2011-05-17 20:41:41 +02:00
|
|
|
ty_param_count = vec::len[ast::ty_param](tps);
|
2011-05-12 16:36:47 -07:00
|
|
|
result_ty = ann_to_type(ntt, ann);
|
2011-01-04 12:46:11 -08:00
|
|
|
}
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ast::item_obj(_, _, ?tps, _, ?ann)) {
|
2011-05-17 20:41:41 +02:00
|
|
|
ty_param_count = vec::len[ast::ty_param](tps);
|
2011-05-12 16:36:47 -07:00
|
|
|
result_ty = ann_to_type(ntt, ann);
|
2011-01-04 12:46:11 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-04-12 15:09:50 -07:00
|
|
|
ret tup(ty_param_count, result_ty);
|
2011-01-04 12:46:11 -08:00
|
|
|
}
|
|
|
|
|
2011-05-19 15:47:15 -07:00
|
|
|
fn stmt_ty(&ctxt cx, &@ast::stmt s) -> t {
|
2010-12-21 12:13:51 -08:00
|
|
|
alt (s.node) {
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ast::stmt_expr(?e,_)) {
|
2011-05-19 15:47:15 -07:00
|
|
|
ret expr_ty(cx, e);
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
case (_) {
|
2011-04-25 12:15:55 -07:00
|
|
|
ret mk_nil(cx);
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-05-19 15:47:15 -07:00
|
|
|
fn block_ty(&ctxt cx, &ast::block b) -> t {
|
2010-12-21 12:13:51 -08:00
|
|
|
alt (b.node.expr) {
|
2011-05-19 15:47:15 -07:00
|
|
|
case (some[@ast::expr](?e)) { ret expr_ty(cx, e); }
|
2011-05-12 17:24:54 +02:00
|
|
|
case (none[@ast::expr]) { ret mk_nil(cx); }
|
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-05-19 15:47:15 -07:00
|
|
|
fn pat_ty(&ctxt cx, &@ast::pat pat) -> t {
|
|
|
|
ret ann_to_monotype(cx, pat_ann(pat));
|
2010-12-21 12:13:51 -08:00
|
|
|
}
|
|
|
|
|
2011-05-16 10:22:14 -07:00
|
|
|
fn item_ann(&@ast::item it) -> ast::ann {
|
|
|
|
alt (it.node) {
|
|
|
|
case (ast::item_const(_,_,_,_,?a)) { ret a; }
|
|
|
|
case (ast::item_fn(_,_,_,_,?a)) { ret a; }
|
|
|
|
case (ast::item_mod(_,_,_)) {
|
|
|
|
log_err "a module was passed to item_ann(), " +
|
|
|
|
"but modules haven't annotations";
|
|
|
|
fail;
|
|
|
|
}
|
|
|
|
case (ast::item_native_mod(_,_,_)) {
|
|
|
|
log_err "a native module was passed to item_ann(), " +
|
|
|
|
"but native modules haven't annotations";
|
|
|
|
fail;
|
|
|
|
}
|
|
|
|
case (ast::item_ty(_,_,_,_,?a)) { ret a; }
|
|
|
|
case (ast::item_tag(_,_,_,_,?a)) { ret a; }
|
|
|
|
case (ast::item_obj(_,_,_,_,?a)) { ret a; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
fn expr_ann(&@ast::expr e) -> ast::ann {
|
2011-05-12 16:57:08 -07:00
|
|
|
alt (e.node) {
|
|
|
|
case (ast::expr_vec(_,_,?a)) { ret a; }
|
|
|
|
case (ast::expr_tup(_,?a)) { ret a; }
|
|
|
|
case (ast::expr_rec(_,_,?a)) { ret a; }
|
|
|
|
case (ast::expr_call(_,_,?a)) { ret a; }
|
|
|
|
case (ast::expr_bind(_,_,?a)) { ret a; }
|
|
|
|
case (ast::expr_binary(_,_,_,?a)) { ret a; }
|
|
|
|
case (ast::expr_unary(_,_,?a)) { ret a; }
|
|
|
|
case (ast::expr_lit(_,?a)) { ret a; }
|
|
|
|
case (ast::expr_cast(_,_,?a)) { ret a; }
|
|
|
|
case (ast::expr_if(_,_,_,?a)) { ret a; }
|
|
|
|
case (ast::expr_while(_,_,?a)) { ret a; }
|
|
|
|
case (ast::expr_for(_,_,_,?a)) { ret a; }
|
|
|
|
case (ast::expr_for_each(_,_,_,?a)) { ret a; }
|
|
|
|
case (ast::expr_do_while(_,_,?a)) { ret a; }
|
|
|
|
case (ast::expr_alt(_,_,?a)) { ret a; }
|
|
|
|
case (ast::expr_block(_,?a)) { ret a; }
|
|
|
|
case (ast::expr_assign(_,_,?a)) { ret a; }
|
|
|
|
case (ast::expr_assign_op(_,_,_,?a)) { ret a; }
|
|
|
|
case (ast::expr_send(_,_,?a)) { ret a; }
|
|
|
|
case (ast::expr_recv(_,_,?a)) { ret a; }
|
|
|
|
case (ast::expr_field(_,_,?a)) { ret a; }
|
|
|
|
case (ast::expr_index(_,_,?a)) { ret a; }
|
|
|
|
case (ast::expr_path(_,?a)) { ret a; }
|
|
|
|
case (ast::expr_ext(_,_,_,_,?a)) { ret a; }
|
|
|
|
case (ast::expr_fail(?a)) { ret a; }
|
|
|
|
case (ast::expr_ret(_,?a)) { ret a; }
|
|
|
|
case (ast::expr_put(_,?a)) { ret a; }
|
|
|
|
case (ast::expr_be(_,?a)) { ret a; }
|
|
|
|
case (ast::expr_log(_,_,?a)) { ret a; }
|
|
|
|
case (ast::expr_assert(_,?a)) { ret a; }
|
|
|
|
case (ast::expr_check(_,?a)) { ret a; }
|
|
|
|
case (ast::expr_port(?a)) { ret a; }
|
|
|
|
case (ast::expr_chan(_,?a)) { ret a; }
|
2011-05-10 19:52:22 -07:00
|
|
|
case (ast::expr_anon_obj(_,_,_,?a)) { ret a; }
|
2011-05-12 16:57:08 -07:00
|
|
|
case (ast::expr_break(?a)) { ret a; }
|
|
|
|
case (ast::expr_cont(?a)) { ret a; }
|
|
|
|
case (ast::expr_self_method(_, ?a)) { ret a; }
|
2011-05-17 11:49:50 -07:00
|
|
|
case (ast::expr_spawn(_, _, _, _, ?a)) { ret a; }
|
2010-12-21 12:13:51 -08: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"
|
|
|
|
// instead of "fn(&T) -> T with T = int". If this isn't what you want, see
|
|
|
|
// expr_ty_params_and_ty() below.
|
2011-05-19 15:47:15 -07:00
|
|
|
fn expr_ty(&ctxt cx, &@ast::expr expr) -> t {
|
|
|
|
ret ann_to_monotype(cx, expr_ann(expr));
|
2011-04-08 21:27:54 -07:00
|
|
|
}
|
|
|
|
|
2011-05-19 15:47:15 -07:00
|
|
|
fn expr_ty_params_and_ty(&ctxt cx, &@ast::expr expr)
|
2011-05-12 16:36:47 -07:00
|
|
|
-> tup(vec[t], t) {
|
2011-05-02 17:47:24 -07:00
|
|
|
auto a = expr_ann(expr);
|
|
|
|
|
2011-05-19 15:47:15 -07:00
|
|
|
ret tup(ann_to_type_params(cx.node_types, a),
|
|
|
|
ann_to_type(cx.node_types, a));
|
2011-04-08 21:27:54 -07:00
|
|
|
}
|
|
|
|
|
2011-05-12 16:36:47 -07:00
|
|
|
fn expr_has_ty_params(&node_type_table ntt, &@ast::expr expr) -> bool {
|
2011-05-13 16:53:35 -07:00
|
|
|
ret ann_has_type_params(ntt, expr_ann(expr));
|
2011-04-08 21:27:54 -07:00
|
|
|
}
|
|
|
|
|
2011-05-17 19:00:29 -07:00
|
|
|
fn stmt_ann(&@ast::stmt s) -> ast::ann {
|
|
|
|
alt (s.node) {
|
|
|
|
case (ast::stmt_decl(_, ?a)) { ret a; }
|
|
|
|
case (ast::stmt_expr(_, ?a)) { ret a; }
|
|
|
|
case (ast::stmt_crate_directive(_)) {
|
|
|
|
log_err "ty::stmt_ann(): crate directive found";
|
|
|
|
fail;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn pat_ann(&@ast::pat p) -> ast::ann {
|
|
|
|
alt (p.node) {
|
|
|
|
case (ast::pat_wild(?a)) { ret a; }
|
|
|
|
case (ast::pat_bind(_, _, ?a)) { ret a; }
|
|
|
|
case (ast::pat_lit(_, ?a)) { ret a; }
|
|
|
|
case (ast::pat_tag(_, _, ?a)) { ret a; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-04-11 15:03:14 -07:00
|
|
|
|
2010-12-21 12:13:51 -08:00
|
|
|
// Expression utilities
|
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
fn field_num(&session::session sess, &span sp,
|
|
|
|
&ast::ident id) -> uint {
|
2010-12-21 12:13:51 -08:00
|
|
|
let uint accum = 0u;
|
|
|
|
let uint i = 0u;
|
|
|
|
for (u8 c in id) {
|
|
|
|
if (i == 0u) {
|
|
|
|
if (c != ('_' as u8)) {
|
|
|
|
sess.span_err(sp,
|
|
|
|
"bad numeric field on tuple: "
|
|
|
|
+ "missing leading underscore");
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (('0' as u8) <= c && c <= ('9' as u8)) {
|
|
|
|
accum *= 10u;
|
|
|
|
accum += (c as uint) - ('0' as uint);
|
|
|
|
} else {
|
|
|
|
auto s = "";
|
2011-05-17 20:41:41 +02:00
|
|
|
s += str::unsafe_from_byte(c);
|
2010-12-21 12:13:51 -08:00
|
|
|
sess.span_err(sp,
|
|
|
|
"bad numeric field on tuple: "
|
|
|
|
+ " non-digit character: "
|
|
|
|
+ s);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
i += 1u;
|
|
|
|
}
|
|
|
|
ret accum;
|
|
|
|
}
|
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
fn field_idx(&session::session sess, &span sp,
|
|
|
|
&ast::ident id, &vec[field] fields) -> uint {
|
2010-12-21 12:13:51 -08:00
|
|
|
let uint i = 0u;
|
|
|
|
for (field f in fields) {
|
2011-05-17 20:41:41 +02:00
|
|
|
if (str::eq(f.ident, id)) {
|
2010-12-21 12:13:51 -08:00
|
|
|
ret i;
|
|
|
|
}
|
|
|
|
i += 1u;
|
|
|
|
}
|
|
|
|
sess.span_err(sp, "unknown field '" + id + "' of record");
|
|
|
|
fail;
|
|
|
|
}
|
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
fn method_idx(&session::session sess, &span sp,
|
|
|
|
&ast::ident id, &vec[method] meths) -> uint {
|
2010-12-21 12:13:51 -08:00
|
|
|
let uint i = 0u;
|
|
|
|
for (method m in meths) {
|
2011-05-17 20:41:41 +02:00
|
|
|
if (str::eq(m.ident, id)) {
|
2010-12-21 12:13:51 -08:00
|
|
|
ret i;
|
|
|
|
}
|
|
|
|
i += 1u;
|
|
|
|
}
|
|
|
|
sess.span_err(sp, "unknown method '" + id + "' of obj");
|
|
|
|
fail;
|
|
|
|
}
|
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn sort_methods(&vec[method] meths) -> vec[method] {
|
2011-03-17 11:40:05 -07:00
|
|
|
fn method_lteq(&method a, &method b) -> bool {
|
2011-05-17 20:41:41 +02:00
|
|
|
ret str::lteq(a.ident, b.ident);
|
2011-03-17 11:40:05 -07:00
|
|
|
}
|
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
ret std::sort::merge_sort[method](bind method_lteq(_,_), meths);
|
2011-03-17 11:40:05 -07:00
|
|
|
}
|
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
fn is_lval(&@ast::expr expr) -> bool {
|
2010-12-21 12:13:51 -08:00
|
|
|
alt (expr.node) {
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ast::expr_field(_,_,_)) { ret true; }
|
|
|
|
case (ast::expr_index(_,_,_)) { ret true; }
|
|
|
|
case (ast::expr_path(_,_)) { ret true; }
|
|
|
|
case (ast::expr_unary(ast::deref,_,_)) { ret true; }
|
2010-12-21 12:13:51 -08:00
|
|
|
case (_) { ret false; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
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
|
2010-12-21 17:47:13 -08:00
|
|
|
|
2011-05-14 21:13:47 -04:00
|
|
|
mod unify {
|
2011-04-21 11:46:31 -07:00
|
|
|
tag result {
|
2011-04-22 12:27:28 -07:00
|
|
|
ures_ok(t);
|
|
|
|
ures_err(type_err, t, t);
|
2011-04-21 11:46:31 -07:00
|
|
|
}
|
|
|
|
|
2011-05-19 16:06:23 -07:00
|
|
|
tag set_result {
|
|
|
|
usr_ok(vec[t]);
|
|
|
|
usr_err(type_err, t, t);
|
|
|
|
}
|
|
|
|
|
2011-05-19 11:14:04 -07:00
|
|
|
type bindings[T] = rec(ufind::ufind sets,
|
|
|
|
hashmap[T,uint] ids,
|
2011-05-19 16:06:23 -07:00
|
|
|
mutable vec[mutable option::t[t]] types);
|
2011-05-18 11:53:26 -07:00
|
|
|
|
2011-05-19 11:14:04 -07:00
|
|
|
fn mk_bindings[T](map::hashfn[T] hasher, map::eqfn[T] eqer)
|
|
|
|
-> @bindings[T] {
|
2011-05-19 16:06:23 -07:00
|
|
|
let vec[mutable option::t[t]] types = [mutable];
|
2011-05-18 13:08:02 -07:00
|
|
|
ret @rec(sets=ufind::make(),
|
2011-05-19 11:14:04 -07:00
|
|
|
ids=map::mk_hashmap[T,uint](hasher, eqer),
|
2011-05-18 13:08:02 -07:00
|
|
|
mutable types=types);
|
2011-05-18 11:53:26 -07:00
|
|
|
}
|
|
|
|
|
2011-05-19 16:06:23 -07:00
|
|
|
fn record_binding[T](&@ctxt cx, &@bindings[T] bindings, &T key, t typ)
|
|
|
|
-> result {
|
|
|
|
auto n = get_or_create_set[T](bindings, key);
|
|
|
|
|
|
|
|
auto result_type = typ;
|
|
|
|
if (n < vec::len[option::t[t]](bindings.types)) {
|
|
|
|
alt (bindings.types.(n)) {
|
|
|
|
case (some[t](?old_type)) {
|
|
|
|
alt (unify_step(cx, old_type, typ)) {
|
|
|
|
case (ures_ok(?unified_type)) {
|
|
|
|
result_type = unified_type;
|
|
|
|
}
|
|
|
|
case (?res) { ret res; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case (none[t]) { /* fall through */ }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
vec::grow_set[option::t[t]](bindings.types, n, none[t],
|
|
|
|
some[t](result_type));
|
|
|
|
|
|
|
|
ret ures_ok(typ);
|
|
|
|
}
|
|
|
|
|
2011-05-19 11:14:04 -07:00
|
|
|
type ctxt = rec(@bindings[int] bindings,
|
2011-04-20 18:52:04 -07:00
|
|
|
unify_handler handler,
|
2011-04-25 12:15:55 -07:00
|
|
|
ty_ctxt tcx);
|
2011-04-08 14:53:16 -07:00
|
|
|
|
2010-12-21 17:47:13 -08:00
|
|
|
// Wraps the given type in an appropriate cname.
|
|
|
|
//
|
|
|
|
// TODO: This doesn't do anything yet. We should carry the cname up from
|
|
|
|
// the expected and/or actual types when unification results in a type
|
|
|
|
// identical to one or both of the two. The precise algorithm for this is
|
|
|
|
// something we'll probably need to develop over time.
|
|
|
|
|
|
|
|
// Simple structural type comparison.
|
2011-04-22 17:00:46 -07:00
|
|
|
fn struct_cmp(@ctxt cx, t expected, t actual) -> result {
|
2011-04-25 12:15:55 -07:00
|
|
|
if (struct(cx.tcx, expected) == struct(cx.tcx, actual)) {
|
2010-12-21 17:47:13 -08:00
|
|
|
ret ures_ok(expected);
|
|
|
|
}
|
|
|
|
|
|
|
|
ret ures_err(terr_mismatch, expected, actual);
|
|
|
|
}
|
|
|
|
|
2011-03-18 11:49:06 -07:00
|
|
|
// Unifies two mutability flags.
|
2011-05-12 17:24:54 +02:00
|
|
|
fn unify_mut(ast::mutability expected, ast::mutability actual)
|
|
|
|
-> option::t[ast::mutability] {
|
2011-03-18 11:49:06 -07:00
|
|
|
if (expected == actual) {
|
2011-05-12 17:24:54 +02:00
|
|
|
ret some[ast::mutability](expected);
|
2011-03-18 11:49:06 -07:00
|
|
|
}
|
2011-05-12 17:24:54 +02:00
|
|
|
if (expected == ast::maybe_mut) {
|
|
|
|
ret some[ast::mutability](actual);
|
2011-03-18 11:49:06 -07:00
|
|
|
}
|
2011-05-12 17:24:54 +02:00
|
|
|
if (actual == ast::maybe_mut) {
|
|
|
|
ret some[ast::mutability](expected);
|
2011-03-18 11:49:06 -07:00
|
|
|
}
|
2011-05-12 17:24:54 +02:00
|
|
|
ret none[ast::mutability];
|
2011-03-18 11:49:06 -07:00
|
|
|
}
|
|
|
|
|
2011-02-23 11:59:07 -05:00
|
|
|
tag fn_common_res {
|
2011-04-21 11:46:31 -07:00
|
|
|
fn_common_res_err(result);
|
2011-04-22 12:27:28 -07:00
|
|
|
fn_common_res_ok(vec[arg], t);
|
2011-02-23 11:59:07 -05:00
|
|
|
}
|
2010-12-31 09:48:54 -08:00
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn unify_fn_common(&@ctxt cx,
|
|
|
|
&t expected,
|
|
|
|
&t actual,
|
|
|
|
&vec[arg] expected_inputs, &t expected_output,
|
|
|
|
&vec[arg] actual_inputs, &t actual_output)
|
2011-02-23 11:59:07 -05:00
|
|
|
-> fn_common_res {
|
2011-05-17 20:41:41 +02:00
|
|
|
auto expected_len = vec::len[arg](expected_inputs);
|
|
|
|
auto actual_len = vec::len[arg](actual_inputs);
|
2011-02-18 17:30:57 -08:00
|
|
|
if (expected_len != actual_len) {
|
2011-02-23 11:59:07 -05:00
|
|
|
ret fn_common_res_err(ures_err(terr_arg_count,
|
|
|
|
expected, actual));
|
2011-02-18 17:30:57 -08:00
|
|
|
}
|
2010-12-31 09:48:54 -08:00
|
|
|
|
2011-02-18 17:30:57 -08:00
|
|
|
// TODO: as above, we should have an iter2 iterator.
|
2011-05-16 18:21:22 -07:00
|
|
|
let vec[arg] result_ins = [];
|
2011-02-18 17:30:57 -08:00
|
|
|
auto i = 0u;
|
|
|
|
while (i < expected_len) {
|
|
|
|
auto expected_input = expected_inputs.(i);
|
|
|
|
auto actual_input = actual_inputs.(i);
|
|
|
|
|
2011-05-09 14:00:02 -07:00
|
|
|
// Unify the result modes. "mo_either" unifies with both modes.
|
2011-02-18 17:30:57 -08:00
|
|
|
auto result_mode;
|
2011-05-09 14:00:02 -07:00
|
|
|
if (expected_input.mode == mo_either) {
|
|
|
|
result_mode = actual_input.mode;
|
|
|
|
} else if (actual_input.mode == mo_either) {
|
|
|
|
result_mode = expected_input.mode;
|
|
|
|
} else if (expected_input.mode != actual_input.mode) {
|
2011-05-13 15:17:24 +02:00
|
|
|
// FIXME this is the wrong error
|
2011-05-09 14:00:02 -07:00
|
|
|
ret fn_common_res_err(ures_err(terr_arg_count,
|
|
|
|
expected, actual));
|
2011-02-18 17:30:57 -08:00
|
|
|
} else {
|
2011-05-09 14:00:02 -07:00
|
|
|
result_mode = expected_input.mode;
|
2011-02-18 17:30:57 -08:00
|
|
|
}
|
2010-12-31 09:48:54 -08:00
|
|
|
|
2011-04-21 12:10:53 -07:00
|
|
|
auto result = unify_step(cx, actual_input.ty, expected_input.ty);
|
2010-12-31 09:48:54 -08:00
|
|
|
|
2011-02-18 17:30:57 -08:00
|
|
|
alt (result) {
|
|
|
|
case (ures_ok(?rty)) {
|
2011-05-16 18:21:22 -07:00
|
|
|
result_ins += [rec(mode=result_mode, ty=rty)];
|
2011-02-18 17:30:57 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
case (_) {
|
2011-02-23 11:59:07 -05:00
|
|
|
ret fn_common_res_err(result);
|
2011-02-18 17:30:57 -08:00
|
|
|
}
|
|
|
|
}
|
2010-12-31 09:48:54 -08:00
|
|
|
|
2011-02-18 17:30:57 -08:00
|
|
|
i += 1u;
|
2010-12-31 09:48:54 -08:00
|
|
|
}
|
|
|
|
|
2011-02-18 17:30:57 -08:00
|
|
|
// Check the output.
|
2011-04-21 12:10:53 -07:00
|
|
|
auto result = unify_step(cx, expected_output, actual_output);
|
2011-02-18 17:30:57 -08:00
|
|
|
alt (result) {
|
|
|
|
case (ures_ok(?rty)) {
|
2011-02-23 11:59:07 -05:00
|
|
|
ret fn_common_res_ok(result_ins, rty);
|
2011-02-18 17:30:57 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
case (_) {
|
2011-02-23 11:59:07 -05:00
|
|
|
ret fn_common_res_err(result);
|
2011-02-18 17:30:57 -08:00
|
|
|
}
|
2010-12-31 09:48:54 -08:00
|
|
|
}
|
2011-02-23 11:59:07 -05:00
|
|
|
}
|
2010-12-31 09:48:54 -08:00
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn unify_fn(&@ctxt cx,
|
2011-05-12 17:24:54 +02:00
|
|
|
&ast::proto e_proto,
|
|
|
|
&ast::proto a_proto,
|
2011-05-09 13:09:20 -07:00
|
|
|
&t expected,
|
|
|
|
&t actual,
|
|
|
|
&vec[arg] expected_inputs, &t expected_output,
|
2011-05-20 14:15:39 -07:00
|
|
|
&vec[arg] actual_inputs, &t actual_output,
|
|
|
|
&controlflow expected_cf, &controlflow actual_cf)
|
2011-04-21 11:46:31 -07:00
|
|
|
-> result {
|
2010-12-31 09:48:54 -08:00
|
|
|
|
2011-02-23 11:59:07 -05:00
|
|
|
if (e_proto != a_proto) {
|
|
|
|
ret ures_err(terr_mismatch, expected, actual);
|
|
|
|
}
|
2011-05-20 14:15:39 -07:00
|
|
|
alt (expected_cf) {
|
|
|
|
case (ast::return) { } // ok
|
|
|
|
case (ast::noreturn) {
|
|
|
|
alt (actual_cf) {
|
|
|
|
case (ast::noreturn) {
|
|
|
|
// ok
|
|
|
|
}
|
|
|
|
case (_) {
|
|
|
|
/* 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 */
|
|
|
|
ret ures_err(terr_controlflow_mismatch,
|
|
|
|
expected, actual);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2011-04-21 12:10:53 -07:00
|
|
|
auto t = unify_fn_common(cx, expected, actual,
|
|
|
|
expected_inputs, expected_output,
|
2011-02-23 11:59:07 -05:00
|
|
|
actual_inputs, actual_output);
|
|
|
|
alt (t) {
|
|
|
|
case (fn_common_res_err(?r)) {
|
|
|
|
ret r;
|
|
|
|
}
|
|
|
|
case (fn_common_res_ok(?result_ins, ?result_out)) {
|
2011-05-20 14:15:39 -07:00
|
|
|
auto t2 = mk_fn(cx.tcx, e_proto, result_ins, result_out,
|
|
|
|
actual_cf);
|
2011-02-23 11:59:07 -05:00
|
|
|
ret ures_ok(t2);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn unify_native_fn(&@ctxt cx,
|
2011-05-12 17:24:54 +02:00
|
|
|
&ast::native_abi e_abi,
|
|
|
|
&ast::native_abi a_abi,
|
2011-05-09 13:09:20 -07:00
|
|
|
&t expected,
|
|
|
|
&t actual,
|
|
|
|
&vec[arg] expected_inputs, &t expected_output,
|
|
|
|
&vec[arg] actual_inputs, &t actual_output)
|
2011-04-21 11:46:31 -07:00
|
|
|
-> result {
|
2011-02-25 15:58:08 -05:00
|
|
|
if (e_abi != a_abi) {
|
|
|
|
ret ures_err(terr_mismatch, expected, actual);
|
|
|
|
}
|
|
|
|
|
2011-04-21 12:10:53 -07:00
|
|
|
auto t = unify_fn_common(cx, expected, actual,
|
|
|
|
expected_inputs, expected_output,
|
2011-02-23 11:59:07 -05:00
|
|
|
actual_inputs, actual_output);
|
|
|
|
alt (t) {
|
|
|
|
case (fn_common_res_err(?r)) {
|
|
|
|
ret r;
|
|
|
|
}
|
|
|
|
case (fn_common_res_ok(?result_ins, ?result_out)) {
|
2011-04-25 12:15:55 -07:00
|
|
|
auto t2 = mk_native_fn(cx.tcx, e_abi, result_ins,
|
2011-04-20 18:52:04 -07:00
|
|
|
result_out);
|
2011-02-23 11:59:07 -05:00
|
|
|
ret ures_ok(t2);
|
|
|
|
}
|
|
|
|
}
|
2010-12-31 09:48:54 -08:00
|
|
|
}
|
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn unify_obj(&@ctxt cx,
|
|
|
|
&t expected,
|
|
|
|
&t actual,
|
|
|
|
&vec[method] expected_meths,
|
|
|
|
&vec[method] actual_meths) -> result {
|
2011-05-16 18:21:22 -07:00
|
|
|
let vec[method] result_meths = [];
|
2010-12-31 10:26:29 -08:00
|
|
|
let uint i = 0u;
|
2011-05-17 20:41:41 +02:00
|
|
|
let uint expected_len = vec::len[method](expected_meths);
|
|
|
|
let uint actual_len = vec::len[method](actual_meths);
|
2010-12-31 10:26:29 -08:00
|
|
|
|
|
|
|
if (expected_len != actual_len) {
|
|
|
|
ret ures_err(terr_meth_count, expected, actual);
|
|
|
|
}
|
|
|
|
|
|
|
|
while (i < expected_len) {
|
|
|
|
auto e_meth = expected_meths.(i);
|
|
|
|
auto a_meth = actual_meths.(i);
|
2011-05-17 20:41:41 +02:00
|
|
|
if (! str::eq(e_meth.ident, a_meth.ident)) {
|
2010-12-31 10:26:29 -08:00
|
|
|
ret ures_err(terr_obj_meths(e_meth.ident, a_meth.ident),
|
|
|
|
expected, actual);
|
|
|
|
}
|
2011-04-21 12:10:53 -07:00
|
|
|
auto r = unify_fn(cx,
|
2011-02-18 17:30:57 -08:00
|
|
|
e_meth.proto, a_meth.proto,
|
2011-04-21 12:10:53 -07:00
|
|
|
expected, actual,
|
2010-12-31 10:26:29 -08:00
|
|
|
e_meth.inputs, e_meth.output,
|
2011-05-20 14:15:39 -07:00
|
|
|
a_meth.inputs, a_meth.output,
|
|
|
|
e_meth.cf, a_meth.cf);
|
2011-03-06 13:56:38 -05:00
|
|
|
alt (r) {
|
|
|
|
case (ures_ok(?tfn)) {
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx.tcx, tfn)) {
|
2011-05-20 14:15:39 -07:00
|
|
|
case (ty_fn(?proto, ?ins, ?out, ?cf)) {
|
2011-05-16 18:21:22 -07:00
|
|
|
result_meths += [rec(inputs = ins,
|
2011-05-20 14:15:39 -07:00
|
|
|
output = out,
|
|
|
|
cf = cf
|
|
|
|
with e_meth)];
|
2011-03-06 13:56:38 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case (_) {
|
|
|
|
ret r;
|
|
|
|
}
|
2010-12-31 10:26:29 -08:00
|
|
|
}
|
|
|
|
i += 1u;
|
|
|
|
}
|
2011-04-25 12:15:55 -07:00
|
|
|
auto t = mk_obj(cx.tcx, result_meths);
|
2010-12-31 10:26:29 -08:00
|
|
|
ret ures_ok(t);
|
|
|
|
}
|
|
|
|
|
2011-05-19 16:06:23 -07:00
|
|
|
fn get_or_create_set[T](&@bindings[T] bindings, &T key) -> uint {
|
2011-04-08 14:53:16 -07:00
|
|
|
auto set_num;
|
2011-05-19 16:06:23 -07:00
|
|
|
alt (bindings.ids.find(key)) {
|
2011-05-18 11:53:26 -07:00
|
|
|
case (none[uint]) {
|
2011-05-19 16:06:23 -07:00
|
|
|
set_num = ufind::make_set(bindings.sets);
|
|
|
|
bindings.ids.insert(key, set_num);
|
2011-05-18 11:53:26 -07:00
|
|
|
}
|
|
|
|
case (some[uint](?n)) { set_num = n; }
|
2011-02-18 14:52:33 -08:00
|
|
|
}
|
2011-04-08 14:53:16 -07:00
|
|
|
ret set_num;
|
2011-02-18 14:52:33 -08:00
|
|
|
}
|
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn unify_step(&@ctxt cx, &t expected, &t actual) -> result {
|
2010-12-21 17:47:13 -08:00
|
|
|
// TODO: rewrite this using tuple pattern matching when available, to
|
|
|
|
// avoid all this rightward drift and spikiness.
|
|
|
|
|
2011-02-18 14:52:33 -08:00
|
|
|
// TODO: occurs check, to make sure we don't loop forever when
|
|
|
|
// unifying e.g. 'a and option['a]
|
|
|
|
|
2011-04-22 10:51:32 -07:00
|
|
|
// Fast path.
|
|
|
|
if (eq_ty(expected, actual)) { ret ures_ok(expected); }
|
|
|
|
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx.tcx, actual)) {
|
2011-01-04 16:52:06 -08:00
|
|
|
// If the RHS is a variable type, then just do the appropriate
|
|
|
|
// binding.
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ty::ty_var(?actual_id)) {
|
2011-05-19 16:06:23 -07:00
|
|
|
auto actual_n = get_or_create_set[int](cx.bindings,
|
|
|
|
actual_id);
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx.tcx, expected)) {
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ty::ty_var(?expected_id)) {
|
2011-05-19 16:06:23 -07:00
|
|
|
auto expected_n = get_or_create_set[int](cx.bindings,
|
|
|
|
expected_id);
|
2011-05-19 11:14:04 -07:00
|
|
|
ufind::union(cx.bindings.sets, expected_n, actual_n);
|
2011-04-08 14:53:16 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
case (_) {
|
|
|
|
// Just bind the type variable to the expected type.
|
2011-05-19 16:06:23 -07:00
|
|
|
alt (record_binding[int](cx, cx.bindings, actual_id,
|
|
|
|
expected)) {
|
|
|
|
case (ures_ok(_)) { /* fall through */ }
|
|
|
|
case (?res) { ret res; }
|
2011-04-08 14:53:16 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ret ures_ok(actual);
|
2010-12-21 17:47:13 -08:00
|
|
|
}
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ty::ty_local(?actual_id)) {
|
2011-04-08 21:27:54 -07:00
|
|
|
auto result_ty;
|
2011-04-21 12:10:53 -07:00
|
|
|
alt (cx.handler.resolve_local(actual_id)) {
|
2011-04-22 12:27:28 -07:00
|
|
|
case (none[t]) { result_ty = expected; }
|
|
|
|
case (some[t](?actual_ty)) {
|
2011-04-21 12:10:53 -07:00
|
|
|
auto result = unify_step(cx, expected, actual_ty);
|
2011-04-08 21:27:54 -07:00
|
|
|
alt (result) {
|
|
|
|
case (ures_ok(?rty)) { result_ty = rty; }
|
|
|
|
case (_) { ret result; }
|
|
|
|
}
|
2010-12-21 17:47:13 -08:00
|
|
|
}
|
|
|
|
}
|
2011-04-08 21:27:54 -07:00
|
|
|
|
2011-04-21 12:10:53 -07:00
|
|
|
cx.handler.record_local(actual_id, result_ty);
|
2011-04-08 21:27:54 -07:00
|
|
|
ret ures_ok(result_ty);
|
2010-12-21 17:47:13 -08:00
|
|
|
}
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ty::ty_bound_param(?actual_id)) {
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx.tcx, expected)) {
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ty::ty_local(_)) {
|
2011-04-19 12:02:06 +02:00
|
|
|
log_err "TODO: bound param unifying with local";
|
2011-04-08 21:27:54 -07:00
|
|
|
fail;
|
|
|
|
}
|
2011-01-18 17:18:51 -08:00
|
|
|
|
|
|
|
case (_) {
|
2011-04-21 12:10:53 -07:00
|
|
|
ret cx.handler.record_param(actual_id, expected);
|
2011-01-18 17:18:51 -08:00
|
|
|
}
|
|
|
|
}
|
2011-01-04 16:52:06 -08:00
|
|
|
}
|
2010-12-21 17:47:13 -08:00
|
|
|
case (_) { /* empty */ }
|
|
|
|
}
|
|
|
|
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx.tcx, expected)) {
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ty::ty_nil) { ret struct_cmp(cx, expected, actual); }
|
2011-05-20 19:50:29 -07:00
|
|
|
// _|_ unifies with anything
|
|
|
|
case (ty::ty_bot) { ret ures_ok(expected); }
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ty::ty_bool) { ret struct_cmp(cx, expected, actual); }
|
|
|
|
case (ty::ty_int) { ret struct_cmp(cx, expected, actual); }
|
|
|
|
case (ty::ty_uint) { ret struct_cmp(cx, expected, actual); }
|
|
|
|
case (ty::ty_machine(_)) { ret struct_cmp(cx, expected, actual); }
|
|
|
|
case (ty::ty_float) { ret struct_cmp(cx, expected, actual); }
|
|
|
|
case (ty::ty_char) { ret struct_cmp(cx, expected, actual); }
|
|
|
|
case (ty::ty_str) { ret struct_cmp(cx, expected, actual); }
|
|
|
|
case (ty::ty_type) { ret struct_cmp(cx, expected, actual); }
|
|
|
|
case (ty::ty_native) { ret struct_cmp(cx, expected, actual); }
|
|
|
|
case (ty::ty_param(_)) { ret struct_cmp(cx, expected, actual); }
|
|
|
|
|
|
|
|
case (ty::ty_tag(?expected_id, ?expected_tps)) {
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx.tcx, actual)) {
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ty::ty_tag(?actual_id, ?actual_tps)) {
|
2011-02-17 16:23:31 -08:00
|
|
|
if (expected_id._0 != actual_id._0 ||
|
|
|
|
expected_id._1 != actual_id._1) {
|
|
|
|
ret ures_err(terr_mismatch, expected, actual);
|
2010-12-21 17:47:13 -08:00
|
|
|
}
|
2011-02-17 16:23:31 -08:00
|
|
|
|
|
|
|
// TODO: factor this cruft out, see the TODO in the
|
2011-05-12 17:24:54 +02:00
|
|
|
// ty::ty_tup case
|
2011-05-16 18:21:22 -07:00
|
|
|
let vec[t] result_tps = [];
|
2011-02-17 16:23:31 -08:00
|
|
|
auto i = 0u;
|
2011-05-17 20:41:41 +02:00
|
|
|
auto expected_len = vec::len[t](expected_tps);
|
2011-02-17 16:23:31 -08:00
|
|
|
while (i < expected_len) {
|
|
|
|
auto expected_tp = expected_tps.(i);
|
|
|
|
auto actual_tp = actual_tps.(i);
|
|
|
|
|
2011-04-21 12:10:53 -07:00
|
|
|
auto result = unify_step(cx,
|
2011-02-17 16:23:31 -08:00
|
|
|
expected_tp,
|
2011-04-21 12:10:53 -07:00
|
|
|
actual_tp);
|
2011-02-17 16:23:31 -08:00
|
|
|
|
|
|
|
alt (result) {
|
|
|
|
case (ures_ok(?rty)) {
|
2011-05-17 20:41:41 +02:00
|
|
|
vec::push[t](result_tps, rty);
|
2011-02-17 16:23:31 -08:00
|
|
|
}
|
|
|
|
case (_) {
|
|
|
|
ret result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
i += 1u;
|
|
|
|
}
|
|
|
|
|
2011-04-25 12:15:55 -07:00
|
|
|
ret ures_ok(mk_tag(cx.tcx, expected_id, result_tps));
|
2010-12-21 17:47:13 -08:00
|
|
|
}
|
|
|
|
case (_) { /* fall through */ }
|
|
|
|
}
|
|
|
|
|
|
|
|
ret ures_err(terr_mismatch, expected, actual);
|
|
|
|
}
|
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ty::ty_box(?expected_mt)) {
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx.tcx, actual)) {
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ty::ty_box(?actual_mt)) {
|
2011-03-18 11:49:06 -07:00
|
|
|
auto mut;
|
|
|
|
alt (unify_mut(expected_mt.mut, actual_mt.mut)) {
|
2011-05-12 17:24:54 +02:00
|
|
|
case (none[ast::mutability]) {
|
2011-03-18 11:49:06 -07:00
|
|
|
ret ures_err(terr_box_mutability, expected,
|
|
|
|
actual);
|
|
|
|
}
|
2011-05-12 17:24:54 +02:00
|
|
|
case (some[ast::mutability](?m)) { mut = m; }
|
2011-03-17 17:39:47 -07:00
|
|
|
}
|
|
|
|
|
2011-04-21 12:10:53 -07:00
|
|
|
auto result = unify_step(cx,
|
2011-03-17 17:39:47 -07:00
|
|
|
expected_mt.ty,
|
2011-04-21 12:10:53 -07:00
|
|
|
actual_mt.ty);
|
2010-12-21 17:47:13 -08:00
|
|
|
alt (result) {
|
|
|
|
case (ures_ok(?result_sub)) {
|
2011-03-18 11:49:06 -07:00
|
|
|
auto mt = rec(ty=result_sub, mut=mut);
|
2011-04-25 12:15:55 -07:00
|
|
|
ret ures_ok(mk_box(cx.tcx, mt));
|
2010-12-21 17:47:13 -08:00
|
|
|
}
|
|
|
|
case (_) {
|
|
|
|
ret result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
case (_) {
|
|
|
|
ret ures_err(terr_mismatch, expected, actual);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ty::ty_vec(?expected_mt)) {
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx.tcx, actual)) {
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ty::ty_vec(?actual_mt)) {
|
2011-03-18 11:49:06 -07:00
|
|
|
auto mut;
|
|
|
|
alt (unify_mut(expected_mt.mut, actual_mt.mut)) {
|
2011-05-12 17:24:54 +02:00
|
|
|
case (none[ast::mutability]) {
|
2011-03-18 11:49:06 -07:00
|
|
|
ret ures_err(terr_vec_mutability, expected,
|
|
|
|
actual);
|
|
|
|
}
|
2011-05-12 17:24:54 +02:00
|
|
|
case (some[ast::mutability](?m)) { mut = m; }
|
2011-03-17 17:39:47 -07:00
|
|
|
}
|
|
|
|
|
2011-04-21 12:10:53 -07:00
|
|
|
auto result = unify_step(cx,
|
2011-03-17 17:39:47 -07:00
|
|
|
expected_mt.ty,
|
2011-04-21 12:10:53 -07:00
|
|
|
actual_mt.ty);
|
2010-12-21 17:47:13 -08:00
|
|
|
alt (result) {
|
|
|
|
case (ures_ok(?result_sub)) {
|
2011-03-18 11:49:06 -07:00
|
|
|
auto mt = rec(ty=result_sub, mut=mut);
|
2011-04-25 12:15:55 -07:00
|
|
|
ret ures_ok(mk_vec(cx.tcx, mt));
|
2010-12-21 17:47:13 -08:00
|
|
|
}
|
|
|
|
case (_) {
|
|
|
|
ret result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
case (_) {
|
|
|
|
ret ures_err(terr_mismatch, expected, actual);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ty::ty_port(?expected_sub)) {
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx.tcx, actual)) {
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ty::ty_port(?actual_sub)) {
|
2011-04-21 12:10:53 -07:00
|
|
|
auto result = unify_step(cx,
|
2011-03-10 22:58:55 -05:00
|
|
|
expected_sub,
|
2011-04-21 12:10:53 -07:00
|
|
|
actual_sub);
|
2011-03-10 22:58:55 -05:00
|
|
|
alt (result) {
|
|
|
|
case (ures_ok(?result_sub)) {
|
2011-04-25 12:15:55 -07:00
|
|
|
ret ures_ok(mk_port(cx.tcx, result_sub));
|
2011-03-10 22:58:55 -05:00
|
|
|
}
|
|
|
|
case (_) {
|
|
|
|
ret result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
case (_) {
|
|
|
|
ret ures_err(terr_mismatch, expected, actual);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ty::ty_chan(?expected_sub)) {
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx.tcx, actual)) {
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ty::ty_chan(?actual_sub)) {
|
2011-04-21 12:10:53 -07:00
|
|
|
auto result = unify_step(cx,
|
2011-03-10 22:58:55 -05:00
|
|
|
expected_sub,
|
2011-04-21 12:10:53 -07:00
|
|
|
actual_sub);
|
2011-03-10 22:58:55 -05:00
|
|
|
alt (result) {
|
|
|
|
case (ures_ok(?result_sub)) {
|
2011-04-25 12:15:55 -07:00
|
|
|
ret ures_ok(mk_chan(cx.tcx, result_sub));
|
2011-03-10 22:58:55 -05:00
|
|
|
}
|
|
|
|
case (_) {
|
|
|
|
ret result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
case (_) {
|
|
|
|
ret ures_err(terr_mismatch, expected, actual);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ty::ty_tup(?expected_elems)) {
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx.tcx, actual)) {
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ty::ty_tup(?actual_elems)) {
|
2011-05-17 20:41:41 +02:00
|
|
|
auto expected_len = vec::len[ty::mt](expected_elems);
|
|
|
|
auto actual_len = vec::len[ty::mt](actual_elems);
|
2010-12-21 17:47:13 -08:00
|
|
|
if (expected_len != actual_len) {
|
|
|
|
auto err = terr_tuple_size(expected_len,
|
|
|
|
actual_len);
|
|
|
|
ret ures_err(err, expected, actual);
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: implement an iterator that can iterate over
|
|
|
|
// two arrays simultaneously.
|
2011-05-16 18:21:22 -07:00
|
|
|
let vec[ty::mt] result_elems = [];
|
2010-12-21 17:47:13 -08:00
|
|
|
auto i = 0u;
|
|
|
|
while (i < expected_len) {
|
|
|
|
auto expected_elem = expected_elems.(i);
|
|
|
|
auto actual_elem = actual_elems.(i);
|
2011-03-18 11:49:06 -07:00
|
|
|
|
|
|
|
auto mut;
|
|
|
|
alt (unify_mut(expected_elem.mut,
|
|
|
|
actual_elem.mut)) {
|
2011-05-12 17:24:54 +02:00
|
|
|
case (none[ast::mutability]) {
|
2011-03-18 11:49:06 -07:00
|
|
|
auto err = terr_tuple_mutability;
|
|
|
|
ret ures_err(err, expected, actual);
|
|
|
|
}
|
2011-05-12 17:24:54 +02:00
|
|
|
case (some[ast::mutability](?m)) { mut = m; }
|
2010-12-21 17:47:13 -08:00
|
|
|
}
|
|
|
|
|
2011-04-21 12:10:53 -07:00
|
|
|
auto result = unify_step(cx,
|
2011-03-17 17:39:47 -07:00
|
|
|
expected_elem.ty,
|
2011-04-21 12:10:53 -07:00
|
|
|
actual_elem.ty);
|
2010-12-21 17:47:13 -08:00
|
|
|
alt (result) {
|
|
|
|
case (ures_ok(?rty)) {
|
2011-03-18 11:49:06 -07:00
|
|
|
auto mt = rec(ty=rty, mut=mut);
|
2011-05-16 18:21:22 -07:00
|
|
|
result_elems += [mt];
|
2010-12-21 17:47:13 -08:00
|
|
|
}
|
|
|
|
case (_) {
|
|
|
|
ret result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
i += 1u;
|
|
|
|
}
|
|
|
|
|
2011-04-25 12:15:55 -07:00
|
|
|
ret ures_ok(mk_tup(cx.tcx, result_elems));
|
2010-12-21 17:47:13 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
case (_) {
|
|
|
|
ret ures_err(terr_mismatch, expected, actual);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ty::ty_rec(?expected_fields)) {
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx.tcx, actual)) {
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ty::ty_rec(?actual_fields)) {
|
2011-05-17 20:41:41 +02:00
|
|
|
auto expected_len = vec::len[field](expected_fields);
|
|
|
|
auto actual_len = vec::len[field](actual_fields);
|
2010-12-21 17:47:13 -08:00
|
|
|
if (expected_len != actual_len) {
|
|
|
|
auto err = terr_record_size(expected_len,
|
|
|
|
actual_len);
|
|
|
|
ret ures_err(err, expected, actual);
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: implement an iterator that can iterate over
|
|
|
|
// two arrays simultaneously.
|
2011-05-16 18:21:22 -07:00
|
|
|
let vec[field] result_fields = [];
|
2010-12-21 17:47:13 -08:00
|
|
|
auto i = 0u;
|
|
|
|
while (i < expected_len) {
|
|
|
|
auto expected_field = expected_fields.(i);
|
|
|
|
auto actual_field = actual_fields.(i);
|
2011-03-18 11:49:06 -07:00
|
|
|
|
|
|
|
auto mut;
|
|
|
|
alt (unify_mut(expected_field.mt.mut,
|
|
|
|
actual_field.mt.mut)) {
|
2011-05-12 17:24:54 +02:00
|
|
|
case (none[ast::mutability]) {
|
2011-03-18 11:49:06 -07:00
|
|
|
ret ures_err(terr_record_mutability,
|
|
|
|
expected, actual);
|
|
|
|
}
|
2011-05-12 17:24:54 +02:00
|
|
|
case (some[ast::mutability](?m)) { mut = m; }
|
2010-12-21 17:47:13 -08:00
|
|
|
}
|
|
|
|
|
2011-05-17 20:41:41 +02:00
|
|
|
if (!str::eq(expected_field.ident,
|
2011-03-17 17:39:47 -07:00
|
|
|
actual_field.ident)) {
|
2010-12-21 17:47:13 -08:00
|
|
|
auto err =
|
|
|
|
terr_record_fields(expected_field.ident,
|
|
|
|
actual_field.ident);
|
|
|
|
ret ures_err(err, expected, actual);
|
|
|
|
}
|
|
|
|
|
2011-04-21 12:10:53 -07:00
|
|
|
auto result = unify_step(cx,
|
2011-03-17 17:39:47 -07:00
|
|
|
expected_field.mt.ty,
|
2011-04-21 12:10:53 -07:00
|
|
|
actual_field.mt.ty);
|
2010-12-21 17:47:13 -08:00
|
|
|
alt (result) {
|
|
|
|
case (ures_ok(?rty)) {
|
2011-03-18 11:49:06 -07:00
|
|
|
auto mt = rec(ty=rty, mut=mut);
|
2011-05-17 20:41:41 +02:00
|
|
|
vec::push[field]
|
2010-12-21 17:47:13 -08:00
|
|
|
(result_fields,
|
2011-03-17 17:39:47 -07:00
|
|
|
rec(mt=mt with expected_field));
|
2010-12-21 17:47:13 -08:00
|
|
|
}
|
|
|
|
case (_) {
|
|
|
|
ret result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
i += 1u;
|
|
|
|
}
|
|
|
|
|
2011-04-25 12:15:55 -07:00
|
|
|
ret ures_ok(mk_rec(cx.tcx, result_fields));
|
2010-12-21 17:47:13 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
case (_) {
|
|
|
|
ret ures_err(terr_mismatch, expected, actual);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-05-20 14:15:39 -07:00
|
|
|
case (ty::ty_fn(?ep, ?expected_inputs,
|
|
|
|
?expected_output, ?expected_cf)) {
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx.tcx, actual)) {
|
2011-05-20 14:15:39 -07:00
|
|
|
case (ty::ty_fn(?ap, ?actual_inputs,
|
|
|
|
?actual_output, ?actual_cf)) {
|
2011-04-21 12:10:53 -07:00
|
|
|
ret unify_fn(cx, ep, ap,
|
|
|
|
expected, actual,
|
2011-02-18 17:30:57 -08:00
|
|
|
expected_inputs, expected_output,
|
2011-05-20 14:15:39 -07:00
|
|
|
actual_inputs, actual_output,
|
|
|
|
expected_cf, actual_cf);
|
2010-12-21 17:47:13 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
case (_) {
|
|
|
|
ret ures_err(terr_mismatch, expected, actual);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ty::ty_native_fn(?e_abi, ?expected_inputs,
|
2011-02-25 15:58:08 -05:00
|
|
|
?expected_output)) {
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx.tcx, actual)) {
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ty::ty_native_fn(?a_abi, ?actual_inputs,
|
2011-02-25 15:58:08 -05:00
|
|
|
?actual_output)) {
|
2011-04-21 12:10:53 -07:00
|
|
|
ret unify_native_fn(cx, e_abi, a_abi,
|
|
|
|
expected, actual,
|
2011-02-23 11:59:07 -05:00
|
|
|
expected_inputs, expected_output,
|
|
|
|
actual_inputs, actual_output);
|
|
|
|
}
|
|
|
|
case (_) {
|
|
|
|
ret ures_err(terr_mismatch, expected, actual);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ty::ty_obj(?expected_meths)) {
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx.tcx, actual)) {
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ty::ty_obj(?actual_meths)) {
|
2011-04-21 12:10:53 -07:00
|
|
|
ret unify_obj(cx, expected, actual,
|
2011-02-18 17:30:57 -08:00
|
|
|
expected_meths, actual_meths);
|
|
|
|
}
|
|
|
|
case (_) {
|
|
|
|
ret ures_err(terr_mismatch, expected, actual);
|
|
|
|
}
|
2010-12-31 10:26:29 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ty::ty_var(?expected_id)) {
|
2011-05-19 16:06:23 -07:00
|
|
|
// Add a binding. (`actual` can't actually be a var here.)
|
|
|
|
alt (record_binding[int](cx, cx.bindings, expected_id,
|
|
|
|
actual)) {
|
|
|
|
case (ures_ok(_)) { /* fall through */ }
|
|
|
|
case (?res) { ret res; }
|
2011-04-08 14:53:16 -07:00
|
|
|
}
|
|
|
|
ret ures_ok(expected);
|
2010-12-21 17:47:13 -08:00
|
|
|
}
|
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ty::ty_local(?expected_id)) {
|
2011-04-08 21:27:54 -07:00
|
|
|
auto result_ty;
|
2011-04-21 12:10:53 -07:00
|
|
|
alt (cx.handler.resolve_local(expected_id)) {
|
2011-04-22 12:27:28 -07:00
|
|
|
case (none[t]) { result_ty = actual; }
|
|
|
|
case (some[t](?expected_ty)) {
|
2011-04-21 12:10:53 -07:00
|
|
|
auto result = unify_step(cx, expected_ty, actual);
|
2011-04-08 21:27:54 -07:00
|
|
|
alt (result) {
|
|
|
|
case (ures_ok(?rty)) { result_ty = rty; }
|
|
|
|
case (_) { ret result; }
|
|
|
|
}
|
2010-12-21 17:47:13 -08:00
|
|
|
}
|
|
|
|
}
|
2011-04-08 21:27:54 -07:00
|
|
|
|
2011-04-21 12:10:53 -07:00
|
|
|
cx.handler.record_local(expected_id, result_ty);
|
2011-04-08 21:27:54 -07:00
|
|
|
ret ures_ok(result_ty);
|
2010-12-21 17:47:13 -08:00
|
|
|
}
|
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ty::ty_bound_param(?expected_id)) {
|
2011-04-21 12:10:53 -07:00
|
|
|
ret cx.handler.record_param(expected_id, actual);
|
2010-12-21 17:47:13 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: remove me once match-exhaustiveness checking works
|
|
|
|
fail;
|
|
|
|
}
|
|
|
|
|
2011-02-18 14:52:33 -08:00
|
|
|
// Performs type binding substitution.
|
2011-05-19 11:14:04 -07:00
|
|
|
fn substitute(&ty_ctxt tcx,
|
|
|
|
&@bindings[int] bindings,
|
|
|
|
&vec[t] set_types,
|
|
|
|
&t typ) -> t {
|
2011-05-18 13:08:02 -07:00
|
|
|
if (!type_contains_vars(tcx, typ)) {
|
2011-04-25 18:05:09 -07:00
|
|
|
ret typ;
|
|
|
|
}
|
|
|
|
|
2011-05-19 11:14:04 -07:00
|
|
|
fn substituter(ty_ctxt tcx,
|
|
|
|
@bindings[int] bindings,
|
|
|
|
vec[t] types,
|
2011-05-18 13:08:02 -07:00
|
|
|
t typ) -> t {
|
|
|
|
alt (struct(tcx, typ)) {
|
2011-04-08 14:53:16 -07:00
|
|
|
case (ty_var(?id)) {
|
2011-05-19 11:14:04 -07:00
|
|
|
alt (bindings.ids.find(id)) {
|
2011-04-15 12:23:00 -07:00
|
|
|
case (some[uint](?n)) {
|
2011-05-19 11:14:04 -07:00
|
|
|
auto root = ufind::find(bindings.sets, n);
|
2011-04-15 12:23:00 -07:00
|
|
|
ret types.(root);
|
|
|
|
}
|
|
|
|
case (none[uint]) { ret typ; }
|
2011-02-18 14:52:33 -08:00
|
|
|
}
|
|
|
|
}
|
2011-04-08 14:53:16 -07:00
|
|
|
case (_) { ret typ; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-05-19 11:14:04 -07:00
|
|
|
auto f = bind substituter(tcx, bindings, set_types, _);
|
2011-05-18 13:08:02 -07:00
|
|
|
ret fold_ty(tcx, f, typ);
|
2011-04-08 14:53:16 -07:00
|
|
|
}
|
|
|
|
|
2011-05-19 16:06:23 -07:00
|
|
|
fn unify_sets[T](&ty_ctxt tcx, &@bindings[T] bindings) -> set_result {
|
|
|
|
obj handler() {
|
|
|
|
fn resolve_local(ast::def_id id) -> option::t[t] {
|
|
|
|
log_err "resolve_local in unify_sets";
|
|
|
|
fail;
|
|
|
|
}
|
|
|
|
fn record_local(ast::def_id id, t ty) {
|
|
|
|
log_err "record_local in unify_sets";
|
|
|
|
fail;
|
|
|
|
}
|
|
|
|
fn record_param(uint index, t binding) -> unify::result {
|
|
|
|
log_err "record_param in unify_sets";
|
|
|
|
fail;
|
|
|
|
}
|
2011-04-08 14:53:16 -07:00
|
|
|
}
|
|
|
|
|
2011-05-19 16:06:23 -07:00
|
|
|
auto node_count = vec::len[option::t[t]](bindings.types);
|
|
|
|
|
|
|
|
let vec[option::t[t]] results =
|
|
|
|
vec::init_elt[option::t[t]](none[t], node_count);
|
|
|
|
|
2011-04-08 14:53:16 -07:00
|
|
|
auto i = 0u;
|
2011-05-19 16:06:23 -07:00
|
|
|
while (i < node_count) {
|
2011-05-19 11:14:04 -07:00
|
|
|
auto root = ufind::find(bindings.sets, i);
|
2011-05-19 16:06:23 -07:00
|
|
|
alt (bindings.types.(i)) {
|
|
|
|
case (none[t]) { /* nothing to do */ }
|
|
|
|
case (some[t](?actual)) {
|
|
|
|
alt (results.(root)) {
|
|
|
|
case (none[t]) { results.(root) = some[t](actual); }
|
|
|
|
case (some[t](?expected)) {
|
|
|
|
// FIXME: Is this right?
|
|
|
|
auto bindings = mk_bindings[int](int::hash,
|
|
|
|
int::eq_alias);
|
|
|
|
alt (unify(expected, actual, handler(), bindings,
|
|
|
|
tcx)) {
|
|
|
|
case (ures_ok(?result_ty)) {
|
|
|
|
results.(i) = some[t](result_ty);
|
|
|
|
}
|
|
|
|
case (ures_err(?e, ?t_a, ?t_b)) {
|
|
|
|
ret usr_err(e, t_a, t_b);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2011-04-08 14:53:16 -07:00
|
|
|
i += 1u;
|
|
|
|
}
|
|
|
|
|
2011-05-19 16:06:23 -07:00
|
|
|
// FIXME: This is equivalent to map(option::get, results) but it
|
|
|
|
// causes an assertion in typeck at the moment.
|
|
|
|
let vec[t] real_results = [];
|
|
|
|
for (option::t[t] typ in results) {
|
|
|
|
real_results += [option::get[t](typ)];
|
2011-02-18 14:52:33 -08:00
|
|
|
}
|
|
|
|
|
2011-05-19 16:06:23 -07:00
|
|
|
ret usr_ok(real_results);
|
2011-02-18 14:52:33 -08:00
|
|
|
}
|
|
|
|
|
2011-05-09 13:09:20 -07:00
|
|
|
fn unify(&t expected,
|
|
|
|
&t actual,
|
2011-04-20 18:52:04 -07:00
|
|
|
&unify_handler handler,
|
2011-05-19 11:14:04 -07:00
|
|
|
&@bindings[int] bindings,
|
2011-05-09 13:09:20 -07:00
|
|
|
&ty_ctxt tcx) -> result {
|
2011-05-19 11:14:04 -07:00
|
|
|
auto cx = @rec(bindings=bindings, handler=handler, tcx=tcx);
|
2011-05-18 13:08:02 -07:00
|
|
|
ret unify_step(cx, expected, actual);
|
|
|
|
}
|
2011-04-22 10:37:51 -07:00
|
|
|
|
2011-05-19 16:06:23 -07:00
|
|
|
fn fixup(&ty_ctxt tcx, &@bindings[int] bindings, t typ) -> result {
|
|
|
|
alt (unify_sets[int](tcx, bindings)) {
|
|
|
|
case (usr_ok(?set_types)) {
|
|
|
|
ret ures_ok(substitute(tcx, bindings, set_types, typ));
|
|
|
|
}
|
|
|
|
case (usr_err(?terr, ?t0, ?t1)) { ret ures_err(terr, t0, t1); }
|
|
|
|
}
|
2011-02-07 14:11:43 -08:00
|
|
|
}
|
2010-12-21 17:47:13 -08:00
|
|
|
}
|
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
fn type_err_to_str(&ty::type_err err) -> str {
|
2010-12-21 17:47:13 -08:00
|
|
|
alt (err) {
|
|
|
|
case (terr_mismatch) {
|
|
|
|
ret "types differ";
|
|
|
|
}
|
2011-05-20 14:15:39 -07:00
|
|
|
case (terr_controlflow_mismatch) {
|
|
|
|
ret "returning function used where non-returning function"
|
|
|
|
+ " was expected";
|
|
|
|
}
|
2011-03-18 11:49:06 -07:00
|
|
|
case (terr_box_mutability) {
|
|
|
|
ret "boxed values differ in mutability";
|
|
|
|
}
|
|
|
|
case (terr_vec_mutability) {
|
|
|
|
ret "vectors differ in mutability";
|
|
|
|
}
|
2010-12-21 17:47:13 -08:00
|
|
|
case (terr_tuple_size(?e_sz, ?a_sz)) {
|
2011-05-17 20:41:41 +02:00
|
|
|
ret "expected a tuple with " + uint::to_str(e_sz, 10u) +
|
|
|
|
" elements but found one with " + uint::to_str(a_sz, 10u) +
|
2010-12-21 17:47:13 -08:00
|
|
|
" elements";
|
|
|
|
}
|
|
|
|
case (terr_tuple_mutability) {
|
|
|
|
ret "tuple elements differ in mutability";
|
|
|
|
}
|
|
|
|
case (terr_record_size(?e_sz, ?a_sz)) {
|
2011-05-17 20:41:41 +02:00
|
|
|
ret "expected a record with " + uint::to_str(e_sz, 10u) +
|
|
|
|
" fields but found one with " + uint::to_str(a_sz, 10u) +
|
2010-12-21 17:47:13 -08:00
|
|
|
" fields";
|
|
|
|
}
|
|
|
|
case (terr_record_mutability) {
|
|
|
|
ret "record elements differ in mutability";
|
|
|
|
}
|
|
|
|
case (terr_record_fields(?e_fld, ?a_fld)) {
|
|
|
|
ret "expected a record with field '" + e_fld +
|
|
|
|
"' but found one with field '" + a_fld +
|
|
|
|
"'";
|
|
|
|
}
|
|
|
|
case (terr_arg_count) {
|
|
|
|
ret "incorrect number of function parameters";
|
|
|
|
}
|
2010-12-31 10:26:29 -08:00
|
|
|
case (terr_meth_count) {
|
|
|
|
ret "incorrect number of object methods";
|
|
|
|
}
|
|
|
|
case (terr_obj_meths(?e_meth, ?a_meth)) {
|
|
|
|
ret "expected an obj with method '" + e_meth +
|
|
|
|
"' but found one with method '" + a_meth +
|
|
|
|
"'";
|
|
|
|
}
|
2010-12-21 17:47:13 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-04-08 21:27:54 -07:00
|
|
|
// Performs bound type parameter replacement using the supplied mapping from
|
2011-02-24 19:24:12 -08:00
|
|
|
// parameter IDs to types.
|
2011-05-09 13:09:20 -07:00
|
|
|
fn substitute_type_params(&ctxt cx, &vec[t] bindings, &t typ) -> t {
|
2011-04-25 12:15:55 -07:00
|
|
|
if (!type_contains_bound_params(cx, typ)) {
|
2011-04-25 05:39:18 +00:00
|
|
|
ret typ;
|
|
|
|
}
|
2011-05-19 15:47:15 -07:00
|
|
|
fn replacer(&ctxt cx, vec[t] bindings, t typ) -> t {
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx, typ)) {
|
2011-04-15 12:23:00 -07:00
|
|
|
case (ty_bound_param(?param_index)) {
|
|
|
|
ret bindings.(param_index);
|
2011-02-24 19:24:12 -08:00
|
|
|
}
|
2011-04-15 12:23:00 -07:00
|
|
|
case (_) { ret typ; }
|
2011-02-24 19:24:12 -08:00
|
|
|
}
|
|
|
|
}
|
2011-04-15 12:23:00 -07:00
|
|
|
|
2011-04-25 12:15:55 -07:00
|
|
|
auto f = bind replacer(cx, bindings, _);
|
|
|
|
ret fold_ty(cx, f, typ);
|
2011-02-24 19:24:12 -08:00
|
|
|
}
|
|
|
|
|
2011-04-08 21:27:54 -07:00
|
|
|
// Converts type parameters in a type to bound type parameters.
|
2011-05-09 13:09:20 -07:00
|
|
|
fn bind_params_in_type(&ctxt cx, &t typ) -> t {
|
2011-04-25 18:05:09 -07:00
|
|
|
if (!type_contains_params(cx, typ)) {
|
|
|
|
ret typ;
|
|
|
|
}
|
2011-05-19 15:47:15 -07:00
|
|
|
fn binder(&ctxt cx, t typ) -> t {
|
2011-04-25 12:15:55 -07:00
|
|
|
alt (struct(cx, typ)) {
|
2011-04-15 12:23:00 -07:00
|
|
|
case (ty_bound_param(?index)) {
|
2011-04-19 12:02:06 +02:00
|
|
|
log_err "bind_params_in_type() called on type that already " +
|
2011-04-15 12:23:00 -07:00
|
|
|
"has bound params in it";
|
|
|
|
fail;
|
2011-04-08 21:27:54 -07:00
|
|
|
}
|
2011-04-25 12:15:55 -07:00
|
|
|
case (ty_param(?index)) { ret mk_bound_param(cx, index); }
|
2011-04-15 12:23:00 -07:00
|
|
|
case (_) { ret typ; }
|
2011-04-08 21:27:54 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-04-25 12:15:55 -07:00
|
|
|
auto f = bind binder(cx, _);
|
|
|
|
ret fold_ty(cx, f, typ);
|
2011-04-08 21:27:54 -07:00
|
|
|
}
|
|
|
|
|
2011-03-30 17:23:25 -07:00
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
fn def_has_ty_params(&ast::def def) -> bool {
|
2011-03-30 17:23:25 -07:00
|
|
|
alt (def) {
|
2011-05-12 17:24:54 +02:00
|
|
|
case (ast::def_fn(_)) { ret true; }
|
|
|
|
case (ast::def_obj(_)) { ret true; }
|
|
|
|
case (ast::def_obj_field(_)) { ret false; }
|
|
|
|
case (ast::def_mod(_)) { ret false; }
|
|
|
|
case (ast::def_const(_)) { ret false; }
|
|
|
|
case (ast::def_arg(_)) { ret false; }
|
|
|
|
case (ast::def_local(_)) { ret false; }
|
|
|
|
case (ast::def_variant(_, _)) { ret true; }
|
|
|
|
case (ast::def_ty(_)) { ret false; }
|
|
|
|
case (ast::def_ty_arg(_)) { ret false; }
|
|
|
|
case (ast::def_binding(_)) { ret false; }
|
|
|
|
case (ast::def_use(_)) { ret false; }
|
|
|
|
case (ast::def_native_ty(_)) { ret false; }
|
|
|
|
case (ast::def_native_fn(_)) { ret true; }
|
2011-03-30 17:23:25 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-05-19 17:21:21 -07:00
|
|
|
|
|
|
|
// Tag information
|
|
|
|
|
|
|
|
type variant_info = rec(vec[ty::t] args, ty::t ctor_ty, ast::def_id id);
|
|
|
|
|
|
|
|
fn tag_variants(&ctxt cx, &ast::def_id id) -> vec[variant_info] {
|
|
|
|
if (cx.sess.get_targ_crate_num() != id._0) {
|
|
|
|
ret creader::get_tag_variants(cx, id);
|
|
|
|
}
|
|
|
|
|
|
|
|
assert (cx.items.contains_key(id));
|
|
|
|
alt (cx.items.get(id)) {
|
|
|
|
case (any_item_rust(?item)) {
|
|
|
|
alt (item.node) {
|
|
|
|
case (ast::item_tag(_, ?variants, _, _, _)) {
|
|
|
|
let vec[variant_info] result = [];
|
|
|
|
for (ast::variant variant in variants) {
|
|
|
|
auto ctor_ty = ann_to_monotype(cx, variant.node.ann);
|
|
|
|
let vec[t] arg_tys = [];
|
|
|
|
if (vec::len[ast::variant_arg](variant.node.args)
|
|
|
|
> 0u) {
|
|
|
|
for (arg a in ty_fn_args(cx, ctor_ty)) {
|
|
|
|
arg_tys += [a.ty];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
auto did = variant.node.id;
|
|
|
|
result += [rec(args=arg_tys,
|
|
|
|
ctor_ty=ctor_ty,
|
|
|
|
id=did)];
|
|
|
|
}
|
|
|
|
ret result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fail; // not reached
|
|
|
|
}
|
|
|
|
|
|
|
|
// Returns information about the tag variant with the given ID:
|
|
|
|
fn tag_variant_with_id(&ctxt cx,
|
|
|
|
&ast::def_id tag_id,
|
|
|
|
&ast::def_id variant_id) -> variant_info {
|
|
|
|
auto variants = tag_variants(cx, tag_id);
|
|
|
|
|
|
|
|
auto i = 0u;
|
|
|
|
while (i < vec::len[variant_info](variants)) {
|
|
|
|
auto variant = variants.(i);
|
|
|
|
if (common::def_eq(variant.id, variant_id)) {
|
|
|
|
ret variant;
|
|
|
|
}
|
|
|
|
i += 1u;
|
|
|
|
}
|
|
|
|
|
|
|
|
log_err "tag_variant_with_id(): no variant exists with that ID";
|
|
|
|
fail;
|
|
|
|
}
|
|
|
|
|
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-05-19 17:21:21 -07:00
|
|
|
fn lookup_item_type(ctxt cx, ast::def_id did) -> ty_param_count_and_ty {
|
|
|
|
if (did._0 == cx.sess.get_targ_crate_num()) {
|
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-05-19 15:47:15 -07:00
|
|
|
ret cx.tcache.get(did);
|
2011-03-30 17:23:25 -07:00
|
|
|
}
|
|
|
|
|
2011-05-19 15:47:15 -07:00
|
|
|
alt (cx.tcache.find(did)) {
|
2011-04-22 10:19:22 -07:00
|
|
|
case (some[ty_param_count_and_ty](?tpt)) { ret tpt; }
|
|
|
|
case (none[ty_param_count_and_ty]) {
|
2011-05-19 17:21:21 -07:00
|
|
|
auto tyt = creader::get_type(cx, did);
|
2011-05-19 15:47:15 -07:00
|
|
|
cx.tcache.insert(did, tyt);
|
2011-04-22 10:19:22 -07:00
|
|
|
ret tyt;
|
|
|
|
}
|
2011-03-30 17:23:25 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-05-14 19:02:30 -07:00
|
|
|
fn ret_ty_of_fn_ty(ty_ctxt tcx, t a_ty) -> t {
|
|
|
|
alt (ty::struct(tcx, a_ty)) {
|
2011-05-20 14:15:39 -07:00
|
|
|
case (ty::ty_fn(_, _, ?ret_ty, _)) {
|
2011-05-14 19:02:30 -07:00
|
|
|
ret ret_ty;
|
|
|
|
}
|
2011-05-19 17:21:21 -07:00
|
|
|
case (_) {
|
2011-05-14 19:02:30 -07:00
|
|
|
fail;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-05-19 15:47:15 -07:00
|
|
|
fn ret_ty_of_fn(ty_ctxt tcx, ast::ann ann) -> t {
|
|
|
|
ret ret_ty_of_fn_ty(tcx, ann_to_type(tcx.node_types, ann));
|
2011-05-14 19:02:30 -07:00
|
|
|
}
|
2011-03-30 17:23:25 -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
|
2011-03-25 15:07:27 -07:00
|
|
|
// compile-command: "make -k -C $RBUILD 2>&1 | sed -e 's/\\/x\\//x:\\//g'";
|
2011-01-13 17:42:28 -08:00
|
|
|
// End:
|