Support treeification of adjacent side-effecting ops in some cases.

This commit is contained in:
Chris Fallin 2024-05-07 15:22:18 -07:00 committed by Graham Kelly
parent 908ad937e1
commit c960d728bf

View file

@ -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() {