diff --git a/hercules_opt/src/dce.rs b/hercules_opt/src/dce.rs index 14fd39899ff18756f4c7ab35897b0093ef807038..75268694fdccb23920cccf3cd60cb5ae6738ef0f 100644 --- a/hercules_opt/src/dce.rs +++ b/hercules_opt/src/dce.rs @@ -25,7 +25,7 @@ pub fn dce(editor: &mut FunctionEditor) { // If a node on the worklist has 0 users, delete it. Add its uses onto // the worklist. - if editor.users(work).len() == 0 { + if editor.get_users(work).len() == 0 { let uses = get_uses(&editor.func().nodes[work.idx()]); let success = editor.edit(|edit| edit.delete_node(work)); if success { diff --git a/hercules_opt/src/editor.rs b/hercules_opt/src/editor.rs index 1b182c5747a603ccc36ad03f0f5307833494ac0c..b9044542a71f51e1f3ad1327955237844e723eda 100644 --- a/hercules_opt/src/editor.rs +++ b/hercules_opt/src/editor.rs @@ -3,9 +3,11 @@ extern crate either; extern crate hercules_ir; extern crate itertools; +use std::cell::{Ref, RefCell}; use std::collections::{BTreeMap, HashMap, HashSet, VecDeque}; use std::iter::FromIterator; use std::mem::take; +use std::ops::Deref; use self::bitvec::prelude::*; use self::either::Either; @@ -26,12 +28,19 @@ pub type Edit = (HashSet<NodeID>, HashSet<NodeID>); /* * Helper object for editing Hercules functions in a trackable manner. Edits are * recorded in order to repair partitions and debug info. + * Edits must be made atomically, that is, only one `.edit` may be called at a time + * across all editors. */ #[derive(Debug)] pub struct FunctionEditor<'a> { // Wraps a mutable reference to a function. Doesn't provide access to this // reference directly, so that we can monitor edits. function: &'a mut Function, + // Keep a RefCell to (dynamic) constants and types to allow function changes + // to update these + constants: &'a RefCell<Vec<Constant>>, + dynamic_constants: &'a RefCell<Vec<DynamicConstant>>, + types: &'a RefCell<Vec<Type>>, // Most optimizations need def use info, so provide an iteratively updated // mutable version that's automatically updated based on recorded edits. mut_def_use: Vec<HashSet<NodeID>>, @@ -63,17 +72,28 @@ pub struct FunctionEdit<'a: 'b, 'b> { // Reference the active function editor. editor: &'b mut FunctionEditor<'a>, // Keep track of deleted node IDs. - deleted: HashSet<NodeID>, + deleted_nodeids: HashSet<NodeID>, // Keep track of added node IDs. - added: HashSet<NodeID>, + added_nodeids: HashSet<NodeID>, // Keep track of added and use updated nodes. - added_and_updated: BTreeMap<NodeID, Node>, + added_and_updated_nodes: BTreeMap<NodeID, Node>, + // Keep track of added (dynamic) constants and types + added_constants: Vec<Constant>, + added_dynamic_constants: Vec<DynamicConstant>, + added_types: Vec<Type>, // Compute a def-use map entries iteratively. updated_def_use: BTreeMap<NodeID, HashSet<NodeID>>, + updated_return_type: Option<TypeID>, } impl<'a: 'b, 'b> FunctionEditor<'a> { - pub fn new(function: &'a mut Function, def_use: &ImmutableDefUseMap) -> Self { + pub fn new( + function: &'a mut Function, + constants: &'a RefCell<Vec<Constant>>, + dynamic_constants: &'a RefCell<Vec<DynamicConstant>>, + types: &'a RefCell<Vec<Type>>, + def_use: &ImmutableDefUseMap, + ) -> Self { let mut_def_use = (0..function.nodes.len()) .map(|idx| { def_use @@ -87,6 +107,9 @@ impl<'a: 'b, 'b> FunctionEditor<'a> { FunctionEditor { function, + constants, + dynamic_constants, + types, mut_def_use, edits: vec![], mutable_nodes, @@ -100,10 +123,14 @@ impl<'a: 'b, 'b> FunctionEditor<'a> { // Create the edit helper struct and perform the edit using it. let edit_obj = FunctionEdit { editor: self, - deleted: HashSet::new(), - added: HashSet::new(), - added_and_updated: BTreeMap::new(), + deleted_nodeids: HashSet::new(), + added_nodeids: HashSet::new(), + added_constants: Vec::new().into(), + added_dynamic_constants: Vec::new().into(), + added_types: Vec::new().into(), + added_and_updated_nodes: BTreeMap::new(), updated_def_use: BTreeMap::new(), + updated_return_type: None, }; if let Ok(populated_edit) = edit(edit_obj) { @@ -111,10 +138,14 @@ impl<'a: 'b, 'b> FunctionEditor<'a> { // without modifying immutable nodes. let FunctionEdit { editor, - deleted, - added, - added_and_updated, + deleted_nodeids, + added_nodeids, + added_constants, + added_dynamic_constants, + added_types, + added_and_updated_nodes: added_and_updated, updated_def_use, + updated_return_type, } = populated_edit; // Step 1: update the mutable def use map. for (u, new_users) in updated_def_use { @@ -148,14 +179,14 @@ impl<'a: 'b, 'b> FunctionEditor<'a> { // Step 3: delete nodes. This is done using "gravestones", where a // node other than node ID 0 being a start node is considered a // gravestone. - for id in deleted.iter() { + for id in deleted_nodeids.iter() { // Check that there are no users of deleted nodes. assert!(editor.mut_def_use[id.idx()].is_empty()); editor.function.nodes[id.idx()] = Node::Start; } // Step 4: add a single edit to the edit list. - editor.edits.push((deleted, added)); + editor.edits.push((deleted_nodeids, added_nodeids)); // Step 5: update the length of mutable_nodes. All added nodes are // mutable. @@ -163,6 +194,20 @@ impl<'a: 'b, 'b> FunctionEditor<'a> { .mutable_nodes .resize(editor.function.nodes.len(), true); + // Step 6: update types and constants + let mut editor_constants = editor.constants.borrow_mut(); + let mut editor_dynamic_constants = editor.dynamic_constants.borrow_mut(); + let mut editor_types = editor.types.borrow_mut(); + + editor_constants.extend(added_constants); + editor_dynamic_constants.extend(added_dynamic_constants); + editor_types.extend(added_types); + + // Step 7: update return type if necessary + if let Some(return_type) = updated_return_type { + editor.function.return_type = return_type; + } + true } else { false @@ -173,7 +218,7 @@ impl<'a: 'b, 'b> FunctionEditor<'a> { &self.function } - pub fn users(&self, id: NodeID) -> impl ExactSizeIterator<Item = NodeID> + '_ { + pub fn get_users(&self, id: NodeID) -> impl ExactSizeIterator<Item = NodeID> + '_ { self.mut_def_use[id.idx()].iter().map(|x| *x) } @@ -200,7 +245,7 @@ impl<'a, 'b> FunctionEdit<'a, 'b> { } pub fn add_node(&mut self, node: Node) -> NodeID { - let id = NodeID::new(self.editor.function.nodes.len() + self.added.len()); + let id = NodeID::new(self.editor.function.nodes.len() + self.added_nodeids.len()); // Added nodes need to have an entry in the def-use map. self.updated_def_use.insert(id, HashSet::new()); // Added nodes use other nodes, and we need to update their def-use @@ -210,8 +255,8 @@ impl<'a, 'b> FunctionEdit<'a, 'b> { self.updated_def_use.get_mut(u).unwrap().insert(id); } // Add the node. - self.added_and_updated.insert(id, node); - self.added.insert(id); + self.added_and_updated_nodes.insert(id, node); + self.added_nodeids.insert(id); id } @@ -220,7 +265,7 @@ impl<'a, 'b> FunctionEdit<'a, 'b> { // immutable node, as it means the whole edit should be aborted. if self.editor.mutable_nodes[id.idx()] { assert!( - !self.added.contains(&id), + !self.added_nodeids.contains(&id), "PANIC: Please don't delete a node that was added in the same edit." ); // Deleted nodes use other nodes, and we need to update their def- @@ -232,7 +277,7 @@ impl<'a, 'b> FunctionEdit<'a, 'b> { self.ensure_updated_def_use_entry(u); self.updated_def_use.get_mut(&u).unwrap().remove(&id); } - self.deleted.insert(id); + self.deleted_nodeids.insert(id); Ok(self) } else { Err(self) @@ -248,14 +293,14 @@ impl<'a, 'b> FunctionEdit<'a, 'b> { self.ensure_updated_def_use_entry(old); for user_id in self.updated_def_use[&old].iter() { // Replace uses of old with new. - let mut updated_user = self.node(*user_id).clone(); + let mut updated_user = self.get_node(*user_id).clone(); for u in get_uses_mut(&mut updated_user).as_mut() { if **u == old { **u = new; } } // Add the updated user to added_and_updated. - self.added_and_updated.insert(*user_id, updated_user); + self.added_and_updated_nodes.insert(*user_id, updated_user); } // All of the users of the old node become users of the new node, so @@ -273,9 +318,9 @@ impl<'a, 'b> FunctionEdit<'a, 'b> { } } - pub fn node(&self, id: NodeID) -> &Node { - assert!(!self.deleted.contains(&id)); - if let Some(node) = self.added_and_updated.get(&id) { + pub fn get_node(&self, id: NodeID) -> &Node { + assert!(!self.deleted_nodeids.contains(&id)); + if let Some(node) = self.added_and_updated_nodes.get(&id) { // Refer to added or updated node. This node is guaranteed to be // updated with uses after replace_all_uses is called. node @@ -285,8 +330,8 @@ impl<'a, 'b> FunctionEdit<'a, 'b> { } } - pub fn users(&self, id: NodeID) -> impl Iterator<Item = NodeID> + '_ { - assert!(!self.deleted.contains(&id)); + pub fn get_users(&self, id: NodeID) -> impl Iterator<Item = NodeID> + '_ { + assert!(!self.deleted_nodeids.contains(&id)); if let Some(users) = self.updated_def_use.get(&id) { // Refer to the updated users set. Either::Left(users.iter().map(|x| *x)) @@ -295,6 +340,107 @@ impl<'a, 'b> FunctionEdit<'a, 'b> { Either::Right(self.editor.mut_def_use[id.idx()].iter().map(|x| *x)) } } + + pub fn add_type(&mut self, ty: Type) -> TypeID { + let pos = self + .editor + .types + .borrow() + .iter() + .chain(self.added_types.iter()) + .position(|t| *t == ty); + if let Some(idx) = pos { + TypeID::new(idx) + } else { + let id = TypeID::new(self.editor.types.borrow().len() + self.added_types.len()); + self.added_types.push(ty); + id + } + } + + pub fn get_type(&self, id: TypeID) -> impl Deref + '_ { + if id.idx() < self.editor.types.borrow().len() { + Either::Left(Ref::map(self.editor.types.borrow(), |types| { + &types[id.idx()] + })) + } else { + Either::Right( + self.added_types + .get(id.idx() - self.editor.types.borrow().len()) + .unwrap(), + ) + } + } + + pub fn add_constant(&mut self, constant: Constant) -> ConstantID { + let pos = self + .editor + .constants + .borrow() + .iter() + .chain(self.added_constants.iter()) + .position(|c| *c == constant); + if let Some(idx) = pos { + ConstantID::new(idx) + } else { + let id = + ConstantID::new(self.editor.constants.borrow().len() + self.added_constants.len()); + self.added_constants.push(constant); + id + } + } + + pub fn get_constant(&self, id: ConstantID) -> impl Deref + '_ { + if id.idx() < self.editor.constants.borrow().len() { + Either::Left(Ref::map(self.editor.constants.borrow(), |constants| { + &constants[id.idx()] + })) + } else { + Either::Right( + self.added_constants + .get(id.idx() - self.editor.constants.borrow().len()) + .unwrap(), + ) + } + } + + pub fn add_dynamic_constant(&mut self, dynamic_constant: DynamicConstant) -> DynamicConstantID { + let pos = self + .editor + .dynamic_constants + .borrow() + .iter() + .chain(self.added_dynamic_constants.iter()) + .position(|c| *c == dynamic_constant); + if let Some(idx) = pos { + DynamicConstantID::new(idx) + } else { + let id = DynamicConstantID::new( + self.editor.dynamic_constants.borrow().len() + self.added_dynamic_constants.len(), + ); + self.added_dynamic_constants.push(dynamic_constant); + id + } + } + + pub fn get_dynamic_constant(&self, id: DynamicConstantID) -> impl Deref + '_ { + if id.idx() < self.editor.dynamic_constants.borrow().len() { + Either::Left(Ref::map( + self.editor.dynamic_constants.borrow(), + |dynamic_constants| &dynamic_constants[id.idx()], + )) + } else { + Either::Right( + self.added_dynamic_constants + .get(id.idx() - self.editor.dynamic_constants.borrow().len()) + .unwrap(), + ) + } + } + + pub fn set_return_type(&mut self, ty: TypeID) { + self.updated_return_type = Some(ty); + } } /* @@ -606,8 +752,17 @@ fn func(x: i32) -> i32 .next() .unwrap(); + let constants_ref = RefCell::new(src_module.constants); + let dynamic_constants_ref = RefCell::new(src_module.dynamic_constants); + let types_ref = RefCell::new(src_module.types); // Edit the function by replacing the add with a multiply. - let mut editor = FunctionEditor::new(func, &def_use(func)); + let mut editor = FunctionEditor::new( + func, + &constants_ref, + &dynamic_constants_ref, + &types_ref, + &def_use(func), + ); let success = editor.edit(|mut edit| { let mul = edit.add_node(Node::Binary { op: BinaryOperator::Mul, diff --git a/hercules_opt/src/gvn.rs b/hercules_opt/src/gvn.rs index e1a179f7986ad7134b27b1e0d87e4947faa44fa6..e3b2fa6084ed0f8aee922ff9ffe1e0ec161bdc3f 100644 --- a/hercules_opt/src/gvn.rs +++ b/hercules_opt/src/gvn.rs @@ -28,7 +28,7 @@ pub fn gvn(editor: &mut FunctionEditor, constants: &Vec<Constant>) { } // Record the users of `work` before making any edits. - let work_users: Vec<NodeID> = editor.users(work).collect(); + let work_users: Vec<NodeID> = editor.get_users(work).collect(); // At this point, we know the number of the node IDed `work` is // `number`. We want to replace `work` with `number`, which means diff --git a/hercules_opt/src/pass.rs b/hercules_opt/src/pass.rs index f27f1f61df6ac80eda8c79fd6d5fb47580013d77..bf4b7898e1b39ba49169eb379cfa0ed9845af117 100644 --- a/hercules_opt/src/pass.rs +++ b/hercules_opt/src/pass.rs @@ -4,6 +4,7 @@ extern crate postcard; extern crate serde; extern crate take_mut; +use std::cell::RefCell; use std::collections::HashMap; use std::env::temp_dir; use std::fs::File; @@ -330,10 +331,24 @@ impl PassManager { self.make_def_uses(); let def_uses = self.def_uses.as_ref().unwrap(); for idx in 0..self.module.functions.len() { - let mut editor = - FunctionEditor::new(&mut self.module.functions[idx], &def_uses[idx]); + let constants_ref = + RefCell::new(std::mem::take(&mut self.module.constants)); + let dynamic_constants_ref = + RefCell::new(std::mem::take(&mut self.module.dynamic_constants)); + let types_ref = RefCell::new(std::mem::take(&mut self.module.types)); + let mut editor = FunctionEditor::new( + &mut self.module.functions[idx], + &constants_ref, + &dynamic_constants_ref, + &types_ref, + &def_uses[idx], + ); dce(&mut editor); + self.module.constants = constants_ref.take(); + self.module.dynamic_constants = dynamic_constants_ref.take(); + self.module.types = types_ref.take(); + let edits = &editor.edits(); if let Some(plans) = self.plans.as_mut() { repair_plan(&mut plans[idx], &self.module.functions[idx], edits); @@ -365,10 +380,24 @@ impl PassManager { self.make_def_uses(); let def_uses = self.def_uses.as_ref().unwrap(); for idx in 0..self.module.functions.len() { - let mut editor = - FunctionEditor::new(&mut self.module.functions[idx], &def_uses[idx]); + let constants_ref = + RefCell::new(std::mem::take(&mut self.module.constants)); + let dynamic_constants_ref = + RefCell::new(std::mem::take(&mut self.module.dynamic_constants)); + let types_ref = RefCell::new(std::mem::take(&mut self.module.types)); + let mut editor = FunctionEditor::new( + &mut self.module.functions[idx], + &constants_ref, + &dynamic_constants_ref, + &types_ref, + &def_uses[idx], + ); gvn(&mut editor, &self.module.constants); + self.module.constants = constants_ref.take(); + self.module.dynamic_constants = dynamic_constants_ref.take(); + self.module.types = types_ref.take(); + let edits = &editor.edits(); if let Some(plans) = self.plans.as_mut() { repair_plan(&mut plans[idx], &self.module.functions[idx], edits);