Skip to content

Commit 4a765c0

Browse files
committed
Auto merge of rust-lang#129618 - Jarcho:backports, r=Mark-Simulacrum
[beta] Clippy backports r? `@Mark-Simulacrum` Backports: - rust-lang/rust-clippy#12892 - rust-lang/rust-clippy#13168 - rust-lang/rust-clippy#13290 Both 12892 and 13290 are fixes for stable regressions. The first is a little large, but mostly just reverts the change that caused the regression. 13168 is to handle the `Error` trait moving to core in 1.81.
2 parents b5fd9f6 + 1f4f9d8 commit 4a765c0

11 files changed

+165
-31
lines changed

src/tools/clippy/clippy_lints/src/lib.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1034,7 +1034,7 @@ pub fn register_lints(store: &mut rustc_lint::LintStore, conf: &'static Conf) {
10341034
allow_comparison_to_zero,
10351035
))
10361036
});
1037-
store.register_late_pass(|_| Box::<std_instead_of_core::StdReexports>::default());
1037+
store.register_late_pass(move |_| Box::new(std_instead_of_core::StdReexports::new(conf)));
10381038
store.register_late_pass(move |_| Box::new(instant_subtraction::InstantSubtraction::new(msrv())));
10391039
store.register_late_pass(|_| Box::new(partialeq_to_none::PartialeqToNone));
10401040
store.register_late_pass(move |_| Box::new(manual_clamp::ManualClamp::new(msrv())));

src/tools/clippy/clippy_lints/src/needless_borrows_for_generic_args.rs

+43-5
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
use clippy_config::msrvs::{self, Msrv};
22
use clippy_utils::diagnostics::span_lint_and_then;
3-
use clippy_utils::mir::PossibleBorrowerMap;
3+
use clippy_utils::mir::{enclosing_mir, expr_local, local_assignments, used_exactly_once, PossibleBorrowerMap};
44
use clippy_utils::source::snippet_with_context;
55
use clippy_utils::ty::{implements_trait, is_copy};
66
use clippy_utils::{expr_use_ctxt, peel_n_hir_expr_refs, DefinedTy, ExprUseNode};
@@ -11,6 +11,7 @@ use rustc_hir::{Body, Expr, ExprKind, Mutability, Path, QPath};
1111
use rustc_index::bit_set::BitSet;
1212
use rustc_infer::infer::TyCtxtInferExt;
1313
use rustc_lint::{LateContext, LateLintPass};
14+
use rustc_middle::mir::{Rvalue, StatementKind};
1415
use rustc_middle::ty::{
1516
self, ClauseKind, EarlyBinder, FnSig, GenericArg, GenericArgKind, ParamTy, ProjectionPredicate, Ty,
1617
};
@@ -107,6 +108,7 @@ impl<'tcx> LateLintPass<'tcx> for NeedlessBorrowsForGenericArgs<'tcx> {
107108
}
108109
&& let count = needless_borrow_count(
109110
cx,
111+
&mut self.possible_borrowers,
110112
fn_id,
111113
cx.typeck_results().node_args(hir_id),
112114
i,
@@ -155,9 +157,14 @@ fn path_has_args(p: &QPath<'_>) -> bool {
155157
/// The following constraints will be checked:
156158
/// * The borrowed expression meets all the generic type's constraints.
157159
/// * The generic type appears only once in the functions signature.
158-
/// * The borrowed value is Copy itself OR not a variable (created by a function call)
160+
/// * The borrowed value is:
161+
/// - `Copy` itself, or
162+
/// - the only use of a mutable reference, or
163+
/// - not a variable (created by a function call)
164+
#[expect(clippy::too_many_arguments)]
159165
fn needless_borrow_count<'tcx>(
160166
cx: &LateContext<'tcx>,
167+
possible_borrowers: &mut Vec<(LocalDefId, PossibleBorrowerMap<'tcx, 'tcx>)>,
161168
fn_id: DefId,
162169
callee_args: ty::GenericArgsRef<'tcx>,
163170
arg_index: usize,
@@ -232,9 +239,9 @@ fn needless_borrow_count<'tcx>(
232239

233240
let referent_ty = cx.typeck_results().expr_ty(referent);
234241

235-
if (!is_copy(cx, referent_ty) && !referent_ty.is_ref())
236-
&& let ExprKind::AddrOf(_, _, inner) = reference.kind
237-
&& !matches!(inner.kind, ExprKind::Call(..) | ExprKind::MethodCall(..))
242+
if !(is_copy(cx, referent_ty)
243+
|| referent_ty.is_ref() && referent_used_exactly_once(cx, possible_borrowers, reference)
244+
|| matches!(referent.kind, ExprKind::Call(..) | ExprKind::MethodCall(..)))
238245
{
239246
return false;
240247
}
@@ -337,6 +344,37 @@ fn is_mixed_projection_predicate<'tcx>(
337344
}
338345
}
339346

347+
fn referent_used_exactly_once<'tcx>(
348+
cx: &LateContext<'tcx>,
349+
possible_borrowers: &mut Vec<(LocalDefId, PossibleBorrowerMap<'tcx, 'tcx>)>,
350+
reference: &Expr<'tcx>,
351+
) -> bool {
352+
if let Some(mir) = enclosing_mir(cx.tcx, reference.hir_id)
353+
&& let Some(local) = expr_local(cx.tcx, reference)
354+
&& let [location] = *local_assignments(mir, local).as_slice()
355+
&& let block_data = &mir.basic_blocks[location.block]
356+
&& let Some(statement) = block_data.statements.get(location.statement_index)
357+
&& let StatementKind::Assign(box (_, Rvalue::Ref(_, _, place))) = statement.kind
358+
&& !place.is_indirect_first_projection()
359+
{
360+
let body_owner_local_def_id = cx.tcx.hir().enclosing_body_owner(reference.hir_id);
361+
if possible_borrowers
362+
.last()
363+
.map_or(true, |&(local_def_id, _)| local_def_id != body_owner_local_def_id)
364+
{
365+
possible_borrowers.push((body_owner_local_def_id, PossibleBorrowerMap::new(cx, mir)));
366+
}
367+
let possible_borrower = &mut possible_borrowers.last_mut().unwrap().1;
368+
// If `only_borrowers` were used here, the `copyable_iterator::warn` test would fail. The reason is
369+
// that `PossibleBorrowerVisitor::visit_terminator` considers `place.local` a possible borrower of
370+
// itself. See the comment in that method for an explanation as to why.
371+
possible_borrower.bounded_borrowers(&[local], &[local, place.local], place.local, location)
372+
&& used_exactly_once(mir, place.local).unwrap_or(false)
373+
} else {
374+
false
375+
}
376+
}
377+
340378
// Iteratively replaces `param_ty` with `new_ty` in `args`, and similarly for each resulting
341379
// projected type that is a type parameter. Returns `false` if replacing the types would have an
342380
// effect on the function signature beyond substituting `new_ty` for `param_ty`.

src/tools/clippy/clippy_lints/src/non_copy_const.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -309,7 +309,7 @@ impl<'tcx> NonCopyConst<'tcx> {
309309

310310
impl<'tcx> LateLintPass<'tcx> for NonCopyConst<'tcx> {
311311
fn check_crate(&mut self, cx: &LateContext<'tcx>) {
312-
self.interior_mut = InteriorMut::new(cx, &self.ignore_interior_mutability);
312+
self.interior_mut = InteriorMut::without_pointers(cx, &self.ignore_interior_mutability);
313313
}
314314

315315
fn check_item(&mut self, cx: &LateContext<'tcx>, it: &'tcx Item<'_>) {

src/tools/clippy/clippy_lints/src/std_instead_of_core.rs

+43-10
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,15 @@
1+
use clippy_config::msrvs::Msrv;
2+
use clippy_config::Conf;
13
use clippy_utils::diagnostics::span_lint_and_sugg;
24
use clippy_utils::is_from_proc_macro;
5+
use rustc_attr::{StabilityLevel, StableSince};
36
use rustc_errors::Applicability;
47
use rustc_hir::def::Res;
58
use rustc_hir::def_id::DefId;
69
use rustc_hir::{HirId, Path, PathSegment};
710
use rustc_lint::{LateContext, LateLintPass, LintContext};
811
use rustc_middle::lint::in_external_macro;
12+
use rustc_semver::RustcVersion;
913
use rustc_session::impl_lint_pass;
1014
use rustc_span::symbol::kw;
1115
use rustc_span::{sym, Span};
@@ -66,6 +70,10 @@ declare_clippy_lint! {
6670
/// imported from core to ensure disabling `alloc` does not cause the crate to fail to compile. This lint
6771
/// is also useful for crates migrating to become `no_std` compatible.
6872
///
73+
/// ### Known problems
74+
/// The lint is only partially aware of the required MSRV for items that were originally in `std` but moved
75+
/// to `core`.
76+
///
6977
/// ### Example
7078
/// ```no_run
7179
/// # extern crate alloc;
@@ -81,20 +89,30 @@ declare_clippy_lint! {
8189
"type is imported from alloc when available in core"
8290
}
8391

84-
#[derive(Default)]
8592
pub struct StdReexports {
8693
// Paths which can be either a module or a macro (e.g. `std::env`) will cause this check to happen
8794
// twice. First for the mod, second for the macro. This is used to avoid the lint reporting for the macro
8895
// when the path could be also be used to access the module.
8996
prev_span: Span,
97+
msrv: Msrv,
98+
}
99+
100+
impl StdReexports {
101+
pub fn new(conf: &'static Conf) -> Self {
102+
Self {
103+
prev_span: Span::default(),
104+
msrv: conf.msrv.clone(),
105+
}
106+
}
90107
}
108+
91109
impl_lint_pass!(StdReexports => [STD_INSTEAD_OF_CORE, STD_INSTEAD_OF_ALLOC, ALLOC_INSTEAD_OF_CORE]);
92110

93111
impl<'tcx> LateLintPass<'tcx> for StdReexports {
94112
fn check_path(&mut self, cx: &LateContext<'tcx>, path: &Path<'tcx>, _: HirId) {
95113
if let Res::Def(_, def_id) = path.res
96114
&& let Some(first_segment) = get_first_segment(path)
97-
&& is_stable(cx, def_id)
115+
&& is_stable(cx, def_id, &self.msrv)
98116
&& !in_external_macro(cx.sess(), path.span)
99117
&& !is_from_proc_macro(cx, &first_segment.ident)
100118
{
@@ -131,6 +149,8 @@ impl<'tcx> LateLintPass<'tcx> for StdReexports {
131149
}
132150
}
133151
}
152+
153+
extract_msrv_attr!(LateContext);
134154
}
135155

136156
/// Returns the first named segment of a [`Path`].
@@ -146,16 +166,29 @@ fn get_first_segment<'tcx>(path: &Path<'tcx>) -> Option<&'tcx PathSegment<'tcx>>
146166
}
147167
}
148168

149-
/// Checks if all ancestors of `def_id` are stable, to avoid linting
150-
/// [unstable moves](https://github.com/rust-lang/rust/pull/95956)
151-
fn is_stable(cx: &LateContext<'_>, mut def_id: DefId) -> bool {
169+
/// Checks if all ancestors of `def_id` meet `msrv` to avoid linting [unstable moves](https://github.com/rust-lang/rust/pull/95956)
170+
/// or now stable moves that were once unstable.
171+
///
172+
/// Does not catch individually moved items
173+
fn is_stable(cx: &LateContext<'_>, mut def_id: DefId, msrv: &Msrv) -> bool {
152174
loop {
153-
if cx
154-
.tcx
155-
.lookup_stability(def_id)
156-
.map_or(false, |stability| stability.is_unstable())
175+
if let Some(stability) = cx.tcx.lookup_stability(def_id)
176+
&& let StabilityLevel::Stable {
177+
since,
178+
allowed_through_unstable_modules: false,
179+
} = stability.level
157180
{
158-
return false;
181+
let stable = match since {
182+
StableSince::Version(v) => {
183+
msrv.meets(RustcVersion::new(v.major.into(), v.minor.into(), v.patch.into()))
184+
},
185+
StableSince::Current => msrv.current().is_none(),
186+
StableSince::Err => false,
187+
};
188+
189+
if !stable {
190+
return false;
191+
}
159192
}
160193

161194
match cx.tcx.opt_parent(def_id) {

src/tools/clippy/tests/ui/declare_interior_mutable_const/others.rs

+17
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@
33
use std::borrow::Cow;
44
use std::cell::Cell;
55
use std::fmt::Display;
6+
use std::ptr;
67
use std::sync::atomic::AtomicUsize;
78
use std::sync::Once;
89

@@ -53,4 +54,20 @@ mod issue_8493 {
5354
issue_8493!();
5455
}
5556

57+
#[repr(C, align(8))]
58+
struct NoAtomic(usize);
59+
#[repr(C, align(8))]
60+
struct WithAtomic(AtomicUsize);
61+
62+
const fn with_non_null() -> *const WithAtomic {
63+
const NO_ATOMIC: NoAtomic = NoAtomic(0);
64+
(&NO_ATOMIC as *const NoAtomic).cast()
65+
}
66+
const WITH_ATOMIC: *const WithAtomic = with_non_null();
67+
68+
struct Generic<T>(T);
69+
impl<T> Generic<T> {
70+
const RAW_POINTER: *const Cell<T> = ptr::null();
71+
}
72+
5673
fn main() {}

src/tools/clippy/tests/ui/declare_interior_mutable_const/others.stderr

+5-5
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
error: a `const` item should not be interior mutable
2-
--> tests/ui/declare_interior_mutable_const/others.rs:9:1
2+
--> tests/ui/declare_interior_mutable_const/others.rs:10:1
33
|
44
LL | const ATOMIC: AtomicUsize = AtomicUsize::new(5);
55
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
@@ -9,23 +9,23 @@ LL | const ATOMIC: AtomicUsize = AtomicUsize::new(5);
99
= help: to override `-D warnings` add `#[allow(clippy::declare_interior_mutable_const)]`
1010

1111
error: a `const` item should not be interior mutable
12-
--> tests/ui/declare_interior_mutable_const/others.rs:10:1
12+
--> tests/ui/declare_interior_mutable_const/others.rs:11:1
1313
|
1414
LL | const CELL: Cell<usize> = Cell::new(6);
1515
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
1616
|
1717
= help: consider making this `Sync` so that it can go in a static item or using a `thread_local`
1818

1919
error: a `const` item should not be interior mutable
20-
--> tests/ui/declare_interior_mutable_const/others.rs:11:1
20+
--> tests/ui/declare_interior_mutable_const/others.rs:12:1
2121
|
2222
LL | const ATOMIC_TUPLE: ([AtomicUsize; 1], Vec<AtomicUsize>, u8) = ([ATOMIC], Vec::new(), 7);
2323
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
2424
|
2525
= help: consider making this a static item
2626

2727
error: a `const` item should not be interior mutable
28-
--> tests/ui/declare_interior_mutable_const/others.rs:16:9
28+
--> tests/ui/declare_interior_mutable_const/others.rs:17:9
2929
|
3030
LL | const $name: $ty = $e;
3131
| ^^^^^^^^^^^^^^^^^^^^^^
@@ -36,7 +36,7 @@ LL | declare_const!(_ONCE: Once = Once::new());
3636
= note: this error originates in the macro `declare_const` (in Nightly builds, run with -Z macro-backtrace for more info)
3737

3838
error: a `const` item should not be interior mutable
39-
--> tests/ui/declare_interior_mutable_const/others.rs:44:13
39+
--> tests/ui/declare_interior_mutable_const/others.rs:45:13
4040
|
4141
LL | const _BAZ: Cell<usize> = Cell::new(0);
4242
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^

src/tools/clippy/tests/ui/needless_borrows_for_generic_args.fixed

+8
Original file line numberDiff line numberDiff line change
@@ -333,4 +333,12 @@ fn main() {
333333
f(&y); // Don't lint
334334
f("".to_owned()); // Lint
335335
}
336+
{
337+
fn takes_writer<T: std::io::Write>(_: T) {}
338+
339+
fn issue_12856(mut buffer: &mut Vec<u8>) {
340+
takes_writer(&mut buffer); // Don't lint, would make buffer unavailable later
341+
buffer.extend(b"\n");
342+
}
343+
}
336344
}

src/tools/clippy/tests/ui/needless_borrows_for_generic_args.rs

+8
Original file line numberDiff line numberDiff line change
@@ -333,4 +333,12 @@ fn main() {
333333
f(&y); // Don't lint
334334
f(&"".to_owned()); // Lint
335335
}
336+
{
337+
fn takes_writer<T: std::io::Write>(_: T) {}
338+
339+
fn issue_12856(mut buffer: &mut Vec<u8>) {
340+
takes_writer(&mut buffer); // Don't lint, would make buffer unavailable later
341+
buffer.extend(b"\n");
342+
}
343+
}
336344
}

src/tools/clippy/tests/ui/std_instead_of_core.fixed

+13-4
Original file line numberDiff line numberDiff line change
@@ -75,8 +75,17 @@ mod std_in_proc_macro_derive {
7575
struct B {}
7676
}
7777

78-
fn main() {
79-
std_instead_of_core();
80-
std_instead_of_alloc();
81-
alloc_instead_of_core();
78+
// Some intrinsics are usable on stable but live in an unstable module, but should still suggest
79+
// replacing std -> core
80+
fn intrinsic(a: *mut u8, b: *mut u8) {
81+
unsafe {
82+
core::intrinsics::copy(a, b, 1);
83+
//~^ std_instead_of_core
84+
}
8285
}
86+
87+
#[clippy::msrv = "1.76"]
88+
fn msrv_1_76(_: std::net::IpAddr) {}
89+
90+
#[clippy::msrv = "1.77"]
91+
fn msrv_1_77(_: core::net::IpAddr) {}

src/tools/clippy/tests/ui/std_instead_of_core.rs

+13-4
Original file line numberDiff line numberDiff line change
@@ -75,8 +75,17 @@ mod std_in_proc_macro_derive {
7575
struct B {}
7676
}
7777

78-
fn main() {
79-
std_instead_of_core();
80-
std_instead_of_alloc();
81-
alloc_instead_of_core();
78+
// Some intrinsics are usable on stable but live in an unstable module, but should still suggest
79+
// replacing std -> core
80+
fn intrinsic(a: *mut u8, b: *mut u8) {
81+
unsafe {
82+
std::intrinsics::copy(a, b, 1);
83+
//~^ std_instead_of_core
84+
}
8285
}
86+
87+
#[clippy::msrv = "1.76"]
88+
fn msrv_1_76(_: std::net::IpAddr) {}
89+
90+
#[clippy::msrv = "1.77"]
91+
fn msrv_1_77(_: std::net::IpAddr) {}

src/tools/clippy/tests/ui/std_instead_of_core.stderr

+13-1
Original file line numberDiff line numberDiff line change
@@ -85,5 +85,17 @@ LL | use alloc::slice::from_ref;
8585
= note: `-D clippy::alloc-instead-of-core` implied by `-D warnings`
8686
= help: to override `-D warnings` add `#[allow(clippy::alloc_instead_of_core)]`
8787

88-
error: aborting due to 13 previous errors
88+
error: used import from `std` instead of `core`
89+
--> tests/ui/std_instead_of_core.rs:82:9
90+
|
91+
LL | std::intrinsics::copy(a, b, 1);
92+
| ^^^ help: consider importing the item from `core`: `core`
93+
94+
error: used import from `std` instead of `core`
95+
--> tests/ui/std_instead_of_core.rs:91:17
96+
|
97+
LL | fn msrv_1_77(_: std::net::IpAddr) {}
98+
| ^^^ help: consider importing the item from `core`: `core`
99+
100+
error: aborting due to 15 previous errors
89101

0 commit comments

Comments
 (0)