@@ -19,13 +19,32 @@ fn needs_drop_raw<'tcx>(tcx: TyCtxt<'tcx>, query: ty::ParamEnvAnd<'tcx, Ty<'tcx>
19
19
// needs drop.
20
20
let adt_has_dtor =
21
21
|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 ( ) ;
24
26
25
27
debug ! ( "needs_drop_raw({:?}) = {:?}" , query, res) ;
26
28
res
27
29
}
28
30
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
+
29
48
fn has_significant_drop_raw < ' tcx > (
30
49
tcx : TyCtxt < ' tcx > ,
31
50
query : ty:: ParamEnvAnd < ' tcx , Ty < ' tcx > > ,
@@ -37,6 +56,7 @@ fn has_significant_drop_raw<'tcx>(
37
56
adt_consider_insignificant_dtor ( tcx) ,
38
57
true ,
39
58
)
59
+ . filter ( filter_array_elements ( tcx, query. param_env ) )
40
60
. next ( )
41
61
. is_some ( ) ;
42
62
debug ! ( "has_significant_drop_raw({:?}) = {:?}" , query, res) ;
0 commit comments