Skip to content

Commit ae9ac0e

Browse files
committed
Remove the DefinitelyInitializedPlaces analysis.
Its only use is in the `tests/ui/mir-dataflow/def_inits-1.rs` where it is tested via `rustc_peek_definite_init`. Also, it's probably buggy. It's supposed to be the inverse of `MaybeUninitializedPlaces`, and it mostly is, except that `apply_terminator_effect` is a little different, and `apply_switch_int_edge_effects` is missing. Unlike `MaybeUninitializedPlaces`, which is used extensively in borrow checking, any bugs in `DefinitelyInitializedPlaces` are easy to overlook because it is only used in one small test. This commit removes the analysis. It also removes `rustc_peek_definite_init`, `Dual` and `MeetSemiLattice`, all of which are no longer needed.
1 parent a1f2999 commit ae9ac0e

File tree

9 files changed

+13
-358
lines changed

9 files changed

+13
-358
lines changed

compiler/rustc_mir_dataflow/src/framework/fmt.rs

-13
Original file line numberDiff line numberDiff line change
@@ -230,16 +230,3 @@ where
230230
write!(f, "{}", ctxt.move_data().move_paths[*self])
231231
}
232232
}
233-
234-
impl<T, C> DebugWithContext<C> for crate::lattice::Dual<T>
235-
where
236-
T: DebugWithContext<C>,
237-
{
238-
fn fmt_with(&self, ctxt: &C, f: &mut fmt::Formatter<'_>) -> fmt::Result {
239-
(self.0).fmt_with(ctxt, f)
240-
}
241-
242-
fn fmt_diff_with(&self, old: &Self, ctxt: &C, f: &mut fmt::Formatter<'_>) -> fmt::Result {
243-
(self.0).fmt_diff_with(&old.0, ctxt, f)
244-
}
245-
}

compiler/rustc_mir_dataflow/src/framework/lattice.rs

+2-106
Original file line numberDiff line numberDiff line change
@@ -25,8 +25,8 @@
2525
//!
2626
//! ## `PartialOrd`
2727
//!
28-
//! Given that they represent partially ordered sets, you may be surprised that [`JoinSemiLattice`]
29-
//! and [`MeetSemiLattice`] do not have [`PartialOrd`] as a supertrait. This
28+
//! Given that it represents a partially ordered set, you may be surprised that [`JoinSemiLattice`]
29+
//! does not have [`PartialOrd`] as a supertrait. This
3030
//! is because most standard library types use lexicographic ordering instead of set inclusion for
3131
//! their `PartialOrd` impl. Since we do not actually need to compare lattice elements to run a
3232
//! dataflow analysis, there's no need for a newtype wrapper with a custom `PartialOrd` impl. The
@@ -58,23 +58,6 @@ pub trait JoinSemiLattice: Eq {
5858
fn join(&mut self, other: &Self) -> bool;
5959
}
6060

61-
/// A [partially ordered set][poset] that has a [greatest lower bound][glb] for any pair of
62-
/// elements in the set.
63-
///
64-
/// Dataflow analyses only require that their domains implement [`JoinSemiLattice`], not
65-
/// `MeetSemiLattice`. However, types that will be used as dataflow domains should implement both
66-
/// so that they can be used with [`Dual`].
67-
///
68-
/// [glb]: https://en.wikipedia.org/wiki/Infimum_and_supremum
69-
/// [poset]: https://en.wikipedia.org/wiki/Partially_ordered_set
70-
pub trait MeetSemiLattice: Eq {
71-
/// Computes the greatest lower bound of two elements, storing the result in `self` and
72-
/// returning `true` if `self` has changed.
73-
///
74-
/// The lattice meet operator is abbreviated as `∧`.
75-
fn meet(&mut self, other: &Self) -> bool;
76-
}
77-
7861
/// A set that has a "bottom" element, which is less than or equal to any other element.
7962
pub trait HasBottom {
8063
const BOTTOM: Self;
@@ -105,17 +88,6 @@ impl JoinSemiLattice for bool {
10588
}
10689
}
10790

108-
impl MeetSemiLattice for bool {
109-
fn meet(&mut self, other: &Self) -> bool {
110-
if let (true, false) = (*self, *other) {
111-
*self = false;
112-
return true;
113-
}
114-
115-
false
116-
}
117-
}
118-
11991
impl HasBottom for bool {
12092
const BOTTOM: Self = false;
12193

@@ -145,18 +117,6 @@ impl<I: Idx, T: JoinSemiLattice> JoinSemiLattice for IndexVec<I, T> {
145117
}
146118
}
147119

148-
impl<I: Idx, T: MeetSemiLattice> MeetSemiLattice for IndexVec<I, T> {
149-
fn meet(&mut self, other: &Self) -> bool {
150-
assert_eq!(self.len(), other.len());
151-
152-
let mut changed = false;
153-
for (a, b) in iter::zip(self, other) {
154-
changed |= a.meet(b);
155-
}
156-
changed
157-
}
158-
}
159-
160120
/// A `BitSet` represents the lattice formed by the powerset of all possible values of
161121
/// the index type `T` ordered by inclusion. Equivalently, it is a tuple of "two-point" lattices,
162122
/// one for each possible value of `T`.
@@ -166,60 +126,12 @@ impl<T: Idx> JoinSemiLattice for BitSet<T> {
166126
}
167127
}
168128

169-
impl<T: Idx> MeetSemiLattice for BitSet<T> {
170-
fn meet(&mut self, other: &Self) -> bool {
171-
self.intersect(other)
172-
}
173-
}
174-
175129
impl<T: Idx> JoinSemiLattice for ChunkedBitSet<T> {
176130
fn join(&mut self, other: &Self) -> bool {
177131
self.union(other)
178132
}
179133
}
180134

181-
impl<T: Idx> MeetSemiLattice for ChunkedBitSet<T> {
182-
fn meet(&mut self, other: &Self) -> bool {
183-
self.intersect(other)
184-
}
185-
}
186-
187-
/// The counterpart of a given semilattice `T` using the [inverse order].
188-
///
189-
/// The dual of a join-semilattice is a meet-semilattice and vice versa. For example, the dual of a
190-
/// powerset has the empty set as its top element and the full set as its bottom element and uses
191-
/// set *intersection* as its join operator.
192-
///
193-
/// [inverse order]: https://en.wikipedia.org/wiki/Duality_(order_theory)
194-
#[derive(Clone, Copy, Debug, PartialEq, Eq)]
195-
pub struct Dual<T>(pub T);
196-
197-
impl<T: Idx> BitSetExt<T> for Dual<BitSet<T>> {
198-
fn contains(&self, elem: T) -> bool {
199-
self.0.contains(elem)
200-
}
201-
202-
fn union(&mut self, other: &HybridBitSet<T>) {
203-
self.0.union(other);
204-
}
205-
206-
fn subtract(&mut self, other: &HybridBitSet<T>) {
207-
self.0.subtract(other);
208-
}
209-
}
210-
211-
impl<T: MeetSemiLattice> JoinSemiLattice for Dual<T> {
212-
fn join(&mut self, other: &Self) -> bool {
213-
self.0.meet(&other.0)
214-
}
215-
}
216-
217-
impl<T: JoinSemiLattice> MeetSemiLattice for Dual<T> {
218-
fn meet(&mut self, other: &Self) -> bool {
219-
self.0.join(&other.0)
220-
}
221-
}
222-
223135
/// Extends a type `T` with top and bottom elements to make it a partially ordered set in which no
224136
/// value of `T` is comparable with any other.
225137
///
@@ -257,22 +169,6 @@ impl<T: Clone + Eq> JoinSemiLattice for FlatSet<T> {
257169
}
258170
}
259171

260-
impl<T: Clone + Eq> MeetSemiLattice for FlatSet<T> {
261-
fn meet(&mut self, other: &Self) -> bool {
262-
let result = match (&*self, other) {
263-
(Self::Bottom, _) | (_, Self::Top) => return false,
264-
(Self::Elem(ref a), Self::Elem(ref b)) if a == b => return false,
265-
266-
(Self::Top, Self::Elem(ref x)) => Self::Elem(x.clone()),
267-
268-
_ => Self::Bottom,
269-
};
270-
271-
*self = result;
272-
true
273-
}
274-
}
275-
276172
impl<T> HasBottom for FlatSet<T> {
277173
const BOTTOM: Self = Self::Bottom;
278174

compiler/rustc_mir_dataflow/src/framework/mod.rs

-10
Original file line numberDiff line numberDiff line change
@@ -378,16 +378,6 @@ impl<T, S: GenKill<T>> GenKill<T> for MaybeReachable<S> {
378378
}
379379
}
380380

381-
impl<T: Idx> GenKill<T> for lattice::Dual<BitSet<T>> {
382-
fn gen_(&mut self, elem: T) {
383-
self.0.insert(elem);
384-
}
385-
386-
fn kill(&mut self, elem: T) {
387-
self.0.remove(elem);
388-
}
389-
}
390-
391381
// NOTE: DO NOT CHANGE VARIANT ORDER. The derived `Ord` impls rely on the current order.
392382
#[derive(Clone, Copy, Debug, PartialEq, Eq, PartialOrd, Ord)]
393383
enum Effect {

compiler/rustc_mir_dataflow/src/impls/initialized.rs

+9-137
Original file line numberDiff line numberDiff line change
@@ -12,7 +12,7 @@ use crate::framework::SwitchIntEdgeEffects;
1212
use crate::move_paths::{HasMoveData, InitIndex, InitKind, LookupResult, MoveData, MovePathIndex};
1313
use crate::{
1414
Analysis, GenKill, MaybeReachable, drop_flag_effects, drop_flag_effects_for_function_entry,
15-
drop_flag_effects_for_location, lattice, on_all_children_bits, on_lookup_result_bits,
15+
drop_flag_effects_for_location, on_all_children_bits, on_lookup_result_bits,
1616
};
1717

1818
/// `MaybeInitializedPlaces` tracks all places that might be
@@ -42,10 +42,10 @@ use crate::{
4242
/// }
4343
/// ```
4444
///
45-
/// To determine whether a place *must* be initialized at a
46-
/// particular control-flow point, one can take the set-difference
47-
/// between this data and the data from `MaybeUninitializedPlaces` at the
48-
/// corresponding control-flow point.
45+
/// To determine whether a place is *definitely* initialized at a
46+
/// particular control-flow point, one can take the set-complement
47+
/// of the data from `MaybeUninitializedPlaces` at the corresponding
48+
/// control-flow point.
4949
///
5050
/// Similarly, at a given `drop` statement, the set-intersection
5151
/// between this data and `MaybeUninitializedPlaces` yields the set of
@@ -117,10 +117,10 @@ impl<'a, 'tcx> HasMoveData<'tcx> for MaybeInitializedPlaces<'a, 'tcx> {
117117
/// }
118118
/// ```
119119
///
120-
/// To determine whether a place *must* be uninitialized at a
121-
/// particular control-flow point, one can take the set-difference
122-
/// between this data and the data from `MaybeInitializedPlaces` at the
123-
/// corresponding control-flow point.
120+
/// To determine whether a place is *definitely* uninitialized at a
121+
/// particular control-flow point, one can take the set-complement
122+
/// of the data from `MaybeInitializedPlaces` at the corresponding
123+
/// control-flow point.
124124
///
125125
/// Similarly, at a given `drop` statement, the set-intersection
126126
/// between this data and `MaybeInitializedPlaces` yields the set of
@@ -170,57 +170,6 @@ impl<'tcx> HasMoveData<'tcx> for MaybeUninitializedPlaces<'_, 'tcx> {
170170
}
171171
}
172172

173-
/// `DefinitelyInitializedPlaces` tracks all places that are definitely
174-
/// initialized upon reaching a particular point in the control flow
175-
/// for a function.
176-
///
177-
/// For example, in code like the following, we have corresponding
178-
/// dataflow information shown in the right-hand comments.
179-
///
180-
/// ```rust
181-
/// struct S;
182-
/// fn foo(pred: bool) { // definite-init:
183-
/// // { }
184-
/// let a = S; let mut b = S; let c; let d; // {a, b }
185-
///
186-
/// if pred {
187-
/// drop(a); // { b, }
188-
/// b = S; // { b, }
189-
///
190-
/// } else {
191-
/// drop(b); // {a, }
192-
/// d = S; // {a, d}
193-
///
194-
/// } // { }
195-
///
196-
/// c = S; // { c }
197-
/// }
198-
/// ```
199-
///
200-
/// To determine whether a place *may* be uninitialized at a
201-
/// particular control-flow point, one can take the set-complement
202-
/// of this data.
203-
///
204-
/// Similarly, at a given `drop` statement, the set-difference between
205-
/// this data and `MaybeInitializedPlaces` yields the set of places
206-
/// that would require a dynamic drop-flag at that statement.
207-
pub struct DefinitelyInitializedPlaces<'a, 'tcx> {
208-
body: &'a Body<'tcx>,
209-
move_data: &'a MoveData<'tcx>,
210-
}
211-
212-
impl<'a, 'tcx> DefinitelyInitializedPlaces<'a, 'tcx> {
213-
pub fn new(body: &'a Body<'tcx>, move_data: &'a MoveData<'tcx>) -> Self {
214-
DefinitelyInitializedPlaces { body, move_data }
215-
}
216-
}
217-
218-
impl<'a, 'tcx> HasMoveData<'tcx> for DefinitelyInitializedPlaces<'a, 'tcx> {
219-
fn move_data(&self) -> &MoveData<'tcx> {
220-
self.move_data
221-
}
222-
}
223-
224173
/// `EverInitializedPlaces` tracks all places that might have ever been
225174
/// initialized upon reaching a particular point in the control flow
226175
/// for a function, without an intervening `StorageDead`.
@@ -293,19 +242,6 @@ impl<'tcx> MaybeUninitializedPlaces<'_, 'tcx> {
293242
}
294243
}
295244

296-
impl<'a, 'tcx> DefinitelyInitializedPlaces<'a, 'tcx> {
297-
fn update_bits(
298-
trans: &mut <Self as Analysis<'tcx>>::Domain,
299-
path: MovePathIndex,
300-
state: DropFlagState,
301-
) {
302-
match state {
303-
DropFlagState::Absent => trans.kill(path),
304-
DropFlagState::Present => trans.gen_(path),
305-
}
306-
}
307-
}
308-
309245
impl<'tcx> Analysis<'tcx> for MaybeInitializedPlaces<'_, 'tcx> {
310246
/// There can be many more `MovePathIndex` than there are locals in a MIR body.
311247
/// We use a chunked bitset to avoid paying too high a memory footprint.
@@ -554,70 +490,6 @@ impl<'tcx> Analysis<'tcx> for MaybeUninitializedPlaces<'_, 'tcx> {
554490
}
555491
}
556492

557-
impl<'a, 'tcx> Analysis<'tcx> for DefinitelyInitializedPlaces<'a, 'tcx> {
558-
/// Use set intersection as the join operator.
559-
type Domain = lattice::Dual<BitSet<MovePathIndex>>;
560-
561-
const NAME: &'static str = "definite_init";
562-
563-
fn bottom_value(&self, _: &mir::Body<'tcx>) -> Self::Domain {
564-
// bottom = initialized (start_block_effect counters this at outset)
565-
lattice::Dual(BitSet::new_filled(self.move_data().move_paths.len()))
566-
}
567-
568-
// sets on_entry bits for Arg places
569-
fn initialize_start_block(&self, _: &mir::Body<'tcx>, state: &mut Self::Domain) {
570-
state.0.clear();
571-
572-
drop_flag_effects_for_function_entry(self.body, self.move_data, |path, s| {
573-
assert!(s == DropFlagState::Present);
574-
state.0.insert(path);
575-
});
576-
}
577-
578-
fn apply_statement_effect(
579-
&mut self,
580-
trans: &mut Self::Domain,
581-
_statement: &mir::Statement<'tcx>,
582-
location: Location,
583-
) {
584-
drop_flag_effects_for_location(self.body, self.move_data, location, |path, s| {
585-
Self::update_bits(trans, path, s)
586-
})
587-
}
588-
589-
fn apply_terminator_effect<'mir>(
590-
&mut self,
591-
trans: &mut Self::Domain,
592-
terminator: &'mir mir::Terminator<'tcx>,
593-
location: Location,
594-
) -> TerminatorEdges<'mir, 'tcx> {
595-
drop_flag_effects_for_location(self.body, self.move_data, location, |path, s| {
596-
Self::update_bits(trans, path, s)
597-
});
598-
terminator.edges()
599-
}
600-
601-
fn apply_call_return_effect(
602-
&mut self,
603-
trans: &mut Self::Domain,
604-
_block: mir::BasicBlock,
605-
return_places: CallReturnPlaces<'_, 'tcx>,
606-
) {
607-
return_places.for_each(|place| {
608-
// when a call returns successfully, that means we need to set
609-
// the bits for that dest_place to 1 (initialized).
610-
on_lookup_result_bits(
611-
self.move_data(),
612-
self.move_data().rev_lookup.find(place.as_ref()),
613-
|mpi| {
614-
trans.gen_(mpi);
615-
},
616-
);
617-
});
618-
}
619-
}
620-
621493
impl<'tcx> Analysis<'tcx> for EverInitializedPlaces<'_, 'tcx> {
622494
/// There can be many more `InitIndex` than there are locals in a MIR body.
623495
/// We use a chunked bitset to avoid paying too high a memory footprint.

compiler/rustc_mir_dataflow/src/impls/mod.rs

+1-2
Original file line numberDiff line numberDiff line change
@@ -9,8 +9,7 @@ mod storage_liveness;
99

1010
pub use self::borrowed_locals::{MaybeBorrowedLocals, borrowed_locals};
1111
pub use self::initialized::{
12-
DefinitelyInitializedPlaces, EverInitializedPlaces, MaybeInitializedPlaces,
13-
MaybeUninitializedPlaces,
12+
EverInitializedPlaces, MaybeInitializedPlaces, MaybeUninitializedPlaces,
1413
};
1514
pub use self::liveness::{
1615
MaybeLiveLocals, MaybeTransitiveLiveLocals, TransferFunction as LivenessTransferFunction,

0 commit comments

Comments
 (0)