Support treeification of adjacent side-effecting ops in some cases.
This commit is contained in:
parent
908ad937e1
commit
c960d728bf
|
@ -41,50 +41,57 @@ impl Trees {
|
||||||
let mut remat = HashSet::default();
|
let mut remat = HashSet::default();
|
||||||
let mut multi_use = HashSet::default();
|
let mut multi_use = HashSet::default();
|
||||||
|
|
||||||
for (value, def) in body.values.entries() {
|
for block_def in body.blocks.values() {
|
||||||
match def {
|
let mut last_non_pure = None;
|
||||||
&ValueDef::Operator(op, args, _) => {
|
for &value in &block_def.insts {
|
||||||
// Ignore operators with invalid args: these must
|
match &body.values[value] {
|
||||||
// always be unreachable.
|
&ValueDef::Operator(op, args, _) => {
|
||||||
if body.arg_pool[args].iter().any(|arg| arg.is_invalid()) {
|
// Ignore operators with invalid args: these must
|
||||||
continue;
|
// always be unreachable.
|
||||||
}
|
if body.arg_pool[args].iter().any(|arg| arg.is_invalid()) {
|
||||||
// If this is an always-rematerialized operator,
|
|
||||||
// mark it as such and continue.
|
|
||||||
if is_remat(&op) {
|
|
||||||
remat.insert(value);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
// For each of the args, if the value is produced
|
|
||||||
// by a single-output op and is movable, and is
|
|
||||||
// not already recorded in `multi_use`, place it
|
|
||||||
// in the arg slot. Otherwise if owned already
|
|
||||||
// somewhere else, undo that and put in
|
|
||||||
// `multi_use`.
|
|
||||||
for (i, &arg) in body.arg_pool[args].iter().enumerate() {
|
|
||||||
let arg = body.resolve_alias(arg);
|
|
||||||
if multi_use.contains(&arg) {
|
|
||||||
continue;
|
continue;
|
||||||
} else if let Some(old_owner) = owner.remove(&arg) {
|
}
|
||||||
owned.remove(&old_owner);
|
// If this is an always-rematerialized operator,
|
||||||
multi_use.insert(arg);
|
// mark it as such and continue.
|
||||||
} else if Self::is_movable(body, arg) {
|
if is_remat(&op) {
|
||||||
let pos = u16::try_from(i).unwrap();
|
remat.insert(value);
|
||||||
let value_arg = ValueArg(value, pos);
|
continue;
|
||||||
owner.insert(arg, value_arg);
|
}
|
||||||
owned.insert(value_arg, arg);
|
|
||||||
|
// For each of the args, if the value is produced
|
||||||
|
// by a single-output op and is movable, and is
|
||||||
|
// not already recorded in `multi_use`, place it
|
||||||
|
// in the arg slot. Otherwise if owned already
|
||||||
|
// somewhere else, undo that and put in
|
||||||
|
// `multi_use`.
|
||||||
|
for (i, &arg) in body.arg_pool[args].iter().enumerate() {
|
||||||
|
let arg = body.resolve_alias(arg);
|
||||||
|
if multi_use.contains(&arg) {
|
||||||
|
continue;
|
||||||
|
} else if let Some(old_owner) = owner.remove(&arg) {
|
||||||
|
owned.remove(&old_owner);
|
||||||
|
multi_use.insert(arg);
|
||||||
|
} else if Self::is_movable(body, arg) || Some(arg) == last_non_pure {
|
||||||
|
let pos = u16::try_from(i).unwrap();
|
||||||
|
let value_arg = ValueArg(value, pos);
|
||||||
|
owner.insert(arg, value_arg);
|
||||||
|
owned.insert(value_arg, arg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if !op.is_pure() {
|
||||||
|
last_non_pure = Some(value);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
&ValueDef::PickOutput(..) => {
|
||||||
|
// Can ignore use: multi-arity values are never treeified.
|
||||||
|
}
|
||||||
|
&ValueDef::BlockParam(..)
|
||||||
|
| &ValueDef::Alias(..)
|
||||||
|
| &ValueDef::Placeholder(..)
|
||||||
|
| &ValueDef::Trace(..)
|
||||||
|
| &ValueDef::None => {}
|
||||||
}
|
}
|
||||||
&ValueDef::PickOutput(..) => {
|
|
||||||
// Can ignore use: multi-arity values are never treeified.
|
|
||||||
}
|
|
||||||
&ValueDef::BlockParam(..)
|
|
||||||
| &ValueDef::Alias(..)
|
|
||||||
| &ValueDef::Placeholder(..)
|
|
||||||
| &ValueDef::Trace(..)
|
|
||||||
| &ValueDef::None => {}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
for block in body.blocks.values() {
|
for block in body.blocks.values() {
|
||||||
|
|
Loading…
Reference in a new issue