Skip to content

Commit 8b459dd

Browse files
committed
Use span of ignored impls for explanatory note
1 parent bd1f09d commit 8b459dd

File tree

4 files changed

+52
-39
lines changed

4 files changed

+52
-39
lines changed

compiler/rustc_passes/src/dead.rs

+27-19
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@ use rustc_middle::middle::privacy;
1616
use rustc_middle::ty::{self, DefIdTree, TyCtxt};
1717
use rustc_session::lint;
1818
use rustc_span::symbol::{sym, Symbol};
19+
use rustc_span::Span;
1920
use std::mem;
2021

2122
// Any local node that may call something in its body block should be
@@ -49,7 +50,9 @@ struct MarkSymbolVisitor<'tcx> {
4950
// maps from tuple struct constructors to tuple struct items
5051
struct_constructors: FxHashMap<LocalDefId, LocalDefId>,
5152
// maps from ADTs to ignored derived traits (e.g. Debug and Clone)
52-
ignored_derived_traits: FxHashMap<DefId, Vec<DefId>>,
53+
// and the span of their respective impl (i.e., part of the derive
54+
// macro)
55+
ignored_derived_traits: FxHashMap<DefId, Vec<(Span, DefId)>>,
5356
}
5457

5558
impl<'tcx> MarkSymbolVisitor<'tcx> {
@@ -255,10 +258,12 @@ impl<'tcx> MarkSymbolVisitor<'tcx> {
255258
if self.tcx.has_attr(trait_of, sym::rustc_trivial_field_reads) {
256259
let trait_ref = self.tcx.impl_trait_ref(impl_of).unwrap();
257260
if let ty::Adt(adt_def, _) = trait_ref.self_ty().kind() {
261+
let impl_span = self.tcx.def_span(impl_of);
258262
if let Some(v) = self.ignored_derived_traits.get_mut(&adt_def.did) {
259-
v.push(trait_of);
263+
v.push((impl_span, trait_of));
260264
} else {
261-
self.ignored_derived_traits.insert(adt_def.did, vec![trait_of]);
265+
self.ignored_derived_traits
266+
.insert(adt_def.did, vec![(impl_span, trait_of)]);
262267
}
263268
}
264269
return true;
@@ -588,7 +593,7 @@ fn create_and_seed_worklist<'tcx>(
588593
fn find_live<'tcx>(
589594
tcx: TyCtxt<'tcx>,
590595
access_levels: &privacy::AccessLevels,
591-
) -> (FxHashSet<LocalDefId>, FxHashMap<DefId, Vec<DefId>>) {
596+
) -> (FxHashSet<LocalDefId>, FxHashMap<DefId, Vec<(Span, DefId)>>) {
592597
let (worklist, struct_constructors) = create_and_seed_worklist(tcx, access_levels);
593598
let mut symbol_visitor = MarkSymbolVisitor {
594599
worklist,
@@ -610,7 +615,7 @@ fn find_live<'tcx>(
610615
struct DeadVisitor<'tcx> {
611616
tcx: TyCtxt<'tcx>,
612617
live_symbols: FxHashSet<LocalDefId>,
613-
ignored_derived_traits: FxHashMap<DefId, Vec<DefId>>,
618+
ignored_derived_traits: FxHashMap<DefId, Vec<(Span, DefId)>>,
614619
}
615620

616621
impl<'tcx> DeadVisitor<'tcx> {
@@ -683,26 +688,29 @@ impl<'tcx> DeadVisitor<'tcx> {
683688
let hir = self.tcx.hir();
684689
if let Some(encl_scope) = hir.get_enclosing_scope(id) {
685690
if let Some(encl_def_id) = hir.opt_local_def_id(encl_scope) {
686-
if let Some(ign_traits) = self.ignored_derived_traits.get(&encl_def_id.to_def_id()) {
691+
if let Some(ign_traits) =
692+
self.ignored_derived_traits.get(&encl_def_id.to_def_id())
693+
{
687694
let traits_str = ign_traits
688695
.iter()
689-
.map(|t| format!("`{}`", self.tcx.item_name(*t))).collect::<Vec<_>>()
696+
.map(|(_, t)| format!("`{}`", self.tcx.item_name(*t)))
697+
.collect::<Vec<_>>()
690698
.join(" and ");
691699
let plural_s = pluralize!(ign_traits.len());
692700
let article = if ign_traits.len() > 1 { "" } else { "a " };
693701
let is_are = if ign_traits.len() > 1 { "these are" } else { "this is" };
694-
let msg = format!("`{}` has {}derived impl{} for the trait{} {}, but {} ignored during dead code analysis",
695-
self.tcx.item_name(encl_def_id.to_def_id()),
696-
article,
697-
plural_s,
698-
plural_s,
699-
traits_str,
700-
is_are);
701-
if let Some(span) = self.tcx.def_ident_span(encl_def_id) {
702-
err.span_note(span, &msg);
703-
} else {
704-
err.note(&msg);
705-
}
702+
let msg = format!(
703+
"`{}` has {}derived impl{} for the trait{} {}, but {} \
704+
intentionally ignored during dead code analysis",
705+
self.tcx.item_name(encl_def_id.to_def_id()),
706+
article,
707+
plural_s,
708+
plural_s,
709+
traits_str,
710+
is_are
711+
);
712+
let multispan = ign_traits.iter().map(|(s, _)| *s).collect::<Vec<_>>();
713+
err.span_note(multispan, &msg);
706714
}
707715
}
708716
}

src/test/ui/derive-uninhabited-enum-38885.stderr

+5-4
Original file line numberDiff line numberDiff line change
@@ -5,11 +5,12 @@ LL | Void(Void),
55
| ^^^^^^^^^^
66
|
77
= note: `-W dead-code` implied by `-W unused`
8-
note: `Foo` has a derived impl for the trait `Debug`, but this is ignored during dead code analysis
9-
--> $DIR/derive-uninhabited-enum-38885.rs:11:6
8+
note: `Foo` has a derived impl for the trait `Debug`, but this is intentionally ignored during dead code analysis
9+
--> $DIR/derive-uninhabited-enum-38885.rs:10:10
1010
|
11-
LL | enum Foo {
12-
| ^^^
11+
LL | #[derive(Debug)]
12+
| ^^^^^
13+
= note: this warning originates in the derive macro `Debug` (in Nightly builds, run with -Z macro-backtrace for more info)
1314

1415
warning: 1 warning emitted
1516

src/test/ui/derives/clone-debug-dead-code.stderr

+15-12
Original file line numberDiff line numberDiff line change
@@ -16,35 +16,38 @@ error: field is never read: `f`
1616
LL | struct B { f: () }
1717
| ^^^^^
1818
|
19-
note: `B` has a derived impl for the trait `Clone`, but this is ignored during dead code analysis
20-
--> $DIR/clone-debug-dead-code.rs:10:8
19+
note: `B` has a derived impl for the trait `Clone`, but this is intentionally ignored during dead code analysis
20+
--> $DIR/clone-debug-dead-code.rs:9:10
2121
|
22-
LL | struct B { f: () }
23-
| ^
22+
LL | #[derive(Clone)]
23+
| ^^^^^
24+
= note: this error originates in the derive macro `Clone` (in Nightly builds, run with -Z macro-backtrace for more info)
2425

2526
error: field is never read: `f`
2627
--> $DIR/clone-debug-dead-code.rs:14:12
2728
|
2829
LL | struct C { f: () }
2930
| ^^^^^
3031
|
31-
note: `C` has a derived impl for the trait `Debug`, but this is ignored during dead code analysis
32-
--> $DIR/clone-debug-dead-code.rs:14:8
32+
note: `C` has a derived impl for the trait `Debug`, but this is intentionally ignored during dead code analysis
33+
--> $DIR/clone-debug-dead-code.rs:13:10
3334
|
34-
LL | struct C { f: () }
35-
| ^
35+
LL | #[derive(Debug)]
36+
| ^^^^^
37+
= note: this error originates in the derive macro `Debug` (in Nightly builds, run with -Z macro-backtrace for more info)
3638

3739
error: field is never read: `f`
3840
--> $DIR/clone-debug-dead-code.rs:18:12
3941
|
4042
LL | struct D { f: () }
4143
| ^^^^^
4244
|
43-
note: `D` has derived impls for the traits `Clone` and `Debug`, but these are ignored during dead code analysis
44-
--> $DIR/clone-debug-dead-code.rs:18:8
45+
note: `D` has derived impls for the traits `Clone` and `Debug`, but these are intentionally ignored during dead code analysis
46+
--> $DIR/clone-debug-dead-code.rs:17:10
4547
|
46-
LL | struct D { f: () }
47-
| ^
48+
LL | #[derive(Debug,Clone)]
49+
| ^^^^^ ^^^^^
50+
= note: this error originates in the derive macro `Debug` (in Nightly builds, run with -Z macro-backtrace for more info)
4851

4952
error: field is never read: `f`
5053
--> $DIR/clone-debug-dead-code.rs:21:12

src/test/ui/lint/dead-code/unused-variant.stderr

+5-4
Original file line numberDiff line numberDiff line change
@@ -9,11 +9,12 @@ note: the lint level is defined here
99
|
1010
LL | #![deny(dead_code)]
1111
| ^^^^^^^^^
12-
note: `Enum` has a derived impl for the trait `Clone`, but this is ignored during dead code analysis
13-
--> $DIR/unused-variant.rs:4:6
12+
note: `Enum` has a derived impl for the trait `Clone`, but this is intentionally ignored during dead code analysis
13+
--> $DIR/unused-variant.rs:3:10
1414
|
15-
LL | enum Enum {
16-
| ^^^^
15+
LL | #[derive(Clone)]
16+
| ^^^^^
17+
= note: this error originates in the derive macro `Clone` (in Nightly builds, run with -Z macro-backtrace for more info)
1718

1819
error: aborting due to previous error
1920

0 commit comments

Comments
 (0)