Skip to content

Commit 6f180ea

Browse files
authored
Rollup merge of #115492 - Enselic:large-box-move, r=oli-obk
Allow `large_assignments` for Box/Arc/Rc initialization Does the `stop linting in box/arc initialization` task of #83518. r? `@oli-obk` who is E-mentor.
2 parents e9d15ec + 789451b commit 6f180ea

File tree

4 files changed

+147
-16
lines changed

4 files changed

+147
-16
lines changed

compiler/rustc_monomorphize/src/collector.rs

+91-10
Original file line numberDiff line numberDiff line change
@@ -179,15 +179,16 @@ use rustc_middle::query::TyCtxtAt;
179179
use rustc_middle::ty::adjustment::{CustomCoerceUnsized, PointerCoercion};
180180
use rustc_middle::ty::print::with_no_trimmed_paths;
181181
use rustc_middle::ty::{
182-
self, GenericParamDefKind, Instance, InstanceDef, Ty, TyCtxt, TypeFoldable, TypeVisitableExt,
183-
VtblEntry,
182+
self, AssocKind, GenericParamDefKind, Instance, InstanceDef, Ty, TyCtxt, TypeFoldable,
183+
TypeVisitableExt, VtblEntry,
184184
};
185185
use rustc_middle::ty::{GenericArgKind, GenericArgs};
186186
use rustc_middle::{middle::codegen_fn_attrs::CodegenFnAttrFlags, mir::visit::TyContext};
187187
use rustc_session::config::EntryFnType;
188188
use rustc_session::lint::builtin::LARGE_ASSIGNMENTS;
189189
use rustc_session::Limit;
190190
use rustc_span::source_map::{dummy_spanned, respan, Span, Spanned, DUMMY_SP};
191+
use rustc_span::symbol::{sym, Ident};
191192
use rustc_target::abi::Size;
192193
use std::path::PathBuf;
193194

@@ -432,7 +433,7 @@ fn collect_items_rec<'tcx>(
432433
hir::InlineAsmOperand::SymFn { anon_const } => {
433434
let fn_ty =
434435
tcx.typeck_body(anon_const.body).node_type(anon_const.hir_id);
435-
visit_fn_use(tcx, fn_ty, false, *op_sp, &mut used_items);
436+
visit_fn_use(tcx, fn_ty, false, *op_sp, &mut used_items, &[]);
436437
}
437438
hir::InlineAsmOperand::SymStatic { path: _, def_id } => {
438439
let instance = Instance::mono(tcx, *def_id);
@@ -593,6 +594,11 @@ struct MirUsedCollector<'a, 'tcx> {
593594
instance: Instance<'tcx>,
594595
/// Spans for move size lints already emitted. Helps avoid duplicate lints.
595596
move_size_spans: Vec<Span>,
597+
/// If true, we should temporarily skip move size checks, because we are
598+
/// processing an operand to a `skip_move_check_fns` function call.
599+
skip_move_size_check: bool,
600+
/// Set of functions for which it is OK to move large data into.
601+
skip_move_check_fns: Vec<DefId>,
596602
}
597603

598604
impl<'a, 'tcx> MirUsedCollector<'a, 'tcx> {
@@ -691,7 +697,14 @@ impl<'a, 'tcx> MirVisitor<'tcx> for MirUsedCollector<'a, 'tcx> {
691697
) => {
692698
let fn_ty = operand.ty(self.body, self.tcx);
693699
let fn_ty = self.monomorphize(fn_ty);
694-
visit_fn_use(self.tcx, fn_ty, false, span, &mut self.output);
700+
visit_fn_use(
701+
self.tcx,
702+
fn_ty,
703+
false,
704+
span,
705+
&mut self.output,
706+
&self.skip_move_check_fns,
707+
);
695708
}
696709
mir::Rvalue::Cast(
697710
mir::CastKind::PointerCoercion(PointerCoercion::ClosureFnPointer(_)),
@@ -790,7 +803,14 @@ impl<'a, 'tcx> MirVisitor<'tcx> for MirUsedCollector<'a, 'tcx> {
790803
mir::TerminatorKind::Call { ref func, .. } => {
791804
let callee_ty = func.ty(self.body, tcx);
792805
let callee_ty = self.monomorphize(callee_ty);
793-
visit_fn_use(self.tcx, callee_ty, true, source, &mut self.output)
806+
self.skip_move_size_check = visit_fn_use(
807+
self.tcx,
808+
callee_ty,
809+
true,
810+
source,
811+
&mut self.output,
812+
&self.skip_move_check_fns,
813+
)
794814
}
795815
mir::TerminatorKind::Drop { ref place, .. } => {
796816
let ty = place.ty(self.body, self.tcx).ty;
@@ -802,7 +822,7 @@ impl<'a, 'tcx> MirVisitor<'tcx> for MirUsedCollector<'a, 'tcx> {
802822
match *op {
803823
mir::InlineAsmOperand::SymFn { ref value } => {
804824
let fn_ty = self.monomorphize(value.literal.ty());
805-
visit_fn_use(self.tcx, fn_ty, false, source, &mut self.output);
825+
visit_fn_use(self.tcx, fn_ty, false, source, &mut self.output, &[]);
806826
}
807827
mir::InlineAsmOperand::SymStatic { def_id } => {
808828
let instance = Instance::mono(self.tcx, def_id);
@@ -841,12 +861,13 @@ impl<'a, 'tcx> MirVisitor<'tcx> for MirUsedCollector<'a, 'tcx> {
841861
}
842862

843863
self.super_terminator(terminator, location);
864+
self.skip_move_size_check = false;
844865
}
845866

846867
fn visit_operand(&mut self, operand: &mir::Operand<'tcx>, location: Location) {
847868
self.super_operand(operand, location);
848869
let move_size_limit = self.tcx.move_size_limit().0;
849-
if move_size_limit > 0 {
870+
if move_size_limit > 0 && !self.skip_move_size_check {
850871
self.check_move_size(move_size_limit, operand, location);
851872
}
852873
}
@@ -877,8 +898,11 @@ fn visit_fn_use<'tcx>(
877898
is_direct_call: bool,
878899
source: Span,
879900
output: &mut MonoItems<'tcx>,
880-
) {
901+
skip_move_check_fns: &[DefId],
902+
) -> bool {
903+
let mut skip_move_size_check = false;
881904
if let ty::FnDef(def_id, args) = *ty.kind() {
905+
skip_move_size_check = skip_move_check_fns.contains(&def_id);
882906
let instance = if is_direct_call {
883907
ty::Instance::expect_resolve(tcx, ty::ParamEnv::reveal_all(), def_id, args)
884908
} else {
@@ -889,6 +913,7 @@ fn visit_fn_use<'tcx>(
889913
};
890914
visit_instance_use(tcx, instance, is_direct_call, source, output);
891915
}
916+
skip_move_size_check
892917
}
893918

894919
fn visit_instance_use<'tcx>(
@@ -1369,6 +1394,31 @@ fn collect_alloc<'tcx>(tcx: TyCtxt<'tcx>, alloc_id: AllocId, output: &mut MonoIt
13691394
}
13701395
}
13711396

1397+
fn add_assoc_fn<'tcx>(
1398+
tcx: TyCtxt<'tcx>,
1399+
def_id: Option<DefId>,
1400+
fn_ident: Ident,
1401+
skip_move_check_fns: &mut Vec<DefId>,
1402+
) {
1403+
if let Some(def_id) = def_id.and_then(|def_id| assoc_fn_of_type(tcx, def_id, fn_ident)) {
1404+
skip_move_check_fns.push(def_id);
1405+
}
1406+
}
1407+
1408+
fn assoc_fn_of_type<'tcx>(tcx: TyCtxt<'tcx>, def_id: DefId, fn_ident: Ident) -> Option<DefId> {
1409+
for impl_def_id in tcx.inherent_impls(def_id) {
1410+
if let Some(new) = tcx.associated_items(impl_def_id).find_by_name_and_kind(
1411+
tcx,
1412+
fn_ident,
1413+
AssocKind::Fn,
1414+
def_id,
1415+
) {
1416+
return Some(new.def_id);
1417+
}
1418+
}
1419+
return None;
1420+
}
1421+
13721422
/// Scans the MIR in order to find function calls, closures, and drop-glue.
13731423
#[instrument(skip(tcx, output), level = "debug")]
13741424
fn collect_used_items<'tcx>(
@@ -1377,8 +1427,39 @@ fn collect_used_items<'tcx>(
13771427
output: &mut MonoItems<'tcx>,
13781428
) {
13791429
let body = tcx.instance_mir(instance.def);
1380-
MirUsedCollector { tcx, body: &body, output, instance, move_size_spans: vec![] }
1381-
.visit_body(&body);
1430+
1431+
let mut skip_move_check_fns = vec![];
1432+
if tcx.move_size_limit().0 > 0 {
1433+
add_assoc_fn(
1434+
tcx,
1435+
tcx.lang_items().owned_box(),
1436+
Ident::from_str("new"),
1437+
&mut skip_move_check_fns,
1438+
);
1439+
add_assoc_fn(
1440+
tcx,
1441+
tcx.get_diagnostic_item(sym::Arc),
1442+
Ident::from_str("new"),
1443+
&mut skip_move_check_fns,
1444+
);
1445+
add_assoc_fn(
1446+
tcx,
1447+
tcx.get_diagnostic_item(sym::Rc),
1448+
Ident::from_str("new"),
1449+
&mut skip_move_check_fns,
1450+
);
1451+
}
1452+
1453+
MirUsedCollector {
1454+
tcx,
1455+
body: &body,
1456+
output,
1457+
instance,
1458+
move_size_spans: vec![],
1459+
skip_move_size_check: false,
1460+
skip_move_check_fns,
1461+
}
1462+
.visit_body(&body);
13821463
}
13831464

13841465
#[instrument(skip(tcx, output), level = "debug")]

tests/ui/async-await/large_moves.attribute.stderr

+19-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
error: moving 10024 bytes
2-
--> $DIR/large_moves.rs:19:14
2+
--> $DIR/large_moves.rs:21:14
33
|
44
LL | let z = (x, 42);
55
| ^ value moved from here
@@ -12,12 +12,28 @@ LL | #![deny(large_assignments)]
1212
| ^^^^^^^^^^^^^^^^^
1313

1414
error: moving 10024 bytes
15-
--> $DIR/large_moves.rs:20:13
15+
--> $DIR/large_moves.rs:22:13
1616
|
1717
LL | let a = z.0;
1818
| ^^^ value moved from here
1919
|
2020
= note: The current maximum size is 1000, but it can be customized with the move_size_limit attribute: `#![move_size_limit = "..."]`
2121

22-
error: aborting due to 2 previous errors
22+
error: moving 9999 bytes
23+
--> $DIR/large_moves.rs:27:13
24+
|
25+
LL | let _ = NotBox::new([0; 9999]);
26+
| ^^^^^^^^^^^^^^^^^^^^^^ value moved from here
27+
|
28+
= note: The current maximum size is 1000, but it can be customized with the move_size_limit attribute: `#![move_size_limit = "..."]`
29+
30+
error: moving 9999 bytes
31+
--> $DIR/large_moves.rs:41:13
32+
|
33+
LL | data,
34+
| ^^^^ value moved from here
35+
|
36+
= note: The current maximum size is 1000, but it can be customized with the move_size_limit attribute: `#![move_size_limit = "..."]`
37+
38+
error: aborting due to 4 previous errors
2339

tests/ui/async-await/large_moves.option.stderr

+19-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
error: moving 10024 bytes
2-
--> $DIR/large_moves.rs:19:14
2+
--> $DIR/large_moves.rs:21:14
33
|
44
LL | let z = (x, 42);
55
| ^ value moved from here
@@ -12,12 +12,28 @@ LL | #![deny(large_assignments)]
1212
| ^^^^^^^^^^^^^^^^^
1313

1414
error: moving 10024 bytes
15-
--> $DIR/large_moves.rs:20:13
15+
--> $DIR/large_moves.rs:22:13
1616
|
1717
LL | let a = z.0;
1818
| ^^^ value moved from here
1919
|
2020
= note: The current maximum size is 1000, but it can be customized with the move_size_limit attribute: `#![move_size_limit = "..."]`
2121

22-
error: aborting due to 2 previous errors
22+
error: moving 9999 bytes
23+
--> $DIR/large_moves.rs:27:13
24+
|
25+
LL | let _ = NotBox::new([0; 9999]);
26+
| ^^^^^^^^^^^^^^^^^^^^^^ value moved from here
27+
|
28+
= note: The current maximum size is 1000, but it can be customized with the move_size_limit attribute: `#![move_size_limit = "..."]`
29+
30+
error: moving 9999 bytes
31+
--> $DIR/large_moves.rs:41:13
32+
|
33+
LL | data,
34+
| ^^^^ value moved from here
35+
|
36+
= note: The current maximum size is 1000, but it can be customized with the move_size_limit attribute: `#![move_size_limit = "..."]`
37+
38+
error: aborting due to 4 previous errors
2339

tests/ui/async-await/large_moves.rs

+18
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,8 @@
99
// edition:2018
1010
// compile-flags: -Zmir-opt-level=0
1111

12+
use std::{sync::Arc, rc::Rc};
13+
1214
fn main() {
1315
let x = async {
1416
let y = [0; 9999];
@@ -19,8 +21,24 @@ fn main() {
1921
let z = (x, 42); //~ ERROR large_assignments
2022
let a = z.0; //~ ERROR large_assignments
2123
let b = z.1;
24+
let _ = Arc::new([0; 9999]); // OK!
25+
let _ = Box::new([0; 9999]); // OK!
26+
let _ = Rc::new([0; 9999]); // OK!
27+
let _ = NotBox::new([0; 9999]); //~ ERROR large_assignments
2228
}
2329

2430
async fn thing(y: &[u8]) {
2531
dbg!(y);
2632
}
33+
34+
struct NotBox {
35+
data: [u8; 9999],
36+
}
37+
38+
impl NotBox {
39+
fn new(data: [u8; 9999]) -> Self {
40+
Self {
41+
data, //~ ERROR large_assignments
42+
}
43+
}
44+
}

0 commit comments

Comments
 (0)