Skip to content

Commit 2fddcfd

Browse files
authoredJun 3, 2021
Rollup merge of #85934 - tmiasko:is-union, r=jackh726
Add `Ty::is_union` predicate
2 parents 80c871c + c898681 commit 2fddcfd

File tree

10 files changed

+32
-59
lines changed

10 files changed

+32
-59
lines changed
 

‎compiler/rustc_middle/src/ty/sty.rs

+6-4
Original file line numberDiff line numberDiff line change
@@ -1837,10 +1837,12 @@ impl<'tcx> TyS<'tcx> {
18371837

18381838
#[inline]
18391839
pub fn is_enum(&self) -> bool {
1840-
match self.kind() {
1841-
Adt(adt_def, _) => adt_def.is_enum(),
1842-
_ => false,
1843-
}
1840+
matches!(self.kind(), Adt(adt_def, _) if adt_def.is_enum())
1841+
}
1842+
1843+
#[inline]
1844+
pub fn is_union(&self) -> bool {
1845+
matches!(self.kind(), Adt(adt_def, _) if adt_def.is_union())
18441846
}
18451847

18461848
#[inline]

‎compiler/rustc_mir/src/borrow_check/mod.rs

+5-7
Original file line numberDiff line numberDiff line change
@@ -1965,13 +1965,11 @@ impl<'cx, 'tcx> MirBorrowckCtxt<'cx, 'tcx> {
19651965
// no move out from an earlier location) then this is an attempt at initialization
19661966
// of the union - we should error in that case.
19671967
let tcx = this.infcx.tcx;
1968-
if let ty::Adt(def, _) = base.ty(this.body(), tcx).ty.kind() {
1969-
if def.is_union() {
1970-
if this.move_data.path_map[mpi].iter().any(|moi| {
1971-
this.move_data.moves[*moi].source.is_predecessor_of(location, this.body)
1972-
}) {
1973-
return;
1974-
}
1968+
if base.ty(this.body(), tcx).ty.is_union() {
1969+
if this.move_data.path_map[mpi].iter().any(|moi| {
1970+
this.move_data.moves[*moi].source.is_predecessor_of(location, this.body)
1971+
}) {
1972+
return;
19751973
}
19761974
}
19771975

‎compiler/rustc_mir/src/borrow_check/places_conflict.rs

+8-11
Original file line numberDiff line numberDiff line change
@@ -331,17 +331,14 @@ fn place_projection_conflict<'tcx>(
331331
Overlap::EqualOrDisjoint
332332
} else {
333333
let ty = Place::ty_from(pi1_local, pi1_proj_base, body, tcx).ty;
334-
match ty.kind() {
335-
ty::Adt(def, _) if def.is_union() => {
336-
// Different fields of a union, we are basically stuck.
337-
debug!("place_element_conflict: STUCK-UNION");
338-
Overlap::Arbitrary
339-
}
340-
_ => {
341-
// Different fields of a struct (`a.x` vs. `a.y`). Disjoint!
342-
debug!("place_element_conflict: DISJOINT-FIELD");
343-
Overlap::Disjoint
344-
}
334+
if ty.is_union() {
335+
// Different fields of a union, we are basically stuck.
336+
debug!("place_element_conflict: STUCK-UNION");
337+
Overlap::Arbitrary
338+
} else {
339+
// Different fields of a struct (`a.x` vs. `a.y`). Disjoint!
340+
debug!("place_element_conflict: DISJOINT-FIELD");
341+
Overlap::Disjoint
345342
}
346343
}
347344
}

‎compiler/rustc_mir/src/dataflow/move_paths/builder.rs

+2-4
Original file line numberDiff line numberDiff line change
@@ -519,10 +519,8 @@ impl<'b, 'a, 'tcx> Gatherer<'b, 'a, 'tcx> {
519519
// Check if we are assigning into a field of a union, if so, lookup the place
520520
// of the union so it is marked as initialized again.
521521
if let Some((place_base, ProjectionElem::Field(_, _))) = place.last_projection() {
522-
if let ty::Adt(def, _) = place_base.ty(self.builder.body, self.builder.tcx).ty.kind() {
523-
if def.is_union() {
524-
place = place_base;
525-
}
522+
if place_base.ty(self.builder.body, self.builder.tcx).ty.is_union() {
523+
place = place_base;
526524
}
527525
}
528526

‎compiler/rustc_mir/src/transform/check_consts/validation.rs

+2-6
Original file line numberDiff line numberDiff line change
@@ -752,12 +752,8 @@ impl Visitor<'tcx> for Validator<'mir, 'tcx> {
752752
| ProjectionElem::Field(..)
753753
| ProjectionElem::Index(_) => {
754754
let base_ty = Place::ty_from(place_local, proj_base, self.body, self.tcx).ty;
755-
match base_ty.ty_adt_def() {
756-
Some(def) if def.is_union() => {
757-
self.check_op(ops::UnionAccess);
758-
}
759-
760-
_ => {}
755+
if base_ty.is_union() {
756+
self.check_op(ops::UnionAccess);
761757
}
762758
}
763759
}

‎compiler/rustc_mir/src/transform/check_unsafety.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -221,7 +221,7 @@ impl<'a, 'tcx> Visitor<'tcx> for UnsafetyChecker<'a, 'tcx> {
221221
}
222222

223223
let base_ty = base.ty(self.body, self.tcx).ty;
224-
if base_ty.ty_adt_def().map_or(false, |adt| adt.is_union()) {
224+
if base_ty.is_union() {
225225
// If we did not hit a `Deref` yet and the overall place use is an assignment, the
226226
// rules are different.
227227
let assign_to_field = !saw_deref

‎compiler/rustc_mir/src/transform/dest_prop.rs

+3-12
Original file line numberDiff line numberDiff line change
@@ -114,7 +114,7 @@ use rustc_middle::mir::{
114114
traversal, Body, InlineAsmOperand, Local, LocalKind, Location, Operand, Place, PlaceElem,
115115
Rvalue, Statement, StatementKind, Terminator, TerminatorKind,
116116
};
117-
use rustc_middle::ty::{self, Ty, TyCtxt};
117+
use rustc_middle::ty::TyCtxt;
118118

119119
// Empirical measurements have resulted in some observations:
120120
// - Running on a body with a single block and 500 locals takes barely any time
@@ -910,17 +910,8 @@ impl<'a, 'tcx> Visitor<'tcx> for FindAssignments<'a, 'tcx> {
910910

911911
// Handle the "subtle case" described above by rejecting any `dest` that is or
912912
// projects through a union.
913-
let is_union = |ty: Ty<'_>| {
914-
if let ty::Adt(def, _) = ty.kind() {
915-
if def.is_union() {
916-
return true;
917-
}
918-
}
919-
920-
false
921-
};
922913
let mut place_ty = PlaceTy::from_ty(self.body.local_decls[dest.local].ty);
923-
if is_union(place_ty.ty) {
914+
if place_ty.ty.is_union() {
924915
return;
925916
}
926917
for elem in dest.projection {
@@ -930,7 +921,7 @@ impl<'a, 'tcx> Visitor<'tcx> for FindAssignments<'a, 'tcx> {
930921
}
931922

932923
place_ty = place_ty.projection_ty(self.tcx, elem);
933-
if is_union(place_ty.ty) {
924+
if place_ty.ty.is_union() {
934925
return;
935926
}
936927
}

‎compiler/rustc_mir/src/transform/promote_consts.rs

+2-4
Original file line numberDiff line numberDiff line change
@@ -415,11 +415,9 @@ impl<'tcx> Validator<'_, 'tcx> {
415415

416416
ProjectionElem::Field(..) => {
417417
let base_ty = place_base.ty(self.body, self.tcx).ty;
418-
if let Some(def) = base_ty.ty_adt_def() {
418+
if base_ty.is_union() {
419419
// No promotion of union field accesses.
420-
if def.is_union() {
421-
return Err(Unpromotable);
422-
}
420+
return Err(Unpromotable);
423421
}
424422
}
425423
}

‎compiler/rustc_mir/src/transform/remove_zsts.rs

+2-9
Original file line numberDiff line numberDiff line change
@@ -69,21 +69,14 @@ fn involves_a_union<'tcx>(
6969
tcx: TyCtxt<'tcx>,
7070
) -> bool {
7171
let mut place_ty = PlaceTy::from_ty(local_decls[place.local].ty);
72-
if is_union(place_ty.ty) {
72+
if place_ty.ty.is_union() {
7373
return true;
7474
}
7575
for elem in place.projection {
7676
place_ty = place_ty.projection_ty(tcx, elem);
77-
if is_union(place_ty.ty) {
77+
if place_ty.ty.is_union() {
7878
return true;
7979
}
8080
}
8181
return false;
8282
}
83-
84-
fn is_union(ty: Ty<'_>) -> bool {
85-
match ty.kind() {
86-
ty::Adt(def, _) if def.is_union() => true,
87-
_ => false,
88-
}
89-
}

‎compiler/rustc_typeck/src/check/place_op.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -248,7 +248,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
248248
// Clear previous flag; after a pointer indirection it does not apply any more.
249249
inside_union = false;
250250
}
251-
if source.ty_adt_def().map_or(false, |adt| adt.is_union()) {
251+
if source.is_union() {
252252
inside_union = true;
253253
}
254254
// Fix up the autoderefs. Autorefs can only occur immediately preceding

0 commit comments

Comments
 (0)
Please sign in to comment.