fuzzbug fixes in stackifier

This commit is contained in:
Chris Fallin 2021-12-15 19:15:08 -08:00
parent ba80cea24b
commit 250e922498

View file

@ -33,27 +33,6 @@ enum Region {
// Backward region records to BackwardDispatch. // Backward region records to BackwardDispatch.
} }
#[derive(Clone, Copy, Debug, PartialEq, Eq, PartialOrd, Ord)]
struct RegionEndpoint {
block: OrderedBlockId,
pt: BlockPoint,
}
impl RegionEndpoint {
fn start(block: OrderedBlockId) -> Self {
RegionEndpoint {
block,
pt: BlockPoint::Start,
}
}
fn end(block: OrderedBlockId) -> Self {
RegionEndpoint {
block,
pt: BlockPoint::Start,
}
}
}
#[derive(Clone, Copy, Debug, PartialEq, Eq, PartialOrd, Ord)] #[derive(Clone, Copy, Debug, PartialEq, Eq, PartialOrd, Ord)]
enum BlockPoint { enum BlockPoint {
Start, Start,
@ -71,7 +50,7 @@ impl Shape {
log::trace!("get_one_shape: start {} regions {:?}", start, regions); log::trace!("get_one_shape: start {} regions {:?}", start, regions);
if start >= order.len() { if start >= order.len() {
None None
} else if regions.is_empty() || start < regions[0].start().block { } else if regions.is_empty() || start < regions[0].start() {
log::trace!(" -> leaf"); log::trace!(" -> leaf");
Some(( Some((
Shape::Leaf { Shape::Leaf {
@ -81,14 +60,14 @@ impl Shape {
&regions, &regions,
)) ))
} else { } else {
assert_eq!(start, regions[0].start().block); assert_eq!(start, regions[0].start());
let end = regions[0].end(); let end = regions[0].end();
let region_end = regions let region_end = regions
.iter() .iter()
.position(|region| region.start() >= end) .position(|region| region.start() >= end)
.unwrap_or(regions.len()); .unwrap_or(regions.len());
let subregions = &regions[1..region_end]; let subregions = &regions[1..region_end];
let (children, next_start) = Self::get_shapes(start, end.block, order, subregions); let (children, next_start) = Self::get_shapes(start, end, order, subregions);
let shape = if let Region::Forward(..) = &regions[0] { let shape = if let Region::Forward(..) = &regions[0] {
Shape::Block { Shape::Block {
head: order[start], head: order[start],
@ -133,15 +112,15 @@ impl Shape {
} }
impl Region { impl Region {
fn start(&self) -> RegionEndpoint { fn start(&self) -> OrderedBlockId {
match self { match self {
&Region::Forward(a, _) | &Region::Backward(a, _) => RegionEndpoint::end(a), &Region::Forward(a, _) | &Region::Backward(a, _) => a,
} }
} }
fn end(&self) -> RegionEndpoint { fn end(&self) -> OrderedBlockId {
match self { match self {
&Region::Forward(_, b) | &Region::Backward(_, b) => RegionEndpoint::start(b), &Region::Forward(_, b) | &Region::Backward(_, b) => b,
} }
} }
@ -149,7 +128,7 @@ impl Region {
self.start() <= other.start() && self.end() >= other.end() self.start() <= other.start() && self.end() >= other.end()
} }
fn contains_endpoint(&self, pt: RegionEndpoint) -> bool { fn contains_endpoint(&self, pt: OrderedBlockId) -> bool {
self.start() <= pt && pt <= self.end() self.start() <= pt && pt <= self.end()
} }
@ -201,6 +180,8 @@ impl Shape {
if succ_pos < block_pos { if succ_pos < block_pos {
let end = loop_header_to_end[succ_pos].unwrap_or(block_pos); let end = loop_header_to_end[succ_pos].unwrap_or(block_pos);
let end = std::cmp::max(end, block_pos); let end = std::cmp::max(end, block_pos);
log::trace!("loop backedge: header RPO {} latch RPO {} (header block {} latch block {})",
succ_pos, block_pos, order[succ_pos], order[block_pos]);
loop_header_to_end[succ_pos] = Some(end); loop_header_to_end[succ_pos] = Some(end);
} else if succ_pos > block_pos + 1 { } else if succ_pos > block_pos + 1 {
forward_edges.push((block_pos, succ_pos)); forward_edges.push((block_pos, succ_pos));
@ -306,16 +287,21 @@ impl Shape {
log::trace!("block_starts = {:?}", block_starts); log::trace!("block_starts = {:?}", block_starts);
log::trace!("block_end_to_start = {:?}", block_end_to_start); log::trace!("block_end_to_start = {:?}", block_end_to_start);
// We can't have a loop header at block 0; otherwise there is
// no way to express a forward edge outside the outermost loop
// nest.
assert!(loop_header_to_end[0].is_none());
// Now generate the region list which we use to produce the "shape". // Now generate the region list which we use to produce the "shape".
let mut regions = vec![]; let mut regions = vec![];
for block in 0..order.len() { for block in 0..order.len() {
for &block_end in block_starts[block].iter().rev() {
regions.push(Region::Forward(block, block_end));
}
if let Some(loop_end) = loop_header_to_end[block] { if let Some(loop_end) = loop_header_to_end[block] {
regions.push(Region::Backward(block, loop_end)); regions.push(Region::Backward(block, loop_end));
} }
for &block_end in block_starts[block].iter().rev() {
regions.push(Region::Forward(block, block_end));
}
} }
log::trace!("after stackifying: {:?}", regions); log::trace!("after stackifying: {:?}", regions);