Skip to content

Commit ee2189d

Browse files
committed
fix: missing_const_for_fn FP on unstable const traits
1 parent 162b0e8 commit ee2189d

File tree

2 files changed

+67
-16
lines changed

2 files changed

+67
-16
lines changed

clippy_utils/src/qualify_min_const_fn.rs

+22-16
Original file line numberDiff line numberDiff line change
@@ -386,24 +386,30 @@ fn check_terminator<'tcx>(
386386

387387
fn is_stable_const_fn(tcx: TyCtxt<'_>, def_id: DefId, msrv: &Msrv) -> bool {
388388
tcx.is_const_fn(def_id)
389-
&& tcx.lookup_const_stability(def_id).is_none_or(|const_stab| {
390-
if let rustc_attr_parsing::StabilityLevel::Stable { since, .. } = const_stab.level {
391-
// Checking MSRV is manually necessary because `rustc` has no such concept. This entire
392-
// function could be removed if `rustc` provided a MSRV-aware version of `is_stable_const_fn`.
393-
// as a part of an unimplemented MSRV check https://github.com/rust-lang/rust/issues/65262.
389+
&& tcx
390+
.lookup_const_stability(def_id)
391+
.or_else(|| {
392+
tcx.trait_of_item(def_id)
393+
.and_then(|trait_def_id| tcx.lookup_const_stability(trait_def_id))
394+
})
395+
.is_none_or(|const_stab| {
396+
if let rustc_attr_parsing::StabilityLevel::Stable { since, .. } = const_stab.level {
397+
// Checking MSRV is manually necessary because `rustc` has no such concept. This entire
398+
// function could be removed if `rustc` provided a MSRV-aware version of `is_stable_const_fn`.
399+
// as a part of an unimplemented MSRV check https://github.com/rust-lang/rust/issues/65262.
394400

395-
let const_stab_rust_version = match since {
396-
StableSince::Version(version) => version,
397-
StableSince::Current => RustcVersion::CURRENT,
398-
StableSince::Err => return false,
399-
};
401+
let const_stab_rust_version = match since {
402+
StableSince::Version(version) => version,
403+
StableSince::Current => RustcVersion::CURRENT,
404+
StableSince::Err => return false,
405+
};
400406

401-
msrv.meets(const_stab_rust_version)
402-
} else {
403-
// Unstable const fn, check if the feature is enabled.
404-
tcx.features().enabled(const_stab.feature) && msrv.current().is_none()
405-
}
406-
})
407+
msrv.meets(const_stab_rust_version)
408+
} else {
409+
// Unstable const fn, check if the feature is enabled.
410+
tcx.features().enabled(const_stab.feature) && msrv.current().is_none()
411+
}
412+
})
407413
}
408414

409415
fn is_ty_const_destruct<'tcx>(tcx: TyCtxt<'tcx>, ty: Ty<'tcx>, body: &Body<'tcx>) -> bool {

tests/ui/missing_const_for_fn/cant_be_const.rs

+45
Original file line numberDiff line numberDiff line change
@@ -217,3 +217,48 @@ mod with_ty_alias {
217217
fn mut_add(x: &mut i32) {
218218
*x += 1;
219219
}
220+
221+
#[clippy::msrv = "1.87"]
222+
mod issue14020 {
223+
use std::ops::Add;
224+
225+
fn f<T: Add>(a: T, b: T) -> <T as Add>::Output {
226+
a + b
227+
}
228+
}
229+
230+
#[clippy::msrv = "1.87"]
231+
mod issue14290 {
232+
use std::ops::{Deref, DerefMut};
233+
234+
struct Wrapper<T> {
235+
t: T,
236+
}
237+
238+
impl<T> Deref for Wrapper<T> {
239+
type Target = T;
240+
241+
fn deref(&self) -> &Self::Target {
242+
&self.t
243+
}
244+
}
245+
impl<T> DerefMut for Wrapper<T> {
246+
fn deref_mut(&mut self) -> &mut Self::Target {
247+
&mut self.t
248+
}
249+
}
250+
251+
struct Example(bool);
252+
253+
fn do_something(mut a: Wrapper<Example>) {
254+
a.0 = !a.0;
255+
}
256+
257+
pub struct Stream(Vec<u8>);
258+
259+
impl Stream {
260+
pub fn bytes(&self) -> &[u8] {
261+
&self.0
262+
}
263+
}
264+
}

0 commit comments

Comments
 (0)