`Results` contains and `Analysis` and an `EntryStates`. The unfortunate thing about this is that the analysis needs to be mutable everywhere (`&mut Analysis`) which forces the `Results` to be mutable everywhere, even though `EntryStates` is immutable everywhere. To fix this, this commit renames `Results` as `AnalysisAndResults`, renames `EntryStates` as `Results`, and separates the analysis and results as much as possible. (`AnalysisAndResults` doesn't get much use, it's mostly there to facilitate method chaining of `iterate_to_fixpoint`.) `Results` is immutable everywhere, which: - is a bit clearer on how the data is used, - avoids an unnecessary clone of entry states in `locals_live_across_suspend_points`, and - moves the results outside the `RefCell` in Formatter. The commit also reformulates `ResultsHandle` as the generic `CowMut`, which is simpler than `ResultsHandle` because it doesn't need the `'tcx` lifetime and the trait bounds. It also which sits nicely alongside the new use of `Cow` in `ResultsCursor`.
270 lines
9.4 KiB
Rust
270 lines
9.4 KiB
Rust
//! Random access inspection of the results of a dataflow analysis.
|
|
|
|
use std::borrow::Cow;
|
|
use std::cmp::Ordering;
|
|
use std::ops::{Deref, DerefMut};
|
|
|
|
#[cfg(debug_assertions)]
|
|
use rustc_index::bit_set::DenseBitSet;
|
|
use rustc_middle::mir::{self, BasicBlock, Location};
|
|
|
|
use super::{Analysis, Direction, Effect, EffectIndex, Results};
|
|
|
|
/// Some `ResultsCursor`s want to own an `Analysis`, and some want to borrow an `Analysis`, either
|
|
/// mutable or immutably. This type allows all of the above. It's similar to `Cow`, but `Cow`
|
|
/// doesn't allow mutable borrowing.
|
|
enum CowMut<'a, T> {
|
|
BorrowedMut(&'a mut T),
|
|
Owned(T),
|
|
}
|
|
|
|
impl<T> Deref for CowMut<'_, T> {
|
|
type Target = T;
|
|
|
|
fn deref(&self) -> &T {
|
|
match self {
|
|
CowMut::BorrowedMut(borrowed) => borrowed,
|
|
CowMut::Owned(owned) => owned,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<T> DerefMut for CowMut<'_, T> {
|
|
fn deref_mut(&mut self) -> &mut T {
|
|
match self {
|
|
CowMut::BorrowedMut(borrowed) => borrowed,
|
|
CowMut::Owned(owned) => owned,
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Allows random access inspection of the results of a dataflow analysis. Use this when you want
|
|
/// to inspect domain values only in certain locations; use `ResultsVisitor` if you want to inspect
|
|
/// domain values in many or all locations.
|
|
///
|
|
/// Because `Results` only has domain values for the entry of each basic block, these inspections
|
|
/// involve some amount of domain value recomputations. This cursor only has linear performance
|
|
/// within a basic block when its statements are visited in the same order as the `DIRECTION` of
|
|
/// the analysis. In the worst case—when statements are visited in *reverse* order—performance will
|
|
/// be quadratic in the number of statements in the block. The order in which basic blocks are
|
|
/// inspected has no impact on performance.
|
|
pub struct ResultsCursor<'mir, 'tcx, A>
|
|
where
|
|
A: Analysis<'tcx>,
|
|
{
|
|
body: &'mir mir::Body<'tcx>,
|
|
analysis: CowMut<'mir, A>,
|
|
results: Cow<'mir, Results<A::Domain>>,
|
|
state: A::Domain,
|
|
|
|
pos: CursorPosition,
|
|
|
|
/// Indicates that `state` has been modified with a custom effect.
|
|
///
|
|
/// When this flag is set, we need to reset to an entry set before doing a seek.
|
|
state_needs_reset: bool,
|
|
|
|
#[cfg(debug_assertions)]
|
|
reachable_blocks: DenseBitSet<BasicBlock>,
|
|
}
|
|
|
|
impl<'mir, 'tcx, A> ResultsCursor<'mir, 'tcx, A>
|
|
where
|
|
A: Analysis<'tcx>,
|
|
{
|
|
/// Returns the dataflow state at the current location.
|
|
pub fn get(&self) -> &A::Domain {
|
|
&self.state
|
|
}
|
|
|
|
/// Returns the body this analysis was run on.
|
|
pub fn body(&self) -> &'mir mir::Body<'tcx> {
|
|
self.body
|
|
}
|
|
|
|
fn new(
|
|
body: &'mir mir::Body<'tcx>,
|
|
analysis: CowMut<'mir, A>,
|
|
results: Cow<'mir, Results<A::Domain>>,
|
|
) -> Self {
|
|
let bottom_value = analysis.bottom_value(body);
|
|
ResultsCursor {
|
|
body,
|
|
analysis,
|
|
results,
|
|
|
|
// Initialize to the `bottom_value` and set `state_needs_reset` to tell the cursor that
|
|
// it needs to reset to block entry before the first seek. The cursor position is
|
|
// immaterial.
|
|
state_needs_reset: true,
|
|
state: bottom_value,
|
|
pos: CursorPosition::block_entry(mir::START_BLOCK),
|
|
|
|
#[cfg(debug_assertions)]
|
|
reachable_blocks: mir::traversal::reachable_as_bitset(body),
|
|
}
|
|
}
|
|
|
|
/// Returns a new cursor that takes ownership of and inspects analysis results.
|
|
pub fn new_owning(
|
|
body: &'mir mir::Body<'tcx>,
|
|
analysis: A,
|
|
results: Results<A::Domain>,
|
|
) -> Self {
|
|
Self::new(body, CowMut::Owned(analysis), Cow::Owned(results))
|
|
}
|
|
|
|
/// Returns a new cursor that borrows and inspects analysis results.
|
|
pub fn new_borrowing(
|
|
body: &'mir mir::Body<'tcx>,
|
|
analysis: &'mir mut A,
|
|
results: &'mir Results<A::Domain>,
|
|
) -> Self {
|
|
Self::new(body, CowMut::BorrowedMut(analysis), Cow::Borrowed(results))
|
|
}
|
|
|
|
/// Allows inspection of unreachable basic blocks even with `debug_assertions` enabled.
|
|
#[cfg(test)]
|
|
pub(crate) fn allow_unreachable(&mut self) {
|
|
#[cfg(debug_assertions)]
|
|
self.reachable_blocks.insert_all()
|
|
}
|
|
|
|
/// Returns the `Analysis` used to generate the underlying `Results`.
|
|
pub fn analysis(&self) -> &A {
|
|
&self.analysis
|
|
}
|
|
|
|
/// Resets the cursor to hold the entry set for the given basic block.
|
|
///
|
|
/// For forward dataflow analyses, this is the dataflow state prior to the first statement.
|
|
///
|
|
/// For backward dataflow analyses, this is the dataflow state after the terminator.
|
|
pub(super) fn seek_to_block_entry(&mut self, block: BasicBlock) {
|
|
#[cfg(debug_assertions)]
|
|
assert!(self.reachable_blocks.contains(block));
|
|
|
|
self.state.clone_from(&self.results[block]);
|
|
self.pos = CursorPosition::block_entry(block);
|
|
self.state_needs_reset = false;
|
|
}
|
|
|
|
/// Resets the cursor to hold the state prior to the first statement in a basic block.
|
|
///
|
|
/// For forward analyses, this is the entry set for the given block.
|
|
///
|
|
/// For backward analyses, this is the state that will be propagated to its
|
|
/// predecessors (ignoring edge-specific effects).
|
|
pub fn seek_to_block_start(&mut self, block: BasicBlock) {
|
|
if A::Direction::IS_FORWARD {
|
|
self.seek_to_block_entry(block)
|
|
} else {
|
|
self.seek_after(Location { block, statement_index: 0 }, Effect::Primary)
|
|
}
|
|
}
|
|
|
|
/// Resets the cursor to hold the state after the terminator in a basic block.
|
|
///
|
|
/// For backward analyses, this is the entry set for the given block.
|
|
///
|
|
/// For forward analyses, this is the state that will be propagated to its
|
|
/// successors (ignoring edge-specific effects).
|
|
pub fn seek_to_block_end(&mut self, block: BasicBlock) {
|
|
if A::Direction::IS_BACKWARD {
|
|
self.seek_to_block_entry(block)
|
|
} else {
|
|
self.seek_after(self.body.terminator_loc(block), Effect::Primary)
|
|
}
|
|
}
|
|
|
|
/// Advances the cursor to hold the dataflow state at `target` before its "primary" effect is
|
|
/// applied.
|
|
///
|
|
/// The "early" effect at the target location *will be* applied.
|
|
pub fn seek_before_primary_effect(&mut self, target: Location) {
|
|
self.seek_after(target, Effect::Early)
|
|
}
|
|
|
|
/// Advances the cursor to hold the dataflow state at `target` after its "primary" effect is
|
|
/// applied.
|
|
///
|
|
/// The "early" effect at the target location will be applied as well.
|
|
pub fn seek_after_primary_effect(&mut self, target: Location) {
|
|
self.seek_after(target, Effect::Primary)
|
|
}
|
|
|
|
fn seek_after(&mut self, target: Location, effect: Effect) {
|
|
assert!(target <= self.body.terminator_loc(target.block));
|
|
|
|
// Reset to the entry of the target block if any of the following are true:
|
|
// - A custom effect has been applied to the cursor state.
|
|
// - We are in a different block than the target.
|
|
// - We are in the same block but have advanced past the target effect.
|
|
if self.state_needs_reset || self.pos.block != target.block {
|
|
self.seek_to_block_entry(target.block);
|
|
} else if let Some(curr_effect) = self.pos.curr_effect_index {
|
|
let mut ord = curr_effect.statement_index.cmp(&target.statement_index);
|
|
if A::Direction::IS_BACKWARD {
|
|
ord = ord.reverse()
|
|
}
|
|
|
|
match ord.then_with(|| curr_effect.effect.cmp(&effect)) {
|
|
Ordering::Equal => return,
|
|
Ordering::Greater => self.seek_to_block_entry(target.block),
|
|
Ordering::Less => {}
|
|
}
|
|
}
|
|
|
|
// At this point, the cursor is in the same block as the target location at an earlier
|
|
// statement.
|
|
debug_assert_eq!(target.block, self.pos.block);
|
|
|
|
let block_data = &self.body[target.block];
|
|
#[rustfmt::skip]
|
|
let next_effect = if A::Direction::IS_FORWARD {
|
|
self.pos.curr_effect_index.map_or_else(
|
|
|| Effect::Early.at_index(0),
|
|
EffectIndex::next_in_forward_order,
|
|
)
|
|
} else {
|
|
self.pos.curr_effect_index.map_or_else(
|
|
|| Effect::Early.at_index(block_data.statements.len()),
|
|
EffectIndex::next_in_backward_order,
|
|
)
|
|
};
|
|
|
|
let target_effect_index = effect.at_index(target.statement_index);
|
|
|
|
A::Direction::apply_effects_in_range(
|
|
&mut *self.analysis,
|
|
&mut self.state,
|
|
target.block,
|
|
block_data,
|
|
next_effect..=target_effect_index,
|
|
);
|
|
|
|
self.pos =
|
|
CursorPosition { block: target.block, curr_effect_index: Some(target_effect_index) };
|
|
}
|
|
|
|
/// Applies `f` to the cursor's internal state.
|
|
///
|
|
/// This can be used, e.g., to apply the call return effect directly to the cursor without
|
|
/// creating an extra copy of the dataflow state.
|
|
pub fn apply_custom_effect(&mut self, f: impl FnOnce(&mut A, &mut A::Domain)) {
|
|
f(&mut self.analysis, &mut self.state);
|
|
self.state_needs_reset = true;
|
|
}
|
|
}
|
|
|
|
#[derive(Clone, Copy, Debug)]
|
|
struct CursorPosition {
|
|
block: BasicBlock,
|
|
curr_effect_index: Option<EffectIndex>,
|
|
}
|
|
|
|
impl CursorPosition {
|
|
fn block_entry(block: BasicBlock) -> CursorPosition {
|
|
CursorPosition { block, curr_effect_index: None }
|
|
}
|
|
}
|