Skip to content

Commit 5bb72cc

Browse files
committed
Improve duplicate derive Copy/Clone diagnostics
1 parent 360f7d7 commit 5bb72cc

File tree

3 files changed

+50
-0
lines changed

3 files changed

+50
-0
lines changed

compiler/rustc_borrowck/src/diagnostics/move_errors.rs

+17
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@ use rustc_middle::ty::{self, Ty};
1010
use rustc_mir_dataflow::move_paths::{LookupResult, MovePathIndex};
1111
use rustc_span::{BytePos, ExpnKind, MacroKind, Span};
1212
use rustc_trait_selection::error_reporting::traits::FindExprBySpan;
13+
use rustc_trait_selection::infer::InferCtxtExt;
1314
use tracing::debug;
1415

1516
use crate::MirBorrowckCtxt;
@@ -267,6 +268,15 @@ impl<'infcx, 'tcx> MirBorrowckCtxt<'_, 'infcx, 'tcx> {
267268
kind,
268269
self.is_upvar_field_projection(original_path.as_ref())
269270
);
271+
if self.has_ambiguous_copy(original_path.ty(self.body, self.infcx.tcx).ty) {
272+
// If the type may implement Copy, skip the error.
273+
// It's an error with the Copy implementation (e.g. duplicate Copy) rather than borrow check
274+
self.dcx().span_delayed_bug(
275+
span,
276+
"Type may implement copy, but there is no other error.",
277+
);
278+
return;
279+
}
270280
(
271281
match kind {
272282
&IllegalMoveOriginKind::BorrowedContent { target_place } => self
@@ -291,6 +301,13 @@ impl<'infcx, 'tcx> MirBorrowckCtxt<'_, 'infcx, 'tcx> {
291301
self.buffer_error(err);
292302
}
293303

304+
fn has_ambiguous_copy(&mut self, ty: Ty<'tcx>) -> bool {
305+
let Some(copy_trait_def) = self.infcx.tcx.lang_items().copy_trait() else { return false };
306+
// This is only going to be ambiguous if there are incoherent impls, because otherwise
307+
// ambiguity should never happen in MIR.
308+
self.infcx.type_implements_trait(copy_trait_def, [ty], self.param_env).may_apply()
309+
}
310+
294311
fn report_cannot_move_from_static(&mut self, place: Place<'tcx>, span: Span) -> Diag<'infcx> {
295312
let description = if place.projection.len() == 1 {
296313
format!("static item {}", self.describe_any_place(place.as_ref()))
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
// Duplicate implementations of Copy/Clone should not trigger
2+
// borrow check warnings
3+
// See #131083
4+
5+
#[derive(Copy, Clone)]
6+
#[derive(Copy, Clone)]
7+
//~^ ERROR conflicting implementations of trait `Copy`
8+
enum E {}
9+
10+
fn main() {}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
error[E0119]: conflicting implementations of trait `Copy` for type `E`
2+
--> $DIR/duplicate-derive-copy-clone-diagnostics.rs:2:10
3+
|
4+
LL | #[derive(Copy, Clone)]
5+
| ---- first implementation here
6+
LL | #[derive(Copy, Clone)]
7+
| ^^^^ conflicting implementation for `E`
8+
|
9+
= note: this error originates in the derive macro `Copy` (in Nightly builds, run with -Z macro-backtrace for more info)
10+
11+
error[E0119]: conflicting implementations of trait `Clone` for type `E`
12+
--> $DIR/duplicate-derive-copy-clone-diagnostics.rs:2:16
13+
|
14+
LL | #[derive(Copy, Clone)]
15+
| ----- first implementation here
16+
LL | #[derive(Copy, Clone)]
17+
| ^^^^^ conflicting implementation for `E`
18+
|
19+
= note: this error originates in the derive macro `Clone` (in Nightly builds, run with -Z macro-backtrace for more info)
20+
21+
error: aborting due to 2 previous errors
22+
23+
For more information about this error, try `rustc --explain E0119`.

0 commit comments

Comments
 (0)