Skip to content

Commit 50448d1

Browse files
committed
Explain why we check variant equality.
1 parent ac4572f commit 50448d1

File tree

1 file changed

+15
-0
lines changed
  • compiler/rustc_mir_transform/src

1 file changed

+15
-0
lines changed

compiler/rustc_mir_transform/src/gvn.rs

+15
Original file line numberDiff line numberDiff line change
@@ -512,6 +512,21 @@ impl<'body, 'tcx> VnState<'body, 'tcx> {
512512
return Some(fields[f.as_usize()]);
513513
} else if let Value::Projection(outer_value, ProjectionElem::Downcast(_, read_variant)) = self.get(value)
514514
&& let Value::Aggregate(_, written_variant, fields) = self.get(*outer_value)
515+
// This pass is not aware of control-flow, so we do not know whether the
516+
// replacement we are doing is actually reachable. We could be in any arm of
517+
// ```
518+
// match Some(x) {
519+
// Some(y) => /* stuff */,
520+
// None => /* other */,
521+
// }
522+
// ```
523+
//
524+
// In surface rust, the current statement would be unreachable.
525+
//
526+
// However, from the reference chapter on enums and RFC 2195,
527+
// accessing the wrong variant is not UB if the enum has repr.
528+
// So it's not impossible for a series of MIR opts to generate
529+
// a downcast to an inactive variant.
515530
&& written_variant == read_variant
516531
{
517532
return Some(fields[f.as_usize()]);

0 commit comments

Comments
 (0)