Use resgister more, fix type_to_vars

This commit is contained in:
Richard Feldman 2019-12-20 18:18:02 -05:00
parent cc3ddd2667
commit 94cc22f9ca
2 changed files with 306 additions and 95 deletions

View file

@ -32,6 +32,10 @@ impl Pools {
Pools(pools) Pools(pools)
} }
pub fn len(&self) -> usize {
self.0.len()
}
pub fn get_mut(&mut self, rank: Rank) -> &mut Vec<Variable> { pub fn get_mut(&mut self, rank: Rank) -> &mut Vec<Variable> {
self.0 self.0
.get_mut(rank.into_usize()) .get_mut(rank.into_usize())
@ -96,8 +100,8 @@ fn solve(
match constraint { match constraint {
True => state, True => state,
Eq(typ, expected_type, _region) => { Eq(typ, expected_type, _region) => {
let actual = type_to_var(subs, typ.clone()); let actual = type_to_var(subs, rank, pools, typ.clone());
let expected = type_to_var(subs, expected_type.clone().get_type()); let expected = type_to_var(subs, rank, pools, expected_type.clone().get_type());
// TODO use region when reporting a problem // TODO use region when reporting a problem
let vars = unify(subs, problems, actual, expected); let vars = unify(subs, problems, actual, expected);
@ -107,20 +111,25 @@ fn solve(
state state
} }
Lookup(symbol, expected_type, _region) => { Lookup(symbol, expected_type, _region) => {
// TODO use region? let var = *vars_by_symbol.get(&dbg!(symbol)).unwrap_or_else(|| {
let actual = subs.copy_var(*vars_by_symbol.get(&symbol).unwrap_or_else(|| {
// TODO Instead of panicking, solve this as True and record // TODO Instead of panicking, solve this as True and record
// a Problem ("module Foo does not expose `bar`") for later. // a Problem ("module Foo does not expose `bar`") for later.
panic!( panic!(
"Could not find symbol {:?} in vars_by_symbol {:?}", "Could not find symbol {:?} in vars_by_symbol {:?}",
symbol, vars_by_symbol symbol, vars_by_symbol
) )
})); });
let expected = type_to_var(subs, expected_type.clone().get_type());
let vars = unify(subs, problems, actual, expected); let actual = var;
let expected = type_to_var(subs, rank, pools, expected_type.clone().get_type());
// TODO use region when reporting a problem
let vars = unify(subs, problems, dbg!(actual), dbg!(expected));
introduce(subs, rank, pools, &vars); introduce(subs, rank, pools, &vars);
dbg!(vars_by_symbol);
state state
} }
And(sub_constraints) => { And(sub_constraints) => {
@ -142,8 +151,8 @@ fn solve(
} }
Pattern(_region, _category, typ, expected) => { Pattern(_region, _category, typ, expected) => {
// TODO use region? // TODO use region?
let actual = type_to_var(subs, typ.clone()); let actual = type_to_var(subs, rank, pools, typ.clone());
let expected = type_to_var(subs, expected.clone().get_type()); let expected = type_to_var(subs, rank, pools, expected.clone().get_type());
let vars = unify(subs, problems, actual, expected); let vars = unify(subs, problems, actual, expected);
introduce(subs, rank, pools, &vars); introduce(subs, rank, pools, &vars);
@ -152,7 +161,7 @@ fn solve(
} }
Let(let_con) => { Let(let_con) => {
match &let_con.ret_constraint { match &let_con.ret_constraint {
True => { True if let_con.rigid_vars.is_empty() => {
introduce(subs, rank, pools, &let_con.flex_vars); introduce(subs, rank, pools, &let_con.flex_vars);
// If the return expression is guaranteed to solve, // If the return expression is guaranteed to solve,
@ -167,6 +176,54 @@ fn solve(
&let_con.defs_constraint, &let_con.defs_constraint,
) )
} }
ret_con if let_con.rigid_vars.is_empty() && let_con.flex_vars.is_empty() => {
let state = solve(
vars_by_symbol,
state,
rank,
pools,
problems,
subs,
&let_con.defs_constraint,
);
// Add a variable for each assignment to the vars_by_symbol.
let mut locals = ImMap::default();
for (symbol, loc_type) in let_con.def_types.iter() {
let var = type_to_var(subs, rank, pools, loc_type.value.clone());
locals.insert(
symbol.clone(),
Located {
value: var,
region: loc_type.region,
},
);
}
let mut new_vars_by_symbol = vars_by_symbol.clone();
for (symbol, loc_var) in locals.iter() {
if !new_vars_by_symbol.contains_key(&symbol) {
new_vars_by_symbol.insert(symbol.clone(), loc_var.value);
}
}
let new_state = solve(
&new_vars_by_symbol,
state,
rank,
pools,
problems,
subs,
ret_con,
);
for (symbol, loc_var) in locals {
check_for_infinite_type(subs, problems, symbol, loc_var);
}
new_state
}
ret_con => { ret_con => {
let rigid_vars = &let_con.rigid_vars; let rigid_vars = &let_con.rigid_vars;
let flex_vars = &let_con.flex_vars; let flex_vars = &let_con.flex_vars;
@ -174,8 +231,6 @@ fn solve(
// work in the next pool to localize header // work in the next pool to localize header
let next_rank = rank.next(); let next_rank = rank.next();
let mut next_pools = pools.clone();
// introduce variables // introduce variables
for &var in rigid_vars.iter() { for &var in rigid_vars.iter() {
subs.set_rank(var, next_rank); subs.set_rank(var, next_rank);
@ -185,94 +240,112 @@ fn solve(
subs.set_rank(var, next_rank); subs.set_rank(var, next_rank);
} }
let pool: &mut Vec<Variable> = next_pools.get_mut(next_rank); let work_in_next_pools = |next_pools: &mut Pools| {
let pool: &mut Vec<Variable> = next_pools.get_mut(next_rank);
pool.reserve(rigid_vars.len() + flex_vars.len()); pool.clear();
pool.extend(rigid_vars.iter()); pool.reserve(rigid_vars.len() + flex_vars.len());
pool.extend(flex_vars.iter()); pool.extend(rigid_vars.iter());
pool.extend(flex_vars.iter());
// Add a variable for each assignment to the vars_by_symbol. // Add a variable for each assignment to the vars_by_symbol.
let mut locals = ImMap::default(); let mut locals = ImMap::default();
for (symbol, loc_type) in let_con.def_types.iter() { for (symbol, loc_type) in let_con.def_types.iter() {
let var = type_to_var(subs, loc_type.value.clone()); let def_type = loc_type.value.clone();
let var = type_to_var(subs, next_rank, next_pools, def_type);
locals.insert( locals.insert(
symbol.clone(), symbol.clone(),
Located { Located {
value: var, value: var,
region: loc_type.region, region: loc_type.region,
}, },
);
}
// run solver in next pool
// Solve the assignments' constraints first.
let new_state = solve(
vars_by_symbol,
state,
next_rank,
next_pools,
problems,
subs,
&let_con.defs_constraint,
); );
} let young_mark = new_state.mark;
let visit_mark = young_mark.next();
let final_mark = visit_mark.next();
// run solver in next pool // pop pool
generalize(subs, young_mark, visit_mark, next_rank, next_pools);
// Solve the assignments' constraints first. next_pools.get_mut(next_rank).clear();
let new_state = solve(
vars_by_symbol,
state,
next_rank,
&mut next_pools,
problems,
subs,
&let_con.defs_constraint,
);
let young_mark = new_state.mark;
let visit_mark = young_mark.next();
let final_mark = visit_mark.next();
// pop pool // check that things went well
generalize(subs, young_mark, visit_mark, next_rank, &mut next_pools); debug_assert!(rigid_vars
.iter()
.all(|&var| subs.get_without_compacting(var).rank == Rank::none()));
next_pools.get_mut(next_rank).clear(); let mut new_vars_by_symbol = vars_by_symbol.clone();
// check that things went well for (symbol, loc_var) in locals.iter() {
debug_assert!(rigid_vars if !new_vars_by_symbol.contains_key(&symbol) {
.iter() new_vars_by_symbol.insert(symbol.clone(), loc_var.value);
.all(|&var| subs.get_without_compacting(var).rank == Rank::none())); }
}
let mut new_vars_by_symbol = vars_by_symbol.clone(); // Note that this vars_by_symbol is the one returned by the
// previous call to solve()
let temp_state = State {
vars_by_symbol: new_state.vars_by_symbol,
mark: final_mark,
};
for (symbol, loc_var) in locals.iter() { // Now solve the body, using the new vars_by_symbol which includes
new_vars_by_symbol.insert(symbol.clone(), loc_var.value); // the assignments' name-to-variable mappings.
} let new_state = solve(
&new_vars_by_symbol,
temp_state,
rank,
next_pools,
problems,
subs,
&ret_con,
);
// Note that this vars_by_symbol is the one returned by the for (symbol, loc_var) in locals {
// previous call to solve() check_for_infinite_type(subs, problems, symbol, loc_var);
let temp_state = State { }
vars_by_symbol: new_state.vars_by_symbol,
mark: final_mark, new_state
}; };
// Now solve the body, using the new vars_by_symbol which includes if next_rank.into_usize() < pools.len() {
// the assignments' name-to-variable mappings. work_in_next_pools(pools)
let new_state = solve( } else {
&new_vars_by_symbol, work_in_next_pools(&mut pools.clone())
temp_state,
rank,
&mut next_pools,
problems,
subs,
&ret_con,
);
for (symbol, loc_var) in locals {
check_for_infinite_type(subs, problems, symbol, loc_var);
} }
new_state
} }
} }
} }
} }
} }
fn type_to_var(subs: &mut Subs, typ: Type) -> Variable { fn type_to_var(subs: &mut Subs, rank: Rank, pools: &mut Pools, typ: Type) -> Variable {
type_to_variable(subs, &ImMap::default(), typ) type_to_variable(subs, rank, pools, &ImMap::default(), typ)
} }
fn type_to_variable(subs: &mut Subs, aliases: &ImMap<Lowercase, Variable>, typ: Type) -> Variable { fn type_to_variable(
subs: &mut Subs,
rank: Rank,
pools: &mut Pools,
aliases: &ImMap<Lowercase, Variable>,
typ: Type,
) -> Variable {
match typ { match typ {
Variable(var) => var, Variable(var) => var,
Apply { Apply {
@ -283,7 +356,7 @@ fn type_to_variable(subs: &mut Subs, aliases: &ImMap<Lowercase, Variable>, typ:
let mut arg_vars = Vec::with_capacity(args.len()); let mut arg_vars = Vec::with_capacity(args.len());
for arg in args { for arg in args {
arg_vars.push(type_to_variable(subs, aliases, arg.clone())) arg_vars.push(type_to_variable(subs, rank, pools, aliases, arg.clone()))
} }
let flat_type = FlatType::Apply { let flat_type = FlatType::Apply {
@ -293,57 +366,60 @@ fn type_to_variable(subs: &mut Subs, aliases: &ImMap<Lowercase, Variable>, typ:
}; };
let content = Content::Structure(flat_type); let content = Content::Structure(flat_type);
subs.fresh(Descriptor::from(content)) register(subs, rank, pools, content)
} }
EmptyRec => { EmptyRec => {
let content = Content::Structure(FlatType::EmptyRecord); let content = Content::Structure(FlatType::EmptyRecord);
subs.fresh(Descriptor::from(content)) register(subs, rank, pools, content)
} }
Function(args, ret_type) => { Function(args, ret_type) => {
let mut arg_vars = Vec::with_capacity(args.len()); let mut arg_vars = Vec::with_capacity(args.len());
for arg in args { for arg in args {
arg_vars.push(type_to_variable(subs, aliases, arg.clone())) arg_vars.push(type_to_variable(subs, rank, pools, aliases, arg.clone()))
} }
let ret_var = type_to_variable(subs, aliases, *ret_type); let ret_var = type_to_variable(subs, rank, pools, aliases, *ret_type);
let content = Content::Structure(FlatType::Func(arg_vars, ret_var)); let content = Content::Structure(FlatType::Func(arg_vars, ret_var));
subs.fresh(Descriptor::from(content)) register(subs, rank, pools, content)
} }
Record(fields, ext) => { Record(fields, ext) => {
let mut field_vars = ImMap::default(); let mut field_vars = ImMap::default();
for (field, field_type) in fields { for (field, field_type) in fields {
field_vars.insert(field, type_to_variable(subs, aliases, field_type)); field_vars.insert(
field,
type_to_variable(subs, rank, pools, aliases, field_type),
);
} }
let ext_var = type_to_variable(subs, aliases, *ext); let ext_var = type_to_variable(subs, rank, pools, aliases, *ext);
let content = Content::Structure(FlatType::Record(field_vars, ext_var)); let content = Content::Structure(FlatType::Record(field_vars, ext_var));
subs.fresh(Descriptor::from(content)) register(subs, rank, pools, content)
} }
Alias(home, name, args, alias_type) => { Alias(home, name, args, alias_type) => {
let mut arg_vars = Vec::with_capacity(args.len()); let mut arg_vars = Vec::with_capacity(args.len());
let mut new_aliases = ImMap::default(); let mut new_aliases = ImMap::default();
for (arg, arg_type) in args { for (arg, arg_type) in args {
let arg_var = type_to_variable(subs, aliases, arg_type.clone()); let arg_var = type_to_variable(subs, rank, pools, aliases, arg_type.clone());
arg_vars.push((arg.clone(), arg_var)); arg_vars.push((arg.clone(), arg_var));
new_aliases.insert(arg, arg_var); new_aliases.insert(arg, arg_var);
} }
let alias_var = type_to_variable(subs, &new_aliases, *alias_type); let alias_var = type_to_variable(subs, rank, pools, &new_aliases, *alias_type);
let content = Content::Alias(home, name, arg_vars, alias_var); let content = Content::Alias(home, name, arg_vars, alias_var);
subs.fresh(Descriptor::from(content)) register(subs, rank, pools, content)
} }
Erroneous(problem) => { Erroneous(problem) => {
let content = Content::Structure(FlatType::Erroneous(problem)); let content = Content::Structure(FlatType::Erroneous(problem));
subs.fresh(Descriptor::from(content)) register(subs, rank, pools, content)
} }
} }
} }
@ -554,3 +630,144 @@ fn introduce(subs: &mut Subs, rank: Rank, pools: &mut Pools, vars: &[Variable])
pool.extend(vars); pool.extend(vars);
} }
fn deep_copy_var(subs: &mut Subs, rank: Rank, pools: &mut Pools, var: Variable) -> Variable {
let copy = deep_copy_var_help(subs, rank, pools, var);
subs.restore(var);
copy
}
fn deep_copy_var_help(
subs: &mut Subs,
max_rank: Rank,
pools: &mut Pools,
var: Variable,
) -> Variable {
use crate::subs::Content::*;
use crate::subs::FlatType::*;
let desc = subs.get(var);
if let Some(copy) = desc.copy {
return copy;
} else if desc.rank != Rank::none() {
return var;
}
let make_descriptor = |content| Descriptor {
content,
rank: max_rank,
mark: Mark::none(),
copy: None,
};
let content = desc.content;
let copy = subs.fresh(make_descriptor(content.clone()));
pools.get_mut(max_rank).push(copy);
// Link the original variable to the new variable. This lets us
// avoid making multiple copies of the variable we are instantiating.
//
// Need to do this before recursively copying to avoid looping.
subs.set(
var,
Descriptor {
content: content.clone(),
rank: desc.rank,
mark: Mark::none(),
copy: Some(copy),
},
);
// Now we recursively copy the content of the variable.
// We have already marked the variable as copied, so we
// will not repeat this work or crawl this variable again.
match content {
Structure(flat_type) => {
let new_flat_type = match flat_type {
Apply {
module_name,
name,
args,
} => {
let args = args
.into_iter()
.map(|var| deep_copy_var_help(subs, max_rank, pools, var))
.collect();
Apply {
module_name,
name,
args,
}
}
Func(arg_vars, ret_var) => {
let new_ret_var = deep_copy_var_help(subs, max_rank, pools, ret_var);
let arg_vars = arg_vars
.into_iter()
.map(|var| deep_copy_var_help(subs, max_rank, pools, var))
.collect();
Func(arg_vars, new_ret_var)
}
same @ EmptyRecord | same @ Erroneous(_) => same,
Record(fields, ext_var) => {
let mut new_fields = ImMap::default();
for (label, var) in fields {
new_fields.insert(label, deep_copy_var_help(subs, max_rank, pools, var));
}
Record(
new_fields,
deep_copy_var_help(subs, max_rank, pools, ext_var),
)
}
};
subs.set(copy, make_descriptor(Structure(new_flat_type)));
copy
}
FlexVar(_) | Error(_) => copy,
RigidVar(name) => {
subs.set(copy, make_descriptor(FlexVar(Some(name))));
copy
}
Alias(module_name, name, args, real_type_var) => {
let new_args = args
.into_iter()
.map(|(name, var)| (name, deep_copy_var_help(subs, max_rank, pools, var)))
.collect();
let new_real_type_var = deep_copy_var_help(subs, max_rank, pools, real_type_var);
let new_content = Alias(module_name, name, new_args, new_real_type_var);
subs.set(copy, make_descriptor(new_content));
copy
}
}
}
fn register(subs: &mut Subs, rank: Rank, pools: &mut Pools, content: Content) -> Variable {
let var = subs.fresh(Descriptor {
content,
rank,
mark: Mark::none(),
copy: None,
});
pools.get_mut(rank).push(var);
var
}

View file

@ -215,12 +215,6 @@ impl Subs {
}); });
} }
pub fn copy_var(&mut self, var: Variable) -> Variable {
// TODO understand the purpose of using a "deep copy" approach here,
// and perform it if necessary. (Seems to be about setting maxRank?)
var
}
pub fn equivalent(&mut self, left: Variable, right: Variable) -> bool { pub fn equivalent(&mut self, left: Variable, right: Variable) -> bool {
self.utable.unioned(left, right) self.utable.unioned(left, right)
} }