Skip to content

Commit f39ef07

Browse files
committed
Auto merge of rust-lang#122258 - RalfJung:required-fns, r=<try>
Draft: monomorphize things from dead code, too This is another attempt at fixing rust-lang#107503. The previous attempt at rust-lang#112879 seems stuck in figuring out where the perf regression comes from. So here I want to take baby steps to see the impact of each step. r? `@ghost`
2 parents 25ee3c6 + 95318b4 commit f39ef07

File tree

4 files changed

+52
-7
lines changed

4 files changed

+52
-7
lines changed

compiler/rustc_middle/src/mir/mod.rs

+6-1
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@ use crate::ty::fold::{FallibleTypeFolder, TypeFoldable};
99
use crate::ty::print::{pretty_print_const, with_no_trimmed_paths};
1010
use crate::ty::print::{FmtPrinter, Printer};
1111
use crate::ty::visit::TypeVisitableExt;
12-
use crate::ty::{self, List, Ty, TyCtxt};
12+
use crate::ty::{self, Instance, List, Ty, TyCtxt};
1313
use crate::ty::{AdtDef, InstanceDef, UserTypeAnnotationIndex};
1414
use crate::ty::{GenericArg, GenericArgsRef};
1515

@@ -375,6 +375,9 @@ pub struct Body<'tcx> {
375375
/// We hold in this field all the constants we are not able to evaluate yet.
376376
pub required_consts: Vec<ConstOperand<'tcx>>,
377377

378+
/// Functions that need to monomorphize successfully for this MIR to be well-formed.
379+
pub required_fns: Vec<Instance<'tcx>>,
380+
378381
/// Does this body use generic parameters. This is used for the `ConstEvaluatable` check.
379382
///
380383
/// Note that this does not actually mean that this body is not computable right now.
@@ -445,6 +448,7 @@ impl<'tcx> Body<'tcx> {
445448
var_debug_info,
446449
span,
447450
required_consts: Vec::new(),
451+
required_fns: Vec::new(),
448452
is_polymorphic: false,
449453
injection_phase: None,
450454
tainted_by_errors,
@@ -473,6 +477,7 @@ impl<'tcx> Body<'tcx> {
473477
spread_arg: None,
474478
span: DUMMY_SP,
475479
required_consts: Vec::new(),
480+
required_fns: Vec::new(),
476481
var_debug_info: Vec::new(),
477482
is_polymorphic: false,
478483
injection_phase: None,

compiler/rustc_mir_build/src/build/custom/mod.rs

+1
Original file line numberDiff line numberDiff line change
@@ -56,6 +56,7 @@ pub(super) fn build_custom_mir<'tcx>(
5656
var_debug_info: Vec::new(),
5757
span,
5858
required_consts: Vec::new(),
59+
required_fns: Vec::new(),
5960
is_polymorphic: false,
6061
tainted_by_errors: None,
6162
injection_phase: None,

compiler/rustc_mir_transform/src/lib.rs

+4-1
Original file line numberDiff line numberDiff line change
@@ -344,11 +344,14 @@ fn mir_promoted(
344344
}
345345

346346
let mut required_consts = Vec::new();
347-
let mut required_consts_visitor = RequiredConstsVisitor::new(&mut required_consts);
347+
let mut required_fns = Vec::new();
348+
let mut required_consts_visitor =
349+
RequiredConstsVisitor::new(tcx, &body, &mut required_consts, &mut required_fns);
348350
for (bb, bb_data) in traversal::reverse_postorder(&body) {
349351
required_consts_visitor.visit_basic_block_data(bb, bb_data);
350352
}
351353
body.required_consts = required_consts;
354+
body.required_fns = required_fns;
352355

353356
// What we need to run borrowck etc.
354357
let promote_pass = promote_consts::PromoteTemps::default();
Original file line numberDiff line numberDiff line change
@@ -1,14 +1,23 @@
1+
use rustc_hir::LangItem;
12
use rustc_middle::mir::visit::Visitor;
2-
use rustc_middle::mir::{Const, ConstOperand, Location};
3-
use rustc_middle::ty::ConstKind;
3+
use rustc_middle::mir::{self, Const, ConstOperand, Location};
4+
use rustc_middle::ty::{self, ConstKind, Instance, InstanceDef, TyCtxt};
45

56
pub struct RequiredConstsVisitor<'a, 'tcx> {
7+
tcx: TyCtxt<'tcx>,
8+
body: &'a mir::Body<'tcx>,
69
required_consts: &'a mut Vec<ConstOperand<'tcx>>,
10+
required_fns: &'a mut Vec<Instance<'tcx>>,
711
}
812

913
impl<'a, 'tcx> RequiredConstsVisitor<'a, 'tcx> {
10-
pub fn new(required_consts: &'a mut Vec<ConstOperand<'tcx>>) -> Self {
11-
RequiredConstsVisitor { required_consts }
14+
pub fn new(
15+
tcx: TyCtxt<'tcx>,
16+
body: &'a mir::Body<'tcx>,
17+
required_consts: &'a mut Vec<ConstOperand<'tcx>>,
18+
required_fns: &'a mut Vec<Instance<'tcx>>,
19+
) -> Self {
20+
RequiredConstsVisitor { tcx, body, required_consts, required_fns }
1221
}
1322
}
1423

@@ -21,7 +30,34 @@ impl<'tcx> Visitor<'tcx> for RequiredConstsVisitor<'_, 'tcx> {
2130
_ => bug!("only ConstKind::Param/Value should be encountered here, got {:#?}", c),
2231
},
2332
Const::Unevaluated(..) => self.required_consts.push(*constant),
24-
Const::Val(..) => {}
33+
Const::Val(_val, ty) => {
34+
// This is how function items get referenced: via zero-sized constants of `FnDef` type
35+
if let ty::FnDef(def_id, args) = ty.kind() {
36+
debug!("adding to required_fns: {def_id:?}");
37+
// FIXME maybe we shouldn't use `Instance`? We can't use `Instance::new`, it is
38+
// for codegen. But `Instance` feels like the right representation... Check what
39+
// the regular collector does.
40+
self.required_fns.push(Instance { def: InstanceDef::Item(*def_id), args });
41+
}
42+
}
43+
}
44+
}
45+
46+
fn visit_terminator(&mut self, terminator: &mir::Terminator<'tcx>, location: Location) {
47+
self.super_terminator(terminator, location);
48+
49+
match terminator.kind {
50+
// We don't need to handle `Call` as we already handled all function type operands in
51+
// `visit_constant`. But we do need to handle `Drop`.
52+
mir::TerminatorKind::Drop { place, .. } => {
53+
let ty = place.ty(self.body, self.tcx).ty;
54+
let def_id = self.tcx.require_lang_item(LangItem::DropInPlace, None);
55+
let args = self.tcx.mk_args(&[ty.into()]);
56+
// FIXME: same as above (we cannot use `Instance::resolve_drop_in_place` as this is
57+
// still generic).
58+
self.required_fns.push(Instance { def: InstanceDef::Item(def_id), args });
59+
}
60+
_ => {}
2561
}
2662
}
2763
}

0 commit comments

Comments
 (0)