Skip to content

Commit a5560a6

Browse files
committed
Auto merge of #85437 - GuillaumeGomez:rollup-3jcirty, r=GuillaumeGomez
Rollup of 7 pull requests Successful merges: - #84587 (rustdoc: Make "rust code block is empty" and "could not parse code block" warnings a lint (`INVALID_RUST_CODEBLOCKS`)) - #85280 (Toggle-wrap items differently than top-doc.) - #85338 (Implement more Iterator methods on core::iter::Repeat) - #85339 (Report an error if a lang item has the wrong number of generic arguments) - #85369 (Suggest borrowing if a trait implementation is found for &/&mut <type>) - #85393 (Suppress spurious errors inside `async fn`) - #85415 (Clean up remnants of BorrowOfPackedField) Failed merges: r? `@ghost` `@rustbot` modify labels: rollup
2 parents 70b9d1e + f4a0d97 commit a5560a6

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

42 files changed

+800
-133
lines changed

compiler/rustc_middle/src/mir/query.rs

-6
Original file line numberDiff line numberDiff line change
@@ -32,7 +32,6 @@ pub enum UnsafetyViolationDetails {
3232
UseOfInlineAssembly,
3333
InitializingTypeWith,
3434
CastOfPointerToInt,
35-
BorrowOfPackedField,
3635
UseOfMutableStatic,
3736
UseOfExternStatic,
3837
DerefOfRawPointer,
@@ -64,11 +63,6 @@ impl UnsafetyViolationDetails {
6463
CastOfPointerToInt => {
6564
("cast of pointer to int", "casting pointers to integers in constants")
6665
}
67-
BorrowOfPackedField => (
68-
"borrow of packed field",
69-
"fields of packed structs might be misaligned: dereferencing a misaligned pointer \
70-
or even just creating a misaligned reference is undefined behavior",
71-
),
7266
UseOfMutableStatic => (
7367
"use of mutable static",
7468
"mutable statics can be mutated by multiple threads: aliasing violations or data \

compiler/rustc_mir/src/borrow_check/region_infer/mod.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1241,7 +1241,7 @@ impl<'tcx> RegionInferenceContext<'tcx> {
12411241
/// it. However, it works pretty well in practice. In particular,
12421242
/// this is needed to deal with projection outlives bounds like
12431243
///
1244-
/// ```ignore (internal compiler representation so lifetime syntax is invalid)
1244+
/// ```text
12451245
/// <T as Foo<'0>>::Item: '1
12461246
/// ```
12471247
///

compiler/rustc_mir_build/src/check_unsafety.rs

+22-37
Original file line numberDiff line numberDiff line change
@@ -64,38 +64,30 @@ impl<'tcx> UnsafetyVisitor<'tcx> {
6464
SafetyContext::UnsafeFn if unsafe_op_in_unsafe_fn_allowed => {}
6565
SafetyContext::UnsafeFn => {
6666
// unsafe_op_in_unsafe_fn is disallowed
67-
if kind == BorrowOfPackedField {
68-
// FIXME handle borrows of packed fields
69-
} else {
70-
struct_span_err!(
71-
self.tcx.sess,
72-
span,
73-
E0133,
74-
"{} is unsafe and requires unsafe block",
75-
description,
76-
)
77-
.span_label(span, description)
78-
.note(note)
79-
.emit();
80-
}
67+
struct_span_err!(
68+
self.tcx.sess,
69+
span,
70+
E0133,
71+
"{} is unsafe and requires unsafe block",
72+
description,
73+
)
74+
.span_label(span, description)
75+
.note(note)
76+
.emit();
8177
}
8278
SafetyContext::Safe => {
83-
if kind == BorrowOfPackedField {
84-
// FIXME handle borrows of packed fields
85-
} else {
86-
let fn_sugg = if unsafe_op_in_unsafe_fn_allowed { " function or" } else { "" };
87-
struct_span_err!(
88-
self.tcx.sess,
89-
span,
90-
E0133,
91-
"{} is unsafe and requires unsafe{} block",
92-
description,
93-
fn_sugg,
94-
)
95-
.span_label(span, description)
96-
.note(note)
97-
.emit();
98-
}
79+
let fn_sugg = if unsafe_op_in_unsafe_fn_allowed { " function or" } else { "" };
80+
struct_span_err!(
81+
self.tcx.sess,
82+
span,
83+
E0133,
84+
"{} is unsafe and requires unsafe{} block",
85+
description,
86+
fn_sugg,
87+
)
88+
.span_label(span, description)
89+
.note(note)
90+
.emit();
9991
}
10092
}
10193
}
@@ -203,8 +195,6 @@ enum UnsafeOpKind {
203195
#[allow(dead_code)] // FIXME
204196
CastOfPointerToInt,
205197
#[allow(dead_code)] // FIXME
206-
BorrowOfPackedField,
207-
#[allow(dead_code)] // FIXME
208198
UseOfMutableStatic,
209199
#[allow(dead_code)] // FIXME
210200
UseOfExternStatic,
@@ -244,11 +234,6 @@ impl UnsafeOpKind {
244234
CastOfPointerToInt => {
245235
("cast of pointer to int", "casting pointers to integers in constants")
246236
}
247-
BorrowOfPackedField => (
248-
"borrow of packed field",
249-
"fields of packed structs might be misaligned: dereferencing a misaligned pointer \
250-
or even just creating a misaligned reference is undefined behavior",
251-
),
252237
UseOfMutableStatic => (
253238
"use of mutable static",
254239
"mutable statics can be mutated by multiple threads: aliasing violations or data \

compiler/rustc_passes/src/lang_items.rs

+124-3
Original file line numberDiff line numberDiff line change
@@ -13,12 +13,13 @@ use crate::weak_lang_items;
1313
use rustc_middle::middle::cstore::ExternCrate;
1414
use rustc_middle::ty::TyCtxt;
1515

16-
use rustc_errors::struct_span_err;
16+
use rustc_errors::{pluralize, struct_span_err};
1717
use rustc_hir as hir;
1818
use rustc_hir::def_id::DefId;
1919
use rustc_hir::itemlikevisit::ItemLikeVisitor;
2020
use rustc_hir::lang_items::{extract, ITEM_REFS};
2121
use rustc_hir::{HirId, LangItem, LanguageItems, Target};
22+
use rustc_span::Span;
2223

2324
use rustc_middle::ty::query::Providers;
2425

@@ -61,8 +62,7 @@ impl LanguageItemCollector<'tcx> {
6162
match ITEM_REFS.get(&value).cloned() {
6263
// Known lang item with attribute on correct target.
6364
Some((item_index, expected_target)) if actual_target == expected_target => {
64-
let def_id = self.tcx.hir().local_def_id(hir_id);
65-
self.collect_item(item_index, def_id.to_def_id());
65+
self.collect_item_extended(item_index, hir_id, span);
6666
}
6767
// Known lang item with attribute on incorrect target.
6868
Some((_, expected_target)) => {
@@ -180,6 +180,127 @@ impl LanguageItemCollector<'tcx> {
180180
self.items.groups[group as usize].push(item_def_id);
181181
}
182182
}
183+
184+
// Like collect_item() above, but also checks whether the lang item is declared
185+
// with the right number of generic arguments if it is a trait.
186+
fn collect_item_extended(&mut self, item_index: usize, hir_id: HirId, span: Span) {
187+
let item_def_id = self.tcx.hir().local_def_id(hir_id).to_def_id();
188+
let lang_item = LangItem::from_u32(item_index as u32).unwrap();
189+
let name = lang_item.name();
190+
191+
self.collect_item(item_index, item_def_id);
192+
193+
// Now check whether the lang_item has the expected number of generic
194+
// arguments if it is a trait. Generally speaking, binary and indexing
195+
// operations have one (for the RHS/index), unary operations have none,
196+
// and the rest also have none except for the closure traits (one for
197+
// the argument list), generators (one for the resume argument),
198+
// ordering/equality relations (one for the RHS), and various conversion
199+
// traits.
200+
201+
let expected_num = match lang_item {
202+
// Binary operations
203+
LangItem::Add
204+
| LangItem::Sub
205+
| LangItem::Mul
206+
| LangItem::Div
207+
| LangItem::Rem
208+
| LangItem::BitXor
209+
| LangItem::BitAnd
210+
| LangItem::BitOr
211+
| LangItem::Shl
212+
| LangItem::Shr
213+
| LangItem::AddAssign
214+
| LangItem::SubAssign
215+
| LangItem::MulAssign
216+
| LangItem::DivAssign
217+
| LangItem::RemAssign
218+
| LangItem::BitXorAssign
219+
| LangItem::BitAndAssign
220+
| LangItem::BitOrAssign
221+
| LangItem::ShlAssign
222+
| LangItem::ShrAssign
223+
| LangItem::Index
224+
| LangItem::IndexMut
225+
226+
// Miscellaneous
227+
| LangItem::Unsize
228+
| LangItem::CoerceUnsized
229+
| LangItem::DispatchFromDyn
230+
| LangItem::Fn
231+
| LangItem::FnMut
232+
| LangItem::FnOnce
233+
| LangItem::Generator
234+
| LangItem::PartialEq
235+
| LangItem::PartialOrd
236+
=> Some(1),
237+
238+
// Unary operations
239+
LangItem::Neg
240+
| LangItem::Not
241+
242+
// Miscellaneous
243+
| LangItem::Deref
244+
| LangItem::DerefMut
245+
| LangItem::Sized
246+
| LangItem::StructuralPeq
247+
| LangItem::StructuralTeq
248+
| LangItem::Copy
249+
| LangItem::Clone
250+
| LangItem::Sync
251+
| LangItem::DiscriminantKind
252+
| LangItem::PointeeTrait
253+
| LangItem::Freeze
254+
| LangItem::Drop
255+
| LangItem::Receiver
256+
| LangItem::Future
257+
| LangItem::Unpin
258+
| LangItem::Termination
259+
| LangItem::Try
260+
| LangItem::Send
261+
| LangItem::UnwindSafe
262+
| LangItem::RefUnwindSafe
263+
=> Some(0),
264+
265+
// Not a trait
266+
_ => None,
267+
};
268+
269+
if let Some(expected_num) = expected_num {
270+
let (actual_num, generics_span) = match self.tcx.hir().get(hir_id) {
271+
hir::Node::Item(hir::Item {
272+
kind: hir::ItemKind::Trait(_, _, generics, ..),
273+
..
274+
}) => (generics.params.len(), generics.span),
275+
_ => bug!("op/index/deref lang item target is not a trait: {:?}", lang_item),
276+
};
277+
278+
if expected_num != actual_num {
279+
// We are issuing E0718 "incorrect target" here, because while the
280+
// item kind of the target is correct, the target is still wrong
281+
// because of the wrong number of generic arguments.
282+
struct_span_err!(
283+
self.tcx.sess,
284+
span,
285+
E0718,
286+
"`{}` language item must be applied to a trait with {} generic argument{}",
287+
name,
288+
expected_num,
289+
pluralize!(expected_num)
290+
)
291+
.span_label(
292+
generics_span,
293+
format!(
294+
"this trait has {} generic argument{}, not {}",
295+
actual_num,
296+
pluralize!(actual_num),
297+
expected_num
298+
),
299+
)
300+
.emit();
301+
}
302+
}
303+
}
183304
}
184305

185306
/// Traverses and collects all the lang items in all crates.

compiler/rustc_trait_selection/src/opaque_types.rs

+1
Original file line numberDiff line numberDiff line change
@@ -46,6 +46,7 @@ pub struct OpaqueTypeDecl<'tcx> {
4646
/// type Foo = impl Baz;
4747
/// fn bar() -> Foo {
4848
/// // ^^^ This is the span we are looking for!
49+
/// }
4950
/// ```
5051
///
5152
/// In cases where the fn returns `(impl Trait, impl Trait)` or

compiler/rustc_trait_selection/src/traits/error_reporting/suggestions.rs

+68-13
Original file line numberDiff line numberDiff line change
@@ -686,17 +686,36 @@ impl<'a, 'tcx> InferCtxtExt<'tcx> for InferCtxt<'a, 'tcx> {
686686
return false;
687687
}
688688

689+
// Blacklist traits for which it would be nonsensical to suggest borrowing.
690+
// For instance, immutable references are always Copy, so suggesting to
691+
// borrow would always succeed, but it's probably not what the user wanted.
692+
let blacklist: Vec<_> =
693+
[LangItem::Copy, LangItem::Clone, LangItem::Unpin, LangItem::Sized, LangItem::Send]
694+
.iter()
695+
.filter_map(|lang_item| self.tcx.lang_items().require(*lang_item).ok())
696+
.collect();
697+
689698
let span = obligation.cause.span;
690699
let param_env = obligation.param_env;
691700
let trait_ref = trait_ref.skip_binder();
692701

693-
if let ObligationCauseCode::ImplDerivedObligation(obligation) = &obligation.cause.code {
694-
// Try to apply the original trait binding obligation by borrowing.
695-
let self_ty = trait_ref.self_ty();
696-
let found = self_ty.to_string();
697-
let new_self_ty = self.tcx.mk_imm_ref(self.tcx.lifetimes.re_static, self_ty);
698-
let substs = self.tcx.mk_substs_trait(new_self_ty, &[]);
699-
let new_trait_ref = ty::TraitRef::new(obligation.parent_trait_ref.def_id(), substs);
702+
let found_ty = trait_ref.self_ty();
703+
let found_ty_str = found_ty.to_string();
704+
let imm_borrowed_found_ty = self.tcx.mk_imm_ref(self.tcx.lifetimes.re_static, found_ty);
705+
let imm_substs = self.tcx.mk_substs_trait(imm_borrowed_found_ty, &[]);
706+
let mut_borrowed_found_ty = self.tcx.mk_mut_ref(self.tcx.lifetimes.re_static, found_ty);
707+
let mut_substs = self.tcx.mk_substs_trait(mut_borrowed_found_ty, &[]);
708+
709+
// Try to apply the original trait binding obligation by borrowing.
710+
let mut try_borrowing = |new_trait_ref: ty::TraitRef<'tcx>,
711+
expected_trait_ref: ty::TraitRef<'tcx>,
712+
mtbl: bool,
713+
blacklist: &[DefId]|
714+
-> bool {
715+
if blacklist.contains(&expected_trait_ref.def_id) {
716+
return false;
717+
}
718+
700719
let new_obligation = Obligation::new(
701720
ObligationCause::dummy(),
702721
param_env,
@@ -713,8 +732,8 @@ impl<'a, 'tcx> InferCtxtExt<'tcx> for InferCtxt<'a, 'tcx> {
713732

714733
let msg = format!(
715734
"the trait bound `{}: {}` is not satisfied",
716-
found,
717-
obligation.parent_trait_ref.skip_binder().print_only_trait_path(),
735+
found_ty_str,
736+
expected_trait_ref.print_only_trait_path(),
718737
);
719738
if has_custom_message {
720739
err.note(&msg);
@@ -730,7 +749,7 @@ impl<'a, 'tcx> InferCtxtExt<'tcx> for InferCtxt<'a, 'tcx> {
730749
span,
731750
&format!(
732751
"expected an implementor of trait `{}`",
733-
obligation.parent_trait_ref.skip_binder().print_only_trait_path(),
752+
expected_trait_ref.print_only_trait_path(),
734753
),
735754
);
736755

@@ -745,16 +764,52 @@ impl<'a, 'tcx> InferCtxtExt<'tcx> for InferCtxt<'a, 'tcx> {
745764

746765
err.span_suggestion(
747766
span,
748-
"consider borrowing here",
749-
format!("&{}", snippet),
767+
&format!(
768+
"consider{} borrowing here",
769+
if mtbl { " mutably" } else { "" }
770+
),
771+
format!("&{}{}", if mtbl { "mut " } else { "" }, snippet),
750772
Applicability::MaybeIncorrect,
751773
);
752774
}
753775
return true;
754776
}
755777
}
778+
return false;
779+
};
780+
781+
if let ObligationCauseCode::ImplDerivedObligation(obligation) = &obligation.cause.code {
782+
let expected_trait_ref = obligation.parent_trait_ref.skip_binder();
783+
let new_imm_trait_ref =
784+
ty::TraitRef::new(obligation.parent_trait_ref.def_id(), imm_substs);
785+
let new_mut_trait_ref =
786+
ty::TraitRef::new(obligation.parent_trait_ref.def_id(), mut_substs);
787+
if try_borrowing(new_imm_trait_ref, expected_trait_ref, false, &[]) {
788+
return true;
789+
} else {
790+
return try_borrowing(new_mut_trait_ref, expected_trait_ref, true, &[]);
791+
}
792+
} else if let ObligationCauseCode::BindingObligation(_, _)
793+
| ObligationCauseCode::ItemObligation(_) = &obligation.cause.code
794+
{
795+
if try_borrowing(
796+
ty::TraitRef::new(trait_ref.def_id, imm_substs),
797+
trait_ref,
798+
false,
799+
&blacklist[..],
800+
) {
801+
return true;
802+
} else {
803+
return try_borrowing(
804+
ty::TraitRef::new(trait_ref.def_id, mut_substs),
805+
trait_ref,
806+
true,
807+
&blacklist[..],
808+
);
809+
}
810+
} else {
811+
false
756812
}
757-
false
758813
}
759814

760815
/// Whenever references are used by mistake, like `for (i, e) in &vec.iter().enumerate()`,

0 commit comments

Comments
 (0)