1use crate::sym;
2use rustc_ast::Attribute;
3use rustc_ast::attr::AttributeExt;
4use rustc_attr_parsing::parse_version;
5use rustc_data_structures::smallvec::SmallVec;
6use rustc_hir::RustcVersion;
7use rustc_lint::LateContext;
8use rustc_session::Session;
9use rustc_span::Symbol;
10use serde::Deserialize;
11use std::iter::once;
12use std::sync::atomic::{AtomicBool, Ordering};
13
14macro_rules! msrv_aliases {
15 ($($major:literal,$minor:literal,$patch:literal {
16 $($name:ident),* $(,)?
17 })*) => {
18 $($(
19 pub const $name: RustcVersion = RustcVersion { major: $major, minor :$minor, patch: $patch };
20 )*)*
21 };
22}
23
24msrv_aliases! {
26 1,97,0 { ISOLATE_LOWEST_ONE }
27 1,93,0 { VEC_DEQUE_POP_BACK_IF, VEC_DEQUE_POP_FRONT_IF }
28 1,91,0 { DURATION_FROM_MINUTES_HOURS }
29 1,88,0 { LET_CHAINS }
30 1,87,0 { OS_STR_DISPLAY, INT_MIDPOINT, CONST_CHAR_IS_DIGIT, UNSIGNED_IS_MULTIPLE_OF, INTEGER_SIGN_CAST }
31 1,86,0 { VEC_POP_IF }
32 1,85,0 { UINT_FLOAT_MIDPOINT, CONST_SIZE_OF_VAL, WAKER_NOOP }
33 1,84,0 { CONST_OPTION_AS_SLICE, MANUAL_DANGLING_PTR }
34 1,83,0 { CONST_EXTERN_FN, CONST_FLOAT_BITS_CONV, CONST_FLOAT_CLASSIFY, CONST_MUT_REFS, CONST_UNWRAP }
35 1,82,0 { IS_NONE_OR, REPEAT_N, RAW_REF_OP, SPECIALIZED_TO_STRING_FOR_REFS }
36 1,81,0 { LINT_REASONS_STABILIZATION, ERROR_IN_CORE, EXPLICIT_SELF_TYPE_ELISION, DURATION_ABS_DIFF }
37 1,80,0 { BOX_INTO_ITER, LAZY_CELL }
38 1,79,0 { CONST_BLOCKS, CSTR_COUNT_BYTES }
39 1,77,0 { C_STR_LITERALS }
40 1,76,0 { PTR_FROM_REF, OPTION_RESULT_INSPECT }
41 1,75,0 { OPTION_AS_SLICE }
42 1,74,0 { REPR_RUST, IO_ERROR_OTHER }
43 1,73,0 { DIV_CEIL }
44 1,71,0 { TUPLE_ARRAY_CONVERSIONS, BUILD_HASHER_HASH_ONE }
45 1,70,0 { OPTION_RESULT_IS_VARIANT_AND, BINARY_HEAP_RETAIN }
46 1,68,0 { PATH_MAIN_SEPARATOR_STR }
47 1,67,0 { ILOG2 }
48 1,65,0 { LET_ELSE, POINTER_CAST_CONSTNESS }
49 1,63,0 { CLONE_INTO, CONST_SLICE_FROM_REF }
50 1,62,0 { BOOL_THEN_SOME, DEFAULT_ENUM_ATTRIBUTE, CONST_EXTERN_C_FN }
51 1,61,0 { CONST_FN_TRAIT_BOUND }
52 1,60,0 { ABS_DIFF }
53 1,59,0 { THREAD_LOCAL_CONST_INIT }
54 1,58,0 { FORMAT_ARGS_CAPTURE, PATTERN_TRAIT_CHAR_ARRAY, CONST_RAW_PTR_DEREF }
55 1,57,0 { MAP_WHILE, CONST_PANIC }
56 1,56,0 { CONST_FN_UNION }
57 1,55,0 { SEEK_REWIND }
58 1,54,0 { INTO_KEYS }
59 1,53,0 { OR_PATTERNS, INTEGER_BITS, BTREE_MAP_RETAIN, BTREE_SET_RETAIN, ARRAY_INTO_ITERATOR }
60 1,52,0 { STR_SPLIT_ONCE, REM_EUCLID_CONST }
61 1,51,0 { BORROW_AS_PTR, SEEK_FROM_CURRENT, UNSIGNED_ABS }
62 1,50,0 { BOOL_THEN, CLAMP, SLICE_FILL }
63 1,47,0 { TAU, IS_ASCII_DIGIT_CONST, ARRAY_IMPL_ANY_LEN, SATURATING_SUB_CONST }
64 1,46,0 { CONST_IF_MATCH, OPTION_ZIP }
65 1,45,0 { STR_STRIP_PREFIX }
66 1,43,0 { LOG2_10, LOG10_2, NUMERIC_ASSOCIATED_CONSTANTS }
67 1,42,0 { MATCHES_MACRO, SLICE_PATTERNS, PTR_SLICE_RAW_PARTS }
68 1,41,0 { RE_REBALANCING_COHERENCE, RESULT_MAP_OR, RESULT_MAP_OR_ELSE }
69 1,40,0 { MEM_TAKE, NON_EXHAUSTIVE, OPTION_AS_DEREF }
70 1,38,0 { POINTER_CAST, REM_EUCLID }
71 1,37,0 { TYPE_ALIAS_ENUM_VARIANTS }
72 1,36,0 { ITERATOR_COPIED }
73 1,35,0 { OPTION_COPIED, RANGE_CONTAINS }
74 1,34,0 { TRY_FROM }
75 1,33,0 { UNDERSCORE_IMPORTS }
76 1,32,0 { CONST_IS_POWER_OF_TWO, CONST_DURATION_FROM_NANOS_MICROS_MILLIS_SECS }
77 1,31,0 { OPTION_REPLACE }
78 1,30,0 { ITERATOR_FIND_MAP, TOOL_ATTRIBUTES }
79 1,29,0 { ITER_FLATTEN }
80 1,28,0 { FROM_BOOL, REPEAT_WITH, SLICE_FROM_REF }
81 1,27,0 { ITERATOR_TRY_FOLD, DOUBLE_ENDED_ITERATOR_RFIND, DURATION_FROM_NANOS_MICROS }
82 1,26,0 { RANGE_INCLUSIVE, STRING_RETAIN, POINTER_ADD_SUB_METHODS }
83 1,24,0 { IS_ASCII_DIGIT, PTR_NULL }
84 1,18,0 { HASH_MAP_RETAIN, HASH_SET_RETAIN }
85 1,17,0 { FIELD_INIT_SHORTHAND, STATIC_IN_CONST, EXPECT_ERR }
86 1,16,0 { STR_REPEAT, RESULT_UNWRAP_OR_DEFAULT }
87 1,15,0 { MAYBE_BOUND_IN_WHERE }
88 1,13,0 { QUESTION_MARK_OPERATOR }
89 1,3,0 { DURATION_FROM_MILLIS_SECS }
90}
91
92static SEEN_MSRV_ATTR: AtomicBool = AtomicBool::new(false);
96
97#[derive(Copy, Clone, Debug, Default)]
100pub struct Msrv(Option<RustcVersion>);
101
102impl<'de> Deserialize<'de> for Msrv {
103 fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
104 where
105 D: serde::Deserializer<'de>,
106 {
107 let v = String::deserialize(deserializer)?;
108 parse_version(Symbol::intern(&v))
109 .map(|v| Self(Some(v)))
110 .ok_or_else(|| serde::de::Error::custom("not a valid Rust version"))
111 }
112}
113
114impl Msrv {
115 pub fn current(self, cx: &LateContext<'_>) -> Option<RustcVersion> {
120 if SEEN_MSRV_ATTR.load(Ordering::Relaxed) {
121 let start = cx.last_node_with_lint_attrs;
122 if let Some(msrv_attr) = once(start)
123 .chain(cx.tcx.hir_parent_id_iter(start))
124 .find_map(|id| parse_attrs(cx.tcx.sess, cx.tcx.hir_attrs(id)))
125 {
126 return Some(msrv_attr);
127 }
128 }
129
130 self.0
131 }
132
133 pub fn meets(self, cx: &LateContext<'_>, required: RustcVersion) -> bool {
138 self.current(cx).is_none_or(|msrv| msrv >= required)
139 }
140
141 pub fn read_cargo(&mut self, sess: &Session) {
142 let cargo_msrv = std::env::var("CARGO_PKG_RUST_VERSION")
143 .ok()
144 .and_then(|v| parse_version(Symbol::intern(&v)));
145
146 match (self.0, cargo_msrv) {
147 (None, Some(cargo_msrv)) => self.0 = Some(cargo_msrv),
148 (Some(clippy_msrv), Some(cargo_msrv)) if clippy_msrv != cargo_msrv => {
149 sess.dcx().warn(format!(
150 "the MSRV in `clippy.toml` and `Cargo.toml` differ; using `{clippy_msrv}` from `clippy.toml`"
151 ));
152 },
153 _ => {},
154 }
155 }
156}
157
158#[derive(Debug, Clone)]
161pub struct MsrvStack {
162 stack: SmallVec<[RustcVersion; 2]>,
163}
164
165impl MsrvStack {
166 pub fn new(initial: Msrv) -> Self {
167 Self {
168 stack: SmallVec::from_iter(initial.0),
169 }
170 }
171
172 pub fn current(&self) -> Option<RustcVersion> {
173 self.stack.last().copied()
174 }
175
176 pub fn meets(&self, required: RustcVersion) -> bool {
177 self.current().is_none_or(|msrv| msrv >= required)
178 }
179
180 pub fn check_attributes(&mut self, sess: &Session, attrs: &[Attribute]) {
181 if let Some(version) = parse_attrs(sess, attrs) {
182 SEEN_MSRV_ATTR.store(true, Ordering::Relaxed);
183 self.stack.push(version);
184 }
185 }
186
187 pub fn check_attributes_post(&mut self, sess: &Session, attrs: &[Attribute]) {
188 if parse_attrs(sess, attrs).is_some() {
189 self.stack.pop();
190 }
191 }
192}
193
194fn parse_attrs(sess: &Session, attrs: &[impl AttributeExt]) -> Option<RustcVersion> {
195 let mut msrv_attrs = attrs.iter().filter(|attr| attr.path_matches(&[sym::clippy, sym::msrv]));
196
197 let msrv_attr = msrv_attrs.next()?;
198
199 if let Some(duplicate) = msrv_attrs.next_back() {
200 sess.dcx()
201 .struct_span_err(duplicate.span(), "`clippy::msrv` is defined multiple times")
202 .with_span_note(msrv_attr.span(), "first definition found here")
203 .emit();
204 }
205
206 let Some(msrv) = msrv_attr.value_str() else {
207 sess.dcx().span_err(msrv_attr.span(), "bad clippy attribute");
208 return None;
209 };
210
211 let Some(version) = parse_version(msrv) else {
212 sess.dcx()
213 .span_err(msrv_attr.span(), format!("`{msrv}` is not a valid Rust version"));
214 return None;
215 };
216
217 Some(version)
218}