Skip to content

Commit 320bb81

Browse files
committed
Don't require Drop for [PhantomData<T>; N] where N and T are generic, if T requires Drop
1 parent 61efe9d commit 320bb81

File tree

2 files changed

+39
-2
lines changed

2 files changed

+39
-2
lines changed

compiler/rustc_ty_utils/src/needs_drop.rs

+22-2
Original file line numberDiff line numberDiff line change
@@ -19,13 +19,32 @@ fn needs_drop_raw<'tcx>(tcx: TyCtxt<'tcx>, query: ty::ParamEnvAnd<'tcx, Ty<'tcx>
1919
// needs drop.
2020
let adt_has_dtor =
2121
|adt_def: ty::AdtDef<'tcx>| adt_def.destructor(tcx).map(|_| DtorType::Significant);
22-
let res =
23-
drop_tys_helper(tcx, query.value, query.param_env, adt_has_dtor, false).next().is_some();
22+
let res = drop_tys_helper(tcx, query.value, query.param_env, adt_has_dtor, false)
23+
.filter(filter_array_elements(tcx, query.param_env))
24+
.next()
25+
.is_some();
2426

2527
debug!("needs_drop_raw({:?}) = {:?}", query, res);
2628
res
2729
}
2830

31+
/// HACK: in order to not mistakenly assume that `[PhantomData<T>; N]` requires drop glue
32+
/// we check the element type for drop glue. The correct fix would be looking at the
33+
/// entirety of the code around `needs_drop_components` and this file and come up with
34+
/// logic that is easier to follow while not repeating any checks that may thus diverge.
35+
fn filter_array_elements<'tcx>(
36+
tcx: TyCtxt<'tcx>,
37+
param_env: ty::ParamEnv<'tcx>,
38+
) -> impl Fn(&Result<Ty<'tcx>, AlwaysRequiresDrop>) -> bool {
39+
move |ty| match ty {
40+
Ok(ty) => match *ty.kind() {
41+
ty::Array(elem, _) => tcx.needs_drop_raw(param_env.and(elem)),
42+
_ => true,
43+
},
44+
Err(AlwaysRequiresDrop) => true,
45+
}
46+
}
47+
2948
fn has_significant_drop_raw<'tcx>(
3049
tcx: TyCtxt<'tcx>,
3150
query: ty::ParamEnvAnd<'tcx, Ty<'tcx>>,
@@ -37,6 +56,7 @@ fn has_significant_drop_raw<'tcx>(
3756
adt_consider_insignificant_dtor(tcx),
3857
true,
3958
)
59+
.filter(filter_array_elements(tcx, query.param_env))
4060
.next()
4161
.is_some();
4262
debug!("has_significant_drop_raw({:?}) = {:?}", query, res);

tests/ui/consts/drop-maybe_uninit.rs

+17
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
// build-pass
2+
3+
pub const fn f<T, const N: usize>(_: [std::mem::MaybeUninit<T>; N]) {}
4+
5+
pub struct Blubb<T>(*const T);
6+
7+
pub const fn g<T, const N: usize>(_: [Blubb<T>; N]) {}
8+
9+
pub struct Blorb<const N: usize>([String; N]);
10+
11+
pub const fn h(_: Blorb<0>) {}
12+
13+
pub struct Wrap(Blorb<0>);
14+
15+
pub const fn i(_: Wrap) {}
16+
17+
fn main() {}

0 commit comments

Comments
 (0)