@@ -14,7 +14,8 @@ use rustc_data_structures::temp_dir::MaybeTempDir;
14
14
use rustc_hir as hir;
15
15
use rustc_hir:: def:: DefKind ;
16
16
use rustc_hir:: def_id:: {
17
- CrateNum , DefId , DefIndex , LocalDefId , CRATE_DEF_ID , CRATE_DEF_INDEX , LOCAL_CRATE ,
17
+ CrateNum , DefId , DefIndex , LocalDefId , LocalDefIdSet , CRATE_DEF_ID , CRATE_DEF_INDEX ,
18
+ LOCAL_CRATE ,
18
19
} ;
19
20
use rustc_hir:: definitions:: DefPathData ;
20
21
use rustc_hir:: lang_items:: LangItem ;
@@ -50,7 +51,6 @@ pub(super) struct EncodeContext<'a, 'tcx> {
50
51
opaque : opaque:: FileEncoder ,
51
52
tcx : TyCtxt < ' tcx > ,
52
53
feat : & ' tcx rustc_feature:: Features ,
53
-
54
54
tables : TableBuilders ,
55
55
56
56
lazy_state : LazyState ,
@@ -1002,15 +1002,31 @@ fn should_encode_stability(def_kind: DefKind) -> bool {
1002
1002
}
1003
1003
}
1004
1004
1005
- /// Whether we should encode MIR.
1005
+ /// Whether we should encode MIR. Return a pair, resp. for CTFE and for LLVM.
1006
1006
///
1007
1007
/// Computing, optimizing and encoding the MIR is a relatively expensive operation.
1008
1008
/// We want to avoid this work when not required. Therefore:
1009
1009
/// - we only compute `mir_for_ctfe` on items with const-eval semantics;
1010
1010
/// - we skip `optimized_mir` for check runs.
1011
+ /// - we only encode `optimized_mir` that could be generated in other crates, that is, a code that
1012
+ /// is either generic or has inline hint, and is reachable from the other crates (contained
1013
+ /// in reachable set).
1014
+ ///
1015
+ /// Note: Reachable set describes definitions that might be generated or referenced from other
1016
+ /// crates and it can be used to limit optimized MIR that needs to be encoded. On the other hand,
1017
+ /// the reachable set doesn't have much to say about which definitions might be evaluated at compile
1018
+ /// time in other crates, so it cannot be used to omit CTFE MIR. For example, `f` below is
1019
+ /// unreachable and yet it can be evaluated in other crates:
1011
1020
///
1012
- /// Return a pair, resp. for CTFE and for LLVM.
1013
- fn should_encode_mir ( tcx : TyCtxt < ' _ > , def_id : LocalDefId ) -> ( bool , bool ) {
1021
+ /// ```
1022
+ /// const fn f() -> usize { 0 }
1023
+ /// pub struct S { pub a: [usize; f()] }
1024
+ /// ```
1025
+ fn should_encode_mir (
1026
+ tcx : TyCtxt < ' _ > ,
1027
+ reachable_set : & LocalDefIdSet ,
1028
+ def_id : LocalDefId ,
1029
+ ) -> ( bool , bool ) {
1014
1030
match tcx. def_kind ( def_id) {
1015
1031
// Constructors
1016
1032
DefKind :: Ctor ( _, _) => {
@@ -1027,14 +1043,15 @@ fn should_encode_mir(tcx: TyCtxt<'_>, def_id: LocalDefId) -> (bool, bool) {
1027
1043
// Full-fledged functions + closures
1028
1044
DefKind :: AssocFn | DefKind :: Fn | DefKind :: Closure => {
1029
1045
let generics = tcx. generics_of ( def_id) ;
1030
- let needs_inline = ( generics. requires_monomorphization ( tcx)
1031
- || tcx. codegen_fn_attrs ( def_id) . requests_inline ( ) )
1032
- && tcx. sess . opts . output_types . should_codegen ( ) ;
1046
+ let opt = tcx. sess . opts . unstable_opts . always_encode_mir
1047
+ || ( tcx. sess . opts . output_types . should_codegen ( )
1048
+ && reachable_set. contains ( & def_id)
1049
+ && ( generics. requires_monomorphization ( tcx)
1050
+ || tcx. codegen_fn_attrs ( def_id) . requests_inline ( ) ) ) ;
1033
1051
// The function has a `const` modifier or is in a `#[const_trait]`.
1034
1052
let is_const_fn = tcx. is_const_fn_raw ( def_id. to_def_id ( ) )
1035
1053
|| tcx. is_const_default_method ( def_id. to_def_id ( ) ) ;
1036
- let always_encode_mir = tcx. sess . opts . unstable_opts . always_encode_mir ;
1037
- ( is_const_fn, needs_inline || always_encode_mir)
1054
+ ( is_const_fn, opt)
1038
1055
}
1039
1056
// Generators require optimized MIR to compute layout.
1040
1057
DefKind :: Generator => ( false , true ) ,
@@ -1580,9 +1597,10 @@ impl<'a, 'tcx> EncodeContext<'a, 'tcx> {
1580
1597
}
1581
1598
1582
1599
let tcx = self . tcx ;
1600
+ let reachable_set = tcx. reachable_set ( ( ) ) ;
1583
1601
1584
1602
let keys_and_jobs = tcx. mir_keys ( ( ) ) . iter ( ) . filter_map ( |& def_id| {
1585
- let ( encode_const, encode_opt) = should_encode_mir ( tcx, def_id) ;
1603
+ let ( encode_const, encode_opt) = should_encode_mir ( tcx, reachable_set , def_id) ;
1586
1604
if encode_const || encode_opt { Some ( ( def_id, encode_const, encode_opt) ) } else { None }
1587
1605
} ) ;
1588
1606
for ( def_id, encode_const, encode_opt) in keys_and_jobs {
@@ -2067,8 +2085,9 @@ fn prefetch_mir(tcx: TyCtxt<'_>) {
2067
2085
return ;
2068
2086
}
2069
2087
2088
+ let reachable_set = tcx. reachable_set ( ( ) ) ;
2070
2089
par_for_each_in ( tcx. mir_keys ( ( ) ) , |& def_id| {
2071
- let ( encode_const, encode_opt) = should_encode_mir ( tcx, def_id) ;
2090
+ let ( encode_const, encode_opt) = should_encode_mir ( tcx, reachable_set , def_id) ;
2072
2091
2073
2092
if encode_const {
2074
2093
tcx. ensure_with_value ( ) . mir_for_ctfe ( def_id) ;
0 commit comments