Skip to main content

rustc_lint_defs/
lib.rs

1use std::borrow::Cow;
2use std::fmt::Display;
3
4use rustc_data_structures::fx::FxIndexSet;
5use rustc_data_structures::stable_hash::{StableCompare, StableHash, StableHashCtxt, StableHasher};
6use rustc_error_messages::{DiagArgValue, IntoDiagArg};
7use rustc_hir_id::HirId;
8use rustc_macros::{Decodable, Encodable, StableHash};
9pub use rustc_span::edition::Edition;
10use rustc_span::{AttrId, Ident, Symbol, sym};
11use serde::{Deserialize, Serialize};
12
13pub use self::Level::*;
14
15pub mod builtin;
16
17#[macro_export]
18macro_rules! pluralize {
19    // Pluralize based on count (e.g., apples)
20    ($x:expr) => {
21        if $x == 1 { "" } else { "s" }
22    };
23    ("has", $x:expr) => {
24        if $x == 1 { "has" } else { "have" }
25    };
26    ("is", $x:expr) => {
27        if $x == 1 { "is" } else { "are" }
28    };
29    ("was", $x:expr) => {
30        if $x == 1 { "was" } else { "were" }
31    };
32    ("this", $x:expr) => {
33        if $x == 1 { "this" } else { "these" }
34    };
35}
36
37/// Grammatical tool for displaying messages to end users in a nice form.
38///
39/// Take a list of items and a function to turn those items into a `String`, and output a display
40/// friendly comma separated list of those items.
41// FIXME(estebank): this needs to be changed to go through the translation machinery.
42pub fn listify<T>(list: &[T], fmt: impl Fn(&T) -> String) -> Option<String> {
43    Some(match list {
44        [only] => fmt(&only),
45        [others @ .., last] => ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("{0} and {1}",
                others.iter().map(|i| fmt(i)).collect::<Vec<_>>().join(", "),
                fmt(&last)))
    })format!(
46            "{} and {}",
47            others.iter().map(|i| fmt(i)).collect::<Vec<_>>().join(", "),
48            fmt(&last),
49        ),
50        [] => return None,
51    })
52}
53
54/// Indicates the confidence in the correctness of a suggestion.
55///
56/// All suggestions are marked with an `Applicability`. Tools use the applicability of a suggestion
57/// to determine whether it should be automatically applied or if the user should be consulted
58/// before applying the suggestion.
59#[derive(#[automatically_derived]
impl ::core::marker::Copy for Applicability { }Copy, #[automatically_derived]
impl ::core::clone::Clone for Applicability {
    #[inline]
    fn clone(&self) -> Applicability { *self }
}Clone, #[automatically_derived]
impl ::core::fmt::Debug for Applicability {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::write_str(f,
            match self {
                Applicability::MachineApplicable => "MachineApplicable",
                Applicability::MaybeIncorrect => "MaybeIncorrect",
                Applicability::HasPlaceholders => "HasPlaceholders",
                Applicability::Unspecified => "Unspecified",
            })
    }
}Debug, #[automatically_derived]
impl ::core::hash::Hash for Applicability {
    #[inline]
    fn hash<__H: ::core::hash::Hasher>(&self, state: &mut __H) {
        let __self_discr = ::core::intrinsics::discriminant_value(self);
        ::core::hash::Hash::hash(&__self_discr, state)
    }
}Hash, const _: () =
    {
        impl<__E: ::rustc_span::SpanEncoder> ::rustc_serialize::Encodable<__E>
            for Applicability {
            fn encode(&self, __encoder: &mut __E) {
                let disc =
                    match *self {
                        Applicability::MachineApplicable => { 0usize }
                        Applicability::MaybeIncorrect => { 1usize }
                        Applicability::HasPlaceholders => { 2usize }
                        Applicability::Unspecified => { 3usize }
                    };
                ::rustc_serialize::Encoder::emit_u8(__encoder, disc as u8);
                match *self {
                    Applicability::MachineApplicable => {}
                    Applicability::MaybeIncorrect => {}
                    Applicability::HasPlaceholders => {}
                    Applicability::Unspecified => {}
                }
            }
        }
    };Encodable, const _: () =
    {
        impl<__D: ::rustc_span::SpanDecoder> ::rustc_serialize::Decodable<__D>
            for Applicability {
            fn decode(__decoder: &mut __D) -> Self {
                match ::rustc_serialize::Decoder::read_u8(__decoder) as usize
                    {
                    0usize => { Applicability::MachineApplicable }
                    1usize => { Applicability::MaybeIncorrect }
                    2usize => { Applicability::HasPlaceholders }
                    3usize => { Applicability::Unspecified }
                    n => {
                        ::core::panicking::panic_fmt(format_args!("invalid enum variant tag while decoding `Applicability`, expected 0..4, actual {0}",
                                n));
                    }
                }
            }
        }
    };Decodable, #[doc(hidden)]
#[allow(non_upper_case_globals, unused_attributes, unused_qualifications,
clippy :: absolute_paths,)]
const _: () =
    {
        #[allow(unused_extern_crates, clippy :: useless_attribute)]
        extern crate serde as _serde;
        ;
        #[automatically_derived]
        impl _serde::Serialize for Applicability {
            fn serialize<__S>(&self, __serializer: __S)
                -> _serde::__private228::Result<__S::Ok, __S::Error> where
                __S: _serde::Serializer {
                match *self {
                    Applicability::MachineApplicable =>
                        _serde::Serializer::serialize_unit_variant(__serializer,
                            "Applicability", 0u32, "MachineApplicable"),
                    Applicability::MaybeIncorrect =>
                        _serde::Serializer::serialize_unit_variant(__serializer,
                            "Applicability", 1u32, "MaybeIncorrect"),
                    Applicability::HasPlaceholders =>
                        _serde::Serializer::serialize_unit_variant(__serializer,
                            "Applicability", 2u32, "HasPlaceholders"),
                    Applicability::Unspecified =>
                        _serde::Serializer::serialize_unit_variant(__serializer,
                            "Applicability", 3u32, "Unspecified"),
                }
            }
        }
    };Serialize, #[doc(hidden)]
#[allow(non_upper_case_globals, unused_attributes, unused_qualifications,
clippy :: absolute_paths,)]
const _: () =
    {
        #[allow(unused_extern_crates, clippy :: useless_attribute)]
        extern crate serde as _serde;
        ;
        #[automatically_derived]
        impl<'de> _serde::Deserialize<'de> for Applicability {
            fn deserialize<__D>(__deserializer: __D)
                -> _serde::__private228::Result<Self, __D::Error> where
                __D: _serde::Deserializer<'de> {
                #[allow(non_camel_case_types)]
                #[doc(hidden)]
                enum __Field { __field0, __field1, __field2, __field3, }
                #[doc(hidden)]
                struct __FieldVisitor;
                #[automatically_derived]
                impl<'de> _serde::de::Visitor<'de> for __FieldVisitor {
                    type Value = __Field;
                    fn expecting(&self,
                        __formatter: &mut _serde::__private228::Formatter)
                        -> _serde::__private228::fmt::Result {
                        _serde::__private228::Formatter::write_str(__formatter,
                            "variant identifier")
                    }
                    fn visit_u64<__E>(self, __value: u64)
                        -> _serde::__private228::Result<Self::Value, __E> where
                        __E: _serde::de::Error {
                        match __value {
                            0u64 => _serde::__private228::Ok(__Field::__field0),
                            1u64 => _serde::__private228::Ok(__Field::__field1),
                            2u64 => _serde::__private228::Ok(__Field::__field2),
                            3u64 => _serde::__private228::Ok(__Field::__field3),
                            _ =>
                                _serde::__private228::Err(_serde::de::Error::invalid_value(_serde::de::Unexpected::Unsigned(__value),
                                        &"variant index 0 <= i < 4")),
                        }
                    }
                    fn visit_str<__E>(self, __value: &str)
                        -> _serde::__private228::Result<Self::Value, __E> where
                        __E: _serde::de::Error {
                        match __value {
                            "MachineApplicable" =>
                                _serde::__private228::Ok(__Field::__field0),
                            "MaybeIncorrect" =>
                                _serde::__private228::Ok(__Field::__field1),
                            "HasPlaceholders" =>
                                _serde::__private228::Ok(__Field::__field2),
                            "Unspecified" =>
                                _serde::__private228::Ok(__Field::__field3),
                            _ => {
                                _serde::__private228::Err(_serde::de::Error::unknown_variant(__value,
                                        VARIANTS))
                            }
                        }
                    }
                    fn visit_bytes<__E>(self, __value: &[u8])
                        -> _serde::__private228::Result<Self::Value, __E> where
                        __E: _serde::de::Error {
                        match __value {
                            b"MachineApplicable" =>
                                _serde::__private228::Ok(__Field::__field0),
                            b"MaybeIncorrect" =>
                                _serde::__private228::Ok(__Field::__field1),
                            b"HasPlaceholders" =>
                                _serde::__private228::Ok(__Field::__field2),
                            b"Unspecified" =>
                                _serde::__private228::Ok(__Field::__field3),
                            _ => {
                                let __value =
                                    &_serde::__private228::from_utf8_lossy(__value);
                                _serde::__private228::Err(_serde::de::Error::unknown_variant(__value,
                                        VARIANTS))
                            }
                        }
                    }
                }
                #[automatically_derived]
                impl<'de> _serde::Deserialize<'de> for __Field {
                    #[inline]
                    fn deserialize<__D>(__deserializer: __D)
                        -> _serde::__private228::Result<Self, __D::Error> where
                        __D: _serde::Deserializer<'de> {
                        _serde::Deserializer::deserialize_identifier(__deserializer,
                            __FieldVisitor)
                    }
                }
                #[doc(hidden)]
                struct __Visitor<'de> {
                    marker: _serde::__private228::PhantomData<Applicability>,
                    lifetime: _serde::__private228::PhantomData<&'de ()>,
                }
                #[automatically_derived]
                impl<'de> _serde::de::Visitor<'de> for __Visitor<'de> {
                    type Value = Applicability;
                    fn expecting(&self,
                        __formatter: &mut _serde::__private228::Formatter)
                        -> _serde::__private228::fmt::Result {
                        _serde::__private228::Formatter::write_str(__formatter,
                            "enum Applicability")
                    }
                    fn visit_enum<__A>(self, __data: __A)
                        -> _serde::__private228::Result<Self::Value, __A::Error>
                        where __A: _serde::de::EnumAccess<'de> {
                        match _serde::de::EnumAccess::variant(__data)? {
                            (__Field::__field0, __variant) => {
                                _serde::de::VariantAccess::unit_variant(__variant)?;
                                _serde::__private228::Ok(Applicability::MachineApplicable)
                            }
                            (__Field::__field1, __variant) => {
                                _serde::de::VariantAccess::unit_variant(__variant)?;
                                _serde::__private228::Ok(Applicability::MaybeIncorrect)
                            }
                            (__Field::__field2, __variant) => {
                                _serde::de::VariantAccess::unit_variant(__variant)?;
                                _serde::__private228::Ok(Applicability::HasPlaceholders)
                            }
                            (__Field::__field3, __variant) => {
                                _serde::de::VariantAccess::unit_variant(__variant)?;
                                _serde::__private228::Ok(Applicability::Unspecified)
                            }
                        }
                    }
                }
                #[doc(hidden)]
                const VARIANTS: &'static [&'static str] =
                    &["MachineApplicable", "MaybeIncorrect", "HasPlaceholders",
                                "Unspecified"];
                _serde::Deserializer::deserialize_enum(__deserializer,
                    "Applicability", VARIANTS,
                    __Visitor {
                        marker: _serde::__private228::PhantomData::<Applicability>,
                        lifetime: _serde::__private228::PhantomData,
                    })
            }
        }
    };Deserialize)]
60#[derive(#[automatically_derived]
impl ::core::cmp::PartialEq for Applicability {
    #[inline]
    fn eq(&self, other: &Applicability) -> bool {
        let __self_discr = ::core::intrinsics::discriminant_value(self);
        let __arg1_discr = ::core::intrinsics::discriminant_value(other);
        __self_discr == __arg1_discr
    }
}PartialEq, #[automatically_derived]
impl ::core::cmp::Eq for Applicability {
    #[inline]
    #[doc(hidden)]
    #[coverage(off)]
    fn assert_fields_are_eq(&self) {}
}Eq, #[automatically_derived]
impl ::core::cmp::PartialOrd for Applicability {
    #[inline]
    fn partial_cmp(&self, other: &Applicability)
        -> ::core::option::Option<::core::cmp::Ordering> {
        let __self_discr = ::core::intrinsics::discriminant_value(self);
        let __arg1_discr = ::core::intrinsics::discriminant_value(other);
        ::core::cmp::PartialOrd::partial_cmp(&__self_discr, &__arg1_discr)
    }
}PartialOrd, #[automatically_derived]
impl ::core::cmp::Ord for Applicability {
    #[inline]
    fn cmp(&self, other: &Applicability) -> ::core::cmp::Ordering {
        let __self_discr = ::core::intrinsics::discriminant_value(self);
        let __arg1_discr = ::core::intrinsics::discriminant_value(other);
        ::core::cmp::Ord::cmp(&__self_discr, &__arg1_discr)
    }
}Ord)]
61pub enum Applicability {
62    /// The suggestion is definitely what the user intended, or maintains the exact meaning of the code.
63    /// This suggestion should be automatically applied.
64    ///
65    /// In case of multiple `MachineApplicable` suggestions (whether as part of
66    /// the same `multipart_suggestion` or not), all of them should be
67    /// automatically applied.
68    MachineApplicable,
69
70    /// The suggestion may be what the user intended, but it is uncertain. The suggestion should
71    /// result in valid Rust code if it is applied.
72    MaybeIncorrect,
73
74    /// The suggestion contains placeholders like `(...)` or `{ /* fields */ }`. The suggestion
75    /// cannot be applied automatically because it will not result in valid Rust code. The user
76    /// will need to fill in the placeholders.
77    HasPlaceholders,
78
79    /// The applicability of the suggestion is unknown.
80    Unspecified,
81}
82
83/// Each lint expectation has a `LintExpectationId` assigned by the `LintLevelsBuilder`.
84/// Expected diagnostics get the lint level `Expect` which stores the `LintExpectationId`
85/// to match it with the actual expectation later on.
86///
87/// The `LintExpectationId` has to be stable between compilations, as diagnostic
88/// instances might be loaded from cache. Lint messages can be emitted during an
89/// `EarlyLintPass` operating on the AST and during a `LateLintPass` traversing the
90/// HIR tree. The AST doesn't have enough information to create a stable id. The
91/// `LintExpectationId` will instead store the [`AttrId`] defining the expectation.
92/// These `LintExpectationId` will be updated to use the stable [`HirId`] once the
93/// AST has been lowered. The transformation is done by the `LintLevelsBuilder`
94///
95/// Each lint inside the `expect` attribute is tracked individually, the `lint_index`
96/// identifies the lint inside the attribute and ensures that the IDs are unique.
97///
98/// The index values have a type of `u16` to reduce the size of the `LintExpectationId`.
99/// It's reasonable to assume that no user will define 2^16 attributes on one node or
100/// have that amount of lints listed. `u16` values should therefore suffice.
101#[derive(#[automatically_derived]
impl ::core::clone::Clone for LintExpectationId {
    #[inline]
    fn clone(&self) -> LintExpectationId {
        let _: ::core::clone::AssertParamIsClone<UnstableLintExpectationId>;
        let _: ::core::clone::AssertParamIsClone<StableLintExpectationId>;
        *self
    }
}Clone, #[automatically_derived]
impl ::core::marker::Copy for LintExpectationId { }Copy, #[automatically_derived]
impl ::core::cmp::PartialEq for LintExpectationId {
    #[inline]
    fn eq(&self, other: &LintExpectationId) -> bool {
        let __self_discr = ::core::intrinsics::discriminant_value(self);
        let __arg1_discr = ::core::intrinsics::discriminant_value(other);
        __self_discr == __arg1_discr &&
            match (self, other) {
                (LintExpectationId::Unstable(__self_0),
                    LintExpectationId::Unstable(__arg1_0)) =>
                    __self_0 == __arg1_0,
                (LintExpectationId::Stable(__self_0),
                    LintExpectationId::Stable(__arg1_0)) =>
                    __self_0 == __arg1_0,
                _ => unsafe { ::core::intrinsics::unreachable() }
            }
    }
}PartialEq, #[automatically_derived]
impl ::core::cmp::Eq for LintExpectationId {
    #[inline]
    #[doc(hidden)]
    #[coverage(off)]
    fn assert_fields_are_eq(&self) {
        let _: ::core::cmp::AssertParamIsEq<UnstableLintExpectationId>;
        let _: ::core::cmp::AssertParamIsEq<StableLintExpectationId>;
    }
}Eq, #[automatically_derived]
impl ::core::fmt::Debug for LintExpectationId {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        match self {
            LintExpectationId::Unstable(__self_0) =>
                ::core::fmt::Formatter::debug_tuple_field1_finish(f,
                    "Unstable", &__self_0),
            LintExpectationId::Stable(__self_0) =>
                ::core::fmt::Formatter::debug_tuple_field1_finish(f, "Stable",
                    &__self_0),
        }
    }
}Debug, #[automatically_derived]
impl ::core::hash::Hash for LintExpectationId {
    #[inline]
    fn hash<__H: ::core::hash::Hasher>(&self, state: &mut __H) {
        let __self_discr = ::core::intrinsics::discriminant_value(self);
        ::core::hash::Hash::hash(&__self_discr, state);
        match self {
            LintExpectationId::Unstable(__self_0) =>
                ::core::hash::Hash::hash(__self_0, state),
            LintExpectationId::Stable(__self_0) =>
                ::core::hash::Hash::hash(__self_0, state),
        }
    }
}Hash, const _: () =
    {
        impl<__E: ::rustc_span::SpanEncoder> ::rustc_serialize::Encodable<__E>
            for LintExpectationId {
            fn encode(&self, __encoder: &mut __E) {
                let disc =
                    match *self {
                        LintExpectationId::Unstable(ref __binding_0) => { 0usize }
                        LintExpectationId::Stable(ref __binding_0) => { 1usize }
                    };
                ::rustc_serialize::Encoder::emit_u8(__encoder, disc as u8);
                match *self {
                    LintExpectationId::Unstable(ref __binding_0) => {
                        ::rustc_serialize::Encodable::<__E>::encode(__binding_0,
                            __encoder);
                    }
                    LintExpectationId::Stable(ref __binding_0) => {
                        ::rustc_serialize::Encodable::<__E>::encode(__binding_0,
                            __encoder);
                    }
                }
            }
        }
    };Encodable, const _: () =
    {
        impl<__D: ::rustc_span::SpanDecoder> ::rustc_serialize::Decodable<__D>
            for LintExpectationId {
            fn decode(__decoder: &mut __D) -> Self {
                match ::rustc_serialize::Decoder::read_u8(__decoder) as usize
                    {
                    0usize => {
                        LintExpectationId::Unstable(::rustc_serialize::Decodable::decode(__decoder))
                    }
                    1usize => {
                        LintExpectationId::Stable(::rustc_serialize::Decodable::decode(__decoder))
                    }
                    n => {
                        ::core::panicking::panic_fmt(format_args!("invalid enum variant tag while decoding `LintExpectationId`, expected 0..2, actual {0}",
                                n));
                    }
                }
            }
        }
    };Decodable)]
102pub enum LintExpectationId {
103    Unstable(UnstableLintExpectationId),
104    Stable(StableLintExpectationId),
105}
106
107/// Used for lints emitted during the `EarlyLintPass`. This id is not hash
108/// stable and should not be cached.
109#[derive(#[automatically_derived]
impl ::core::clone::Clone for UnstableLintExpectationId {
    #[inline]
    fn clone(&self) -> UnstableLintExpectationId {
        let _: ::core::clone::AssertParamIsClone<AttrId>;
        let _: ::core::clone::AssertParamIsClone<Option<u16>>;
        *self
    }
}Clone, #[automatically_derived]
impl ::core::marker::Copy for UnstableLintExpectationId { }Copy, #[automatically_derived]
impl ::core::cmp::PartialEq for UnstableLintExpectationId {
    #[inline]
    fn eq(&self, other: &UnstableLintExpectationId) -> bool {
        self.attr_id == other.attr_id && self.lint_index == other.lint_index
    }
}PartialEq, #[automatically_derived]
impl ::core::cmp::Eq for UnstableLintExpectationId {
    #[inline]
    #[doc(hidden)]
    #[coverage(off)]
    fn assert_fields_are_eq(&self) {
        let _: ::core::cmp::AssertParamIsEq<AttrId>;
        let _: ::core::cmp::AssertParamIsEq<Option<u16>>;
    }
}Eq, #[automatically_derived]
impl ::core::fmt::Debug for UnstableLintExpectationId {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::debug_struct_field2_finish(f,
            "UnstableLintExpectationId", "attr_id", &self.attr_id,
            "lint_index", &&self.lint_index)
    }
}Debug, #[automatically_derived]
impl ::core::hash::Hash for UnstableLintExpectationId {
    #[inline]
    fn hash<__H: ::core::hash::Hasher>(&self, state: &mut __H) {
        ::core::hash::Hash::hash(&self.attr_id, state);
        ::core::hash::Hash::hash(&self.lint_index, state)
    }
}Hash, const _: () =
    {
        impl<__E: ::rustc_span::SpanEncoder> ::rustc_serialize::Encodable<__E>
            for UnstableLintExpectationId {
            fn encode(&self, __encoder: &mut __E) {
                match *self {
                    UnstableLintExpectationId {
                        attr_id: ref __binding_0, lint_index: ref __binding_1 } => {
                        ::rustc_serialize::Encodable::<__E>::encode(__binding_0,
                            __encoder);
                        ::rustc_serialize::Encodable::<__E>::encode(__binding_1,
                            __encoder);
                    }
                }
            }
        }
    };Encodable, const _: () =
    {
        impl<__D: ::rustc_span::SpanDecoder> ::rustc_serialize::Decodable<__D>
            for UnstableLintExpectationId {
            fn decode(__decoder: &mut __D) -> Self {
                UnstableLintExpectationId {
                    attr_id: ::rustc_serialize::Decodable::decode(__decoder),
                    lint_index: ::rustc_serialize::Decodable::decode(__decoder),
                }
            }
        }
    };Decodable)]
110pub struct UnstableLintExpectationId {
111    pub attr_id: AttrId,
112    pub lint_index: Option<u16>,
113}
114
115impl From<UnstableLintExpectationId> for LintExpectationId {
116    fn from(id: UnstableLintExpectationId) -> LintExpectationId {
117        LintExpectationId::Unstable(id)
118    }
119}
120
121/// The [`HirId`] that the lint expectation is attached to. This id is stable
122/// and can be cached. The additional index ensures that nodes with several
123/// expectations can correctly match diagnostics to the individual expectation.
124#[derive(#[automatically_derived]
impl ::core::clone::Clone for StableLintExpectationId {
    #[inline]
    fn clone(&self) -> StableLintExpectationId {
        let _: ::core::clone::AssertParamIsClone<HirId>;
        let _: ::core::clone::AssertParamIsClone<u16>;
        let _: ::core::clone::AssertParamIsClone<Option<u16>>;
        *self
    }
}Clone, #[automatically_derived]
impl ::core::marker::Copy for StableLintExpectationId { }Copy, #[automatically_derived]
impl ::core::cmp::PartialEq for StableLintExpectationId {
    #[inline]
    fn eq(&self, other: &StableLintExpectationId) -> bool {
        self.attr_index == other.attr_index && self.hir_id == other.hir_id &&
            self.lint_index == other.lint_index
    }
}PartialEq, #[automatically_derived]
impl ::core::cmp::Eq for StableLintExpectationId {
    #[inline]
    #[doc(hidden)]
    #[coverage(off)]
    fn assert_fields_are_eq(&self) {
        let _: ::core::cmp::AssertParamIsEq<HirId>;
        let _: ::core::cmp::AssertParamIsEq<u16>;
        let _: ::core::cmp::AssertParamIsEq<Option<u16>>;
    }
}Eq, #[automatically_derived]
impl ::core::fmt::Debug for StableLintExpectationId {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::debug_struct_field3_finish(f,
            "StableLintExpectationId", "hir_id", &self.hir_id, "attr_index",
            &self.attr_index, "lint_index", &&self.lint_index)
    }
}Debug, #[automatically_derived]
impl ::core::hash::Hash for StableLintExpectationId {
    #[inline]
    fn hash<__H: ::core::hash::Hasher>(&self, state: &mut __H) {
        ::core::hash::Hash::hash(&self.hir_id, state);
        ::core::hash::Hash::hash(&self.attr_index, state);
        ::core::hash::Hash::hash(&self.lint_index, state)
    }
}Hash, const _: () =
    {
        impl<__E: ::rustc_span::SpanEncoder> ::rustc_serialize::Encodable<__E>
            for StableLintExpectationId {
            fn encode(&self, __encoder: &mut __E) {
                match *self {
                    StableLintExpectationId {
                        hir_id: ref __binding_0,
                        attr_index: ref __binding_1,
                        lint_index: ref __binding_2 } => {
                        ::rustc_serialize::Encodable::<__E>::encode(__binding_0,
                            __encoder);
                        ::rustc_serialize::Encodable::<__E>::encode(__binding_1,
                            __encoder);
                        ::rustc_serialize::Encodable::<__E>::encode(__binding_2,
                            __encoder);
                    }
                }
            }
        }
    };Encodable, const _: () =
    {
        impl<__D: ::rustc_span::SpanDecoder> ::rustc_serialize::Decodable<__D>
            for StableLintExpectationId {
            fn decode(__decoder: &mut __D) -> Self {
                StableLintExpectationId {
                    hir_id: ::rustc_serialize::Decodable::decode(__decoder),
                    attr_index: ::rustc_serialize::Decodable::decode(__decoder),
                    lint_index: ::rustc_serialize::Decodable::decode(__decoder),
                }
            }
        }
    };Decodable)]
125pub struct StableLintExpectationId {
126    pub hir_id: HirId,
127    pub attr_index: u16,
128    pub lint_index: Option<u16>,
129}
130
131impl StableHash for StableLintExpectationId {
132    #[inline]
133    fn stable_hash<Hcx: StableHashCtxt>(&self, hcx: &mut Hcx, hasher: &mut StableHasher) {
134        let StableLintExpectationId { hir_id, attr_index, lint_index } = self;
135
136        hir_id.stable_hash(hcx, hasher);
137        attr_index.stable_hash(hcx, hasher);
138        lint_index.expect("must be filled to call `stable_hash`").stable_hash(hcx, hasher);
139    }
140}
141
142impl From<StableLintExpectationId> for LintExpectationId {
143    fn from(id: StableLintExpectationId) -> LintExpectationId {
144        LintExpectationId::Stable(id)
145    }
146}
147
148/// Setting for how to handle a lint.
149///
150/// See: <https://doc.rust-lang.org/rustc/lints/levels.html>
151#[derive(
152    #[automatically_derived]
impl ::core::clone::Clone for Level {
    #[inline]
    fn clone(&self) -> Level { *self }
}Clone, #[automatically_derived]
impl ::core::marker::Copy for Level { }Copy, #[automatically_derived]
impl ::core::cmp::PartialEq for Level {
    #[inline]
    fn eq(&self, other: &Level) -> bool {
        let __self_discr = ::core::intrinsics::discriminant_value(self);
        let __arg1_discr = ::core::intrinsics::discriminant_value(other);
        __self_discr == __arg1_discr
    }
}PartialEq, #[automatically_derived]
impl ::core::cmp::PartialOrd for Level {
    #[inline]
    fn partial_cmp(&self, other: &Level)
        -> ::core::option::Option<::core::cmp::Ordering> {
        let __self_discr = ::core::intrinsics::discriminant_value(self);
        let __arg1_discr = ::core::intrinsics::discriminant_value(other);
        ::core::cmp::PartialOrd::partial_cmp(&__self_discr, &__arg1_discr)
    }
}PartialOrd, #[automatically_derived]
impl ::core::cmp::Eq for Level {
    #[inline]
    #[doc(hidden)]
    #[coverage(off)]
    fn assert_fields_are_eq(&self) {}
}Eq, #[automatically_derived]
impl ::core::cmp::Ord for Level {
    #[inline]
    fn cmp(&self, other: &Level) -> ::core::cmp::Ordering {
        let __self_discr = ::core::intrinsics::discriminant_value(self);
        let __arg1_discr = ::core::intrinsics::discriminant_value(other);
        ::core::cmp::Ord::cmp(&__self_discr, &__arg1_discr)
    }
}Ord, #[automatically_derived]
impl ::core::fmt::Debug for Level {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::write_str(f,
            match self {
                Level::Allow => "Allow",
                Level::Expect => "Expect",
                Level::Warn => "Warn",
                Level::ForceWarn => "ForceWarn",
                Level::Deny => "Deny",
                Level::Forbid => "Forbid",
            })
    }
}Debug, #[automatically_derived]
impl ::core::hash::Hash for Level {
    #[inline]
    fn hash<__H: ::core::hash::Hasher>(&self, state: &mut __H) {
        let __self_discr = ::core::intrinsics::discriminant_value(self);
        ::core::hash::Hash::hash(&__self_discr, state)
    }
}Hash, const _: () =
    {
        impl<__E: ::rustc_span::SpanEncoder> ::rustc_serialize::Encodable<__E>
            for Level {
            fn encode(&self, __encoder: &mut __E) {
                let disc =
                    match *self {
                        Level::Allow => { 0usize }
                        Level::Expect => { 1usize }
                        Level::Warn => { 2usize }
                        Level::ForceWarn => { 3usize }
                        Level::Deny => { 4usize }
                        Level::Forbid => { 5usize }
                    };
                ::rustc_serialize::Encoder::emit_u8(__encoder, disc as u8);
                match *self {
                    Level::Allow => {}
                    Level::Expect => {}
                    Level::Warn => {}
                    Level::ForceWarn => {}
                    Level::Deny => {}
                    Level::Forbid => {}
                }
            }
        }
    };Encodable, const _: () =
    {
        impl<__D: ::rustc_span::SpanDecoder> ::rustc_serialize::Decodable<__D>
            for Level {
            fn decode(__decoder: &mut __D) -> Self {
                match ::rustc_serialize::Decoder::read_u8(__decoder) as usize
                    {
                    0usize => { Level::Allow }
                    1usize => { Level::Expect }
                    2usize => { Level::Warn }
                    3usize => { Level::ForceWarn }
                    4usize => { Level::Deny }
                    5usize => { Level::Forbid }
                    n => {
                        ::core::panicking::panic_fmt(format_args!("invalid enum variant tag while decoding `Level`, expected 0..6, actual {0}",
                                n));
                    }
                }
            }
        }
    };Decodable, const _: () =
    {
        impl ::rustc_data_structures::stable_hash::StableHash for Level {
            #[inline]
            fn stable_hash<__Hcx: ::rustc_data_structures::stable_hash::StableHashCtxt>(&self,
                __hcx: &mut __Hcx,
                __hasher:
                    &mut ::rustc_data_structures::stable_hash::StableHasher) {
                ::std::mem::discriminant(self).stable_hash(__hcx, __hasher);
                match *self {
                    Level::Allow => {}
                    Level::Expect => {}
                    Level::Warn => {}
                    Level::ForceWarn => {}
                    Level::Deny => {}
                    Level::Forbid => {}
                }
            }
        }
    };StableHash
153)]
154pub enum Level {
155    /// The `allow` level will not issue any message.
156    Allow,
157    /// The `expect` level will suppress the lint message but in turn produce a message
158    /// if the lint wasn't issued in the expected scope. `Expect` should not be used as
159    /// an initial level for a lint.
160    ///
161    /// Note that this still means that the lint is enabled in this position and should
162    /// be emitted, this will in turn fulfill the expectation and suppress the lint.
163    ///
164    /// See RFC 2383.
165    ///
166    /// Requires a [`LintExpectationId`] to later link a lint emission to the actual
167    /// expectation. It can be ignored in most cases.
168    Expect,
169    /// The `warn` level will produce a warning if the lint was violated, however the
170    /// compiler will continue with its execution.
171    Warn,
172    /// This lint level is a special case of [`Warn`], that can't be overridden. This is used
173    /// to ensure that a lint can't be suppressed. This lint level can currently only be set
174    /// via the console and is therefore session specific.
175    ///
176    /// Requires a [`LintExpectationId`] to fulfill expectations marked via the
177    /// `#[expect]` attribute, that will still be suppressed due to the level.
178    ForceWarn,
179    /// The `deny` level will produce an error and stop further execution after the lint
180    /// pass is complete.
181    Deny,
182    /// `Forbid` is equivalent to the `deny` level but can't be overwritten like the previous
183    /// levels.
184    Forbid,
185}
186
187impl Level {
188    /// Converts a level to a lower-case string.
189    pub fn as_str(self) -> &'static str {
190        match self {
191            Level::Allow => "allow",
192            Level::Expect => "expect",
193            Level::Warn => "warn",
194            Level::ForceWarn => "force-warn",
195            Level::Deny => "deny",
196            Level::Forbid => "forbid",
197        }
198    }
199
200    /// Converts a lower-case string to a level. This will never construct the expect
201    /// level as that would require a [`LintExpectationId`].
202    pub fn from_str(x: &str) -> Option<Self> {
203        match x {
204            "allow" => Some(Level::Allow),
205            "warn" => Some(Level::Warn),
206            "deny" => Some(Level::Deny),
207            "forbid" => Some(Level::Forbid),
208            "expect" | _ => None,
209        }
210    }
211
212    /// Converts an `Option<Symbol>` to a level.
213    pub fn from_opt_symbol(s: Option<Symbol>) -> Option<Self> {
214        s.and_then(Self::from_symbol)
215    }
216
217    /// Converts a `Symbol` to a level.
218    pub fn from_symbol(s: Symbol) -> Option<Self> {
219        match s {
220            sym::allow => Some(Level::Allow),
221            sym::expect => Some(Level::Expect),
222            sym::warn => Some(Level::Warn),
223            sym::deny => Some(Level::Deny),
224            sym::forbid => Some(Level::Forbid),
225            _ => None,
226        }
227    }
228
229    pub fn to_cmd_flag(self) -> &'static str {
230        match self {
231            Level::Warn => "-W",
232            Level::Deny => "-D",
233            Level::Forbid => "-F",
234            Level::Allow => "-A",
235            Level::ForceWarn => "--force-warn",
236            Level::Expect => {
237                {
    ::core::panicking::panic_fmt(format_args!("internal error: entered unreachable code: {0}",
            format_args!("the expect level does not have a commandline flag")));
}unreachable!("the expect level does not have a commandline flag")
238            }
239        }
240    }
241
242    pub fn is_error(self) -> bool {
243        match self {
244            Level::Allow | Level::Expect | Level::Warn | Level::ForceWarn => false,
245            Level::Deny | Level::Forbid => true,
246        }
247    }
248}
249
250impl IntoDiagArg for Level {
251    fn into_diag_arg(self, _: &mut Option<std::path::PathBuf>) -> DiagArgValue {
252        DiagArgValue::Str(Cow::Borrowed(self.to_cmd_flag()))
253    }
254}
255
256/// Specification of a single lint.
257#[derive(#[automatically_derived]
impl ::core::marker::Copy for Lint { }Copy, #[automatically_derived]
impl ::core::clone::Clone for Lint {
    #[inline]
    fn clone(&self) -> Lint {
        let _: ::core::clone::AssertParamIsClone<&'static str>;
        let _: ::core::clone::AssertParamIsClone<Level>;
        let _: ::core::clone::AssertParamIsClone<&'static str>;
        let _: ::core::clone::AssertParamIsClone<Option<(Edition, Level)>>;
        let _: ::core::clone::AssertParamIsClone<bool>;
        let _:
                ::core::clone::AssertParamIsClone<Option<FutureIncompatibleInfo>>;
        let _: ::core::clone::AssertParamIsClone<Option<Symbol>>;
        *self
    }
}Clone, #[automatically_derived]
impl ::core::fmt::Debug for Lint {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        let names: &'static _ =
            &["name", "default_level", "desc", "edition_lint_opts",
                        "report_in_external_macro", "future_incompatible",
                        "is_externally_loaded", "feature_gate", "crate_level_only",
                        "eval_always", "ignore_deny_warnings"];
        let values: &[&dyn ::core::fmt::Debug] =
            &[&self.name, &self.default_level, &self.desc,
                        &self.edition_lint_opts, &self.report_in_external_macro,
                        &self.future_incompatible, &self.is_externally_loaded,
                        &self.feature_gate, &self.crate_level_only,
                        &self.eval_always, &&self.ignore_deny_warnings];
        ::core::fmt::Formatter::debug_struct_fields_finish(f, "Lint", names,
            values)
    }
}Debug)]
258pub struct Lint {
259    /// A string identifier for the lint.
260    ///
261    /// This identifies the lint in attributes and in command-line arguments.
262    /// In those contexts it is always lowercase, but this field is compared
263    /// in a way which is case-insensitive for ASCII characters. This allows
264    /// `declare_lint!()` invocations to follow the convention of upper-case
265    /// statics without repeating the name.
266    ///
267    /// The name is written with underscores, e.g., "unused_imports".
268    /// On the command line, underscores become dashes.
269    ///
270    /// See <https://rustc-dev-guide.rust-lang.org/diagnostics.html#lint-naming>
271    /// for naming guidelines.
272    pub name: &'static str,
273
274    /// Default level for the lint.
275    ///
276    /// See <https://rustc-dev-guide.rust-lang.org/diagnostics.html#diagnostic-levels>
277    /// for guidelines on choosing a default level.
278    pub default_level: Level,
279
280    /// Description of the lint or the issue it detects.
281    ///
282    /// e.g., "imports that are never used"
283    pub desc: &'static str,
284
285    /// Starting at the given edition, default to the given lint level. If this is `None`, then use
286    /// `default_level`.
287    pub edition_lint_opts: Option<(Edition, Level)>,
288
289    /// `true` if this lint is reported even inside expansions of external macros.
290    pub report_in_external_macro: bool,
291
292    pub future_incompatible: Option<FutureIncompatibleInfo>,
293
294    /// `true` if this lint is being loaded by another tool (e.g. Clippy).
295    pub is_externally_loaded: bool,
296
297    /// `Some` if this lint is feature gated, otherwise `None`.
298    pub feature_gate: Option<Symbol>,
299
300    pub crate_level_only: bool,
301
302    /// `true` if this lint should not be filtered out under any circustamces
303    /// (e.g. the unknown_attributes lint)
304    pub eval_always: bool,
305
306    /// `true` if this lint is unaffected by `-D warnings`
307    pub ignore_deny_warnings: bool,
308}
309
310/// Extra information for a future incompatibility lint.
311#[derive(#[automatically_derived]
impl ::core::marker::Copy for FutureIncompatibleInfo { }Copy, #[automatically_derived]
impl ::core::clone::Clone for FutureIncompatibleInfo {
    #[inline]
    fn clone(&self) -> FutureIncompatibleInfo {
        let _: ::core::clone::AssertParamIsClone<FutureIncompatibilityReason>;
        let _: ::core::clone::AssertParamIsClone<bool>;
        *self
    }
}Clone, #[automatically_derived]
impl ::core::fmt::Debug for FutureIncompatibleInfo {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::debug_struct_field3_finish(f,
            "FutureIncompatibleInfo", "reason", &self.reason,
            "explain_reason", &self.explain_reason, "report_in_deps",
            &&self.report_in_deps)
    }
}Debug)]
312pub struct FutureIncompatibleInfo {
313    /// The reason for the lint used by diagnostics to provide
314    /// the right help message
315    pub reason: FutureIncompatibilityReason,
316    /// Whether to explain the reason to the user.
317    ///
318    /// Set to false for lints that already include a more detailed
319    /// explanation.
320    pub explain_reason: bool,
321    /// If set to `true`, this will make future incompatibility warnings show up in cargo's
322    /// reports.
323    ///
324    /// When a future incompatibility warning is first inroduced, set this to `false`
325    /// (or, rather, don't override the default). This allows crate developers an opportunity
326    /// to fix the warning before blasting all dependents with a warning they can't fix
327    /// (dependents have to wait for a new release of the affected crate to be published).
328    ///
329    /// After a lint has been in this state for a while, consider setting this to true, so it
330    /// warns for everyone. It is a good signal that it is ready if you can determine that all
331    /// or most affected crates on crates.io have been updated.
332    pub report_in_deps: bool,
333}
334
335#[derive(#[automatically_derived]
impl ::core::marker::Copy for EditionFcw { }Copy, #[automatically_derived]
impl ::core::clone::Clone for EditionFcw {
    #[inline]
    fn clone(&self) -> EditionFcw {
        let _: ::core::clone::AssertParamIsClone<Edition>;
        let _: ::core::clone::AssertParamIsClone<&'static str>;
        *self
    }
}Clone, #[automatically_derived]
impl ::core::fmt::Debug for EditionFcw {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::debug_struct_field2_finish(f, "EditionFcw",
            "edition", &self.edition, "page_slug", &&self.page_slug)
    }
}Debug)]
336pub struct EditionFcw {
337    pub edition: Edition,
338    pub page_slug: &'static str,
339}
340
341#[derive(#[automatically_derived]
impl ::core::marker::Copy for ReleaseFcw { }Copy, #[automatically_derived]
impl ::core::clone::Clone for ReleaseFcw {
    #[inline]
    fn clone(&self) -> ReleaseFcw {
        let _: ::core::clone::AssertParamIsClone<usize>;
        *self
    }
}Clone, #[automatically_derived]
impl ::core::fmt::Debug for ReleaseFcw {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::debug_struct_field1_finish(f, "ReleaseFcw",
            "issue_number", &&self.issue_number)
    }
}Debug)]
342pub struct ReleaseFcw {
343    pub issue_number: usize,
344}
345
346/// The reason for future incompatibility
347///
348/// Future-incompatible lints come in roughly two categories:
349///
350/// 1. There was a mistake in the compiler (such as a soundness issue), and
351///    we're trying to fix it, but it may be a breaking change.
352/// 2. A change across an Edition boundary, typically used for the
353///    introduction of new language features that can't otherwise be
354///    introduced in a backwards-compatible way.
355///
356/// See <https://rustc-dev-guide.rust-lang.org/bug-fix-procedure.html> and
357/// <https://rustc-dev-guide.rust-lang.org/diagnostics.html#future-incompatible-lints>
358/// for more information.
359#[derive(#[automatically_derived]
impl ::core::marker::Copy for FutureIncompatibilityReason { }Copy, #[automatically_derived]
impl ::core::clone::Clone for FutureIncompatibilityReason {
    #[inline]
    fn clone(&self) -> FutureIncompatibilityReason {
        let _: ::core::clone::AssertParamIsClone<ReleaseFcw>;
        let _: ::core::clone::AssertParamIsClone<EditionFcw>;
        let _: ::core::clone::AssertParamIsClone<&'static str>;
        *self
    }
}Clone, #[automatically_derived]
impl ::core::fmt::Debug for FutureIncompatibilityReason {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        match self {
            FutureIncompatibilityReason::FutureReleaseError(__self_0) =>
                ::core::fmt::Formatter::debug_tuple_field1_finish(f,
                    "FutureReleaseError", &__self_0),
            FutureIncompatibilityReason::FutureReleaseSemanticsChange(__self_0)
                =>
                ::core::fmt::Formatter::debug_tuple_field1_finish(f,
                    "FutureReleaseSemanticsChange", &__self_0),
            FutureIncompatibilityReason::EditionError(__self_0) =>
                ::core::fmt::Formatter::debug_tuple_field1_finish(f,
                    "EditionError", &__self_0),
            FutureIncompatibilityReason::EditionSemanticsChange(__self_0) =>
                ::core::fmt::Formatter::debug_tuple_field1_finish(f,
                    "EditionSemanticsChange", &__self_0),
            FutureIncompatibilityReason::EditionAndFutureReleaseError(__self_0)
                =>
                ::core::fmt::Formatter::debug_tuple_field1_finish(f,
                    "EditionAndFutureReleaseError", &__self_0),
            FutureIncompatibilityReason::EditionAndFutureReleaseSemanticsChange(__self_0)
                =>
                ::core::fmt::Formatter::debug_tuple_field1_finish(f,
                    "EditionAndFutureReleaseSemanticsChange", &__self_0),
            FutureIncompatibilityReason::Custom(__self_0, __self_1) =>
                ::core::fmt::Formatter::debug_tuple_field2_finish(f, "Custom",
                    __self_0, &__self_1),
            FutureIncompatibilityReason::Unreachable =>
                ::core::fmt::Formatter::write_str(f, "Unreachable"),
        }
    }
}Debug)]
360pub enum FutureIncompatibilityReason {
361    /// This will be an error in a future release for all editions
362    ///
363    /// Choose this variant when you are first introducing a "future
364    /// incompatible" warning that is intended to eventually be fixed in the
365    /// future.
366    ///
367    /// After a lint has been in this state for a while and you feel like it is ready to graduate
368    /// to warning everyone, consider setting [`FutureIncompatibleInfo::report_in_deps`] to true.
369    /// (see its documentation for more guidance)
370    ///
371    /// After some period of time, lints with this variant can be turned into
372    /// hard errors (and the lint removed). Preferably when there is some
373    /// confidence that the number of impacted projects is very small (few
374    /// should have a broken dependency in their dependency tree).
375    FutureReleaseError(ReleaseFcw),
376    /// Code that changes meaning in some way in a
377    /// future release.
378    ///
379    /// Choose this variant when the semantics of existing code is changing,
380    /// (as opposed to [`FutureIncompatibilityReason::FutureReleaseError`],
381    /// which is for when code is going to be rejected in the future).
382    FutureReleaseSemanticsChange(ReleaseFcw),
383    /// Previously accepted code that will become an
384    /// error in the provided edition
385    ///
386    /// Choose this variant for code that you want to start rejecting across
387    /// an edition boundary. This will automatically include the lint in the
388    /// `rust-20xx-compatibility` lint group, which is used by `cargo fix
389    /// --edition` to do migrations. The lint *should* be auto-fixable with
390    /// [`Applicability::MachineApplicable`].
391    ///
392    /// The lint can either be `Allow` or `Warn` by default. If it is `Allow`,
393    /// users usually won't see this warning unless they are doing an edition
394    /// migration manually or there is a problem during the migration (cargo's
395    /// automatic migrations will force the level to `Warn`). If it is `Warn`
396    /// by default, users on all editions will see this warning (only do this
397    /// if you think it is important for everyone to be aware of the change,
398    /// and to encourage people to update their code on all editions).
399    ///
400    /// See also [`FutureIncompatibilityReason::EditionSemanticsChange`] if
401    /// you have code that is changing semantics across the edition (as
402    /// opposed to being rejected).
403    EditionError(EditionFcw),
404    /// Code that changes meaning in some way in
405    /// the provided edition
406    ///
407    /// This is the same as [`FutureIncompatibilityReason::EditionError`],
408    /// except for situations where the semantics change across an edition. It
409    /// slightly changes the text of the diagnostic, but is otherwise the
410    /// same.
411    EditionSemanticsChange(EditionFcw),
412    /// This will be an error in the provided edition *and* in a future
413    /// release.
414    ///
415    /// This variant a combination of [`FutureReleaseError`] and [`EditionError`].
416    /// This is useful in rare cases when we want to have "preview" of a breaking
417    /// change in an edition, but do a breaking change later on all editions anyway.
418    ///
419    /// [`EditionError`]: FutureIncompatibilityReason::EditionError
420    /// [`FutureReleaseError`]: FutureIncompatibilityReason::FutureReleaseError
421    EditionAndFutureReleaseError(EditionFcw),
422    /// This will change meaning in the provided edition *and* in a future
423    /// release.
424    ///
425    /// This variant a combination of [`FutureReleaseSemanticsChange`]
426    /// and [`EditionSemanticsChange`]. This is useful in rare cases when we
427    /// want to have "preview" of a breaking change in an edition, but do a
428    /// breaking change later on all editions anyway.
429    ///
430    /// [`EditionSemanticsChange`]: FutureIncompatibilityReason::EditionSemanticsChange
431    /// [`FutureReleaseSemanticsChange`]: FutureIncompatibilityReason::FutureReleaseSemanticsChange
432    EditionAndFutureReleaseSemanticsChange(EditionFcw),
433    /// A custom reason.
434    ///
435    /// Choose this variant if the built-in text of the diagnostic of the
436    /// other variants doesn't match your situation. This is behaviorally
437    /// equivalent to
438    /// [`FutureIncompatibilityReason::FutureReleaseError`].
439    Custom(&'static str, ReleaseFcw),
440
441    /// Using the declare_lint macro a reason always needs to be specified.
442    /// So, this case can't actually be reached but a variant needs to exist for it.
443    /// Any code panics on seeing this variant. Do not use.
444    Unreachable,
445}
446
447impl FutureIncompatibleInfo {
448    pub const fn default_fields_for_macro() -> Self {
449        FutureIncompatibleInfo {
450            reason: FutureIncompatibilityReason::Unreachable,
451            explain_reason: true,
452            report_in_deps: false,
453        }
454    }
455}
456
457impl FutureIncompatibilityReason {
458    pub fn edition(self) -> Option<Edition> {
459        match self {
460            Self::EditionError(e)
461            | Self::EditionSemanticsChange(e)
462            | Self::EditionAndFutureReleaseError(e)
463            | Self::EditionAndFutureReleaseSemanticsChange(e) => Some(e.edition),
464
465            FutureIncompatibilityReason::FutureReleaseError(_)
466            | FutureIncompatibilityReason::FutureReleaseSemanticsChange(_)
467            | FutureIncompatibilityReason::Custom(_, _) => None,
468            Self::Unreachable => ::core::panicking::panic("internal error: entered unreachable code")unreachable!(),
469        }
470    }
471
472    pub fn reference(&self) -> String {
473        match self {
474            Self::FutureReleaseSemanticsChange(release_fcw)
475            | Self::FutureReleaseError(release_fcw)
476            | Self::Custom(_, release_fcw) => release_fcw.to_string(),
477            Self::EditionError(edition_fcw)
478            | Self::EditionSemanticsChange(edition_fcw)
479            | Self::EditionAndFutureReleaseError(edition_fcw)
480            | Self::EditionAndFutureReleaseSemanticsChange(edition_fcw) => edition_fcw.to_string(),
481            Self::Unreachable => ::core::panicking::panic("internal error: entered unreachable code")unreachable!(),
482        }
483    }
484}
485
486impl Display for ReleaseFcw {
487    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
488        let issue_number = self.issue_number;
489        f.write_fmt(format_args!("issue #{0} <https://github.com/rust-lang/rust/issues/{0}>",
        issue_number))write!(f, "issue #{issue_number} <https://github.com/rust-lang/rust/issues/{issue_number}>")
490    }
491}
492
493impl Display for EditionFcw {
494    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
495        f.write_fmt(format_args!("<https://doc.rust-lang.org/edition-guide/{0}/{1}.html>",
        match self.edition {
            Edition::Edition2015 => "rust-2015",
            Edition::Edition2018 => "rust-2018",
            Edition::Edition2021 => "rust-2021",
            Edition::Edition2024 => "rust-2024",
            Edition::EditionFuture => "future",
        }, self.page_slug))write!(
496            f,
497            "<https://doc.rust-lang.org/edition-guide/{}/{}.html>",
498            match self.edition {
499                Edition::Edition2015 => "rust-2015",
500                Edition::Edition2018 => "rust-2018",
501                Edition::Edition2021 => "rust-2021",
502                Edition::Edition2024 => "rust-2024",
503                Edition::EditionFuture => "future",
504            },
505            self.page_slug,
506        )
507    }
508}
509
510impl Lint {
511    pub const fn default_fields_for_macro() -> Self {
512        Lint {
513            name: "",
514            default_level: Level::Forbid,
515            desc: "",
516            edition_lint_opts: None,
517            is_externally_loaded: false,
518            report_in_external_macro: false,
519            future_incompatible: None,
520            feature_gate: None,
521            crate_level_only: false,
522            eval_always: false,
523            ignore_deny_warnings: false,
524        }
525    }
526
527    /// Gets the lint's name, with ASCII letters converted to lowercase.
528    pub fn name_lower(&self) -> String {
529        self.name.to_ascii_lowercase()
530    }
531
532    pub fn default_level(&self, edition: Edition) -> Level {
533        self.edition_lint_opts
534            .filter(|(e, _)| *e <= edition)
535            .map(|(_, l)| l)
536            .unwrap_or(self.default_level)
537    }
538}
539
540/// Identifies a lint known to the compiler.
541#[derive(#[automatically_derived]
impl ::core::clone::Clone for LintId {
    #[inline]
    fn clone(&self) -> LintId {
        let _: ::core::clone::AssertParamIsClone<&'static Lint>;
        *self
    }
}Clone, #[automatically_derived]
impl ::core::marker::Copy for LintId { }Copy, #[automatically_derived]
impl ::core::fmt::Debug for LintId {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::debug_struct_field1_finish(f, "LintId",
            "lint", &&self.lint)
    }
}Debug)]
542pub struct LintId {
543    // Identity is based on pointer equality of this field.
544    pub lint: &'static Lint,
545}
546
547impl PartialEq for LintId {
548    fn eq(&self, other: &LintId) -> bool {
549        std::ptr::eq(self.lint, other.lint)
550    }
551}
552
553impl Eq for LintId {}
554
555impl std::hash::Hash for LintId {
556    fn hash<H: std::hash::Hasher>(&self, state: &mut H) {
557        let ptr = self.lint as *const Lint;
558        ptr.hash(state);
559    }
560}
561
562impl LintId {
563    /// Gets the `LintId` for a `Lint`.
564    pub fn of(lint: &'static Lint) -> LintId {
565        LintId { lint }
566    }
567
568    pub fn lint_name_raw(&self) -> &'static str {
569        self.lint.name
570    }
571
572    /// Gets the name of the lint.
573    pub fn to_string(&self) -> String {
574        self.lint.name_lower()
575    }
576}
577
578impl StableHash for LintId {
579    #[inline]
580    fn stable_hash<Hcx: StableHashCtxt>(&self, hcx: &mut Hcx, hasher: &mut StableHasher) {
581        self.lint_name_raw().stable_hash(hcx, hasher);
582    }
583}
584
585impl StableCompare for LintId {
586    const CAN_USE_UNSTABLE_SORT: bool = true;
587
588    fn stable_cmp(&self, other: &Self) -> std::cmp::Ordering {
589        self.lint_name_raw().cmp(&other.lint_name_raw())
590    }
591}
592
593#[derive(#[automatically_derived]
impl ::core::fmt::Debug for DeprecatedSinceKind {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        match self {
            DeprecatedSinceKind::InEffect =>
                ::core::fmt::Formatter::write_str(f, "InEffect"),
            DeprecatedSinceKind::InFuture =>
                ::core::fmt::Formatter::write_str(f, "InFuture"),
            DeprecatedSinceKind::InVersion(__self_0) =>
                ::core::fmt::Formatter::debug_tuple_field1_finish(f,
                    "InVersion", &__self_0),
        }
    }
}Debug, #[automatically_derived]
impl ::core::clone::Clone for DeprecatedSinceKind {
    #[inline]
    fn clone(&self) -> DeprecatedSinceKind {
        match self {
            DeprecatedSinceKind::InEffect => DeprecatedSinceKind::InEffect,
            DeprecatedSinceKind::InFuture => DeprecatedSinceKind::InFuture,
            DeprecatedSinceKind::InVersion(__self_0) =>
                DeprecatedSinceKind::InVersion(::core::clone::Clone::clone(__self_0)),
        }
    }
}Clone)]
594pub enum DeprecatedSinceKind {
595    InEffect,
596    InFuture,
597    InVersion(String),
598}
599
600pub type RegisteredTools = FxIndexSet<Ident>;
601
602/// Declares a static item of type `&'static Lint`.
603///
604/// See <https://rustc-dev-guide.rust-lang.org/diagnostics.html> for
605/// documentation and guidelines on writing lints.
606///
607/// The macro call should start with a doc comment explaining the lint
608/// which will be embedded in the rustc user documentation book. It should
609/// be written in markdown and have a format that looks like this:
610///
611/// ```rust,ignore (doc-example)
612/// /// The `my_lint_name` lint detects [short explanation here].
613/// ///
614/// /// ### Example
615/// ///
616/// /// ```rust
617/// /// [insert a concise example that triggers the lint]
618/// /// ```
619/// ///
620/// /// {{produces}}
621/// ///
622/// /// ### Explanation
623/// ///
624/// /// This should be a detailed explanation of *why* the lint exists,
625/// /// and also include suggestions on how the user should fix the problem.
626/// /// Try to keep the text simple enough that a beginner can understand,
627/// /// and include links to other documentation for terminology that a
628/// /// beginner may not be familiar with. If this is "allow" by default,
629/// /// it should explain why (are there false positives or other issues?). If
630/// /// this is a future-incompatible lint, it should say so, with text that
631/// /// looks roughly like this:
632/// ///
633/// /// This is a [future-incompatible] lint to transition this to a hard
634/// /// error in the future. See [issue #xxxxx] for more details.
635/// ///
636/// /// [issue #xxxxx]: https://github.com/rust-lang/rust/issues/xxxxx
637/// ```
638///
639/// The `{{produces}}` tag will be automatically replaced with the output from
640/// the example by the build system. If the lint example is too complex to run
641/// as a simple example (for example, it needs an extern crate), mark the code
642/// block with `ignore` and manually replace the `{{produces}}` line with the
643/// expected output in a `text` code block.
644///
645/// If this is a rustdoc-only lint, then only include a brief introduction
646/// with a link with the text `[rustdoc book]` so that the validator knows
647/// that this is for rustdoc only (see BROKEN_INTRA_DOC_LINKS as an example).
648///
649/// Commands to view and test the documentation:
650///
651/// * `./x.py doc --stage=1 src/doc/rustc --open`: Builds the rustc book and opens it.
652/// * `./x.py test src/tools/lint-docs`: Validates that the lint docs have the
653///   correct style, and that the code example actually emits the expected
654///   lint.
655///
656/// If you have already built the compiler, and you want to make changes to
657/// just the doc comments, then use the `--keep-stage=0` flag with the above
658/// commands to avoid rebuilding the compiler.
659#[macro_export]
660macro_rules! declare_lint {
661    ($(#[$attr:meta])* $vis: vis $NAME: ident, $Level: ident, $desc: expr) => (
662        $crate::declare_lint!(
663            $(#[$attr])* $vis $NAME, $Level, $desc,
664        );
665    );
666    ($(#[$attr:meta])* $vis: vis $NAME: ident, $Level: ident, $desc: expr,
667     $(@eval_always = $eval_always:literal)?
668     $(@feature_gate = $gate:ident;)?
669     $(@future_incompatible = FutureIncompatibleInfo {
670        reason: $reason:expr,
671        $($field:ident : $val:expr),* $(,)*
672     }; )?
673     $(@edition $lint_edition:ident => $edition_level:ident;)?
674     $($v:ident),*) => (
675        $(#[$attr])*
676        $vis static $NAME: &$crate::Lint = &$crate::Lint {
677            name: stringify!($NAME),
678            default_level: $crate::$Level,
679            desc: $desc,
680            is_externally_loaded: false,
681            $($v: true,)*
682            $(feature_gate: Some(rustc_span::sym::$gate),)?
683            $(future_incompatible: Some($crate::FutureIncompatibleInfo {
684                reason: $reason,
685                $($field: $val,)*
686                ..$crate::FutureIncompatibleInfo::default_fields_for_macro()
687            }),)?
688            $(edition_lint_opts: Some(($crate::Edition::$lint_edition, $crate::$edition_level)),)?
689            $(eval_always: $eval_always,)?
690            ..$crate::Lint::default_fields_for_macro()
691        };
692    );
693}
694
695#[macro_export]
696macro_rules! declare_tool_lint {
697    (
698        $(#[$attr:meta])* $vis:vis $tool:ident ::$NAME:ident, $Level: ident, $desc: expr
699        $(, @eval_always = $eval_always:literal)?
700        $(, @feature_gate = $gate:ident;)?
701    ) => (
702        $crate::declare_tool_lint!{$(#[$attr])* $vis $tool::$NAME, $Level, $desc, false $(, @eval_always = $eval_always)? $(, @feature_gate = $gate;)?}
703    );
704    (
705        $(#[$attr:meta])* $vis:vis $tool:ident ::$NAME:ident, $Level:ident, $desc:expr,
706        report_in_external_macro: $rep:expr
707        $(, @eval_always = $eval_always: literal)?
708        $(, @feature_gate = $gate:ident;)?
709    ) => (
710         $crate::declare_tool_lint!{$(#[$attr])* $vis $tool::$NAME, $Level, $desc, $rep  $(, @eval_always = $eval_always)? $(, @feature_gate = $gate;)?}
711    );
712    (
713        $(#[$attr:meta])* $vis:vis $tool:ident ::$NAME:ident, $Level:ident, $desc:expr,
714        $external:expr
715        $(, @eval_always = $eval_always: literal)?
716        $(, @feature_gate = $gate:ident;)?
717    ) => (
718        $(#[$attr])*
719        $vis static $NAME: &$crate::Lint = &$crate::Lint {
720            name: &concat!(stringify!($tool), "::", stringify!($NAME)),
721            default_level: $crate::$Level,
722            desc: $desc,
723            edition_lint_opts: None,
724            report_in_external_macro: $external,
725            future_incompatible: None,
726            is_externally_loaded: true,
727            $(feature_gate: Some(rustc_span::sym::$gate),)?
728            crate_level_only: false,
729            $(eval_always: $eval_always,)?
730            ..$crate::Lint::default_fields_for_macro()
731        };
732    );
733}
734
735pub type LintVec = Vec<&'static Lint>;
736
737pub trait LintPass {
738    fn name(&self) -> &'static str;
739    fn get_lints(&self) -> LintVec;
740}
741
742/// Implements `LintPass for $ty` with the given list of `Lint` statics.
743#[macro_export]
744macro_rules! impl_lint_pass {
745    ($ty:ty => [$($lint:expr),* $(,)?]) => {
746        impl $crate::LintPass for $ty {
747            fn name(&self) -> &'static str { stringify!($ty) }
748            fn get_lints(&self) -> $crate::LintVec { vec![$($lint),*] }
749        }
750        impl $ty {
751            #[allow(unused)]
752            pub fn lint_vec() -> $crate::LintVec { vec![$($lint),*] }
753        }
754    };
755}
756
757/// Declares a type named `$name` which implements `LintPass`.
758/// To the right of `=>` a comma separated list of `Lint` statics is given.
759#[macro_export]
760macro_rules! declare_lint_pass {
761    ($(#[$m:meta])* $name:ident => [$($lint:expr),* $(,)?]) => {
762        $(#[$m])* #[derive(Copy, Clone)] pub struct $name;
763        $crate::impl_lint_pass!($name => [$($lint),*]);
764    };
765}
766
767#[macro_export]
768macro_rules! fcw {
769    (FutureReleaseError # $issue_number: literal) => {
770       $crate:: FutureIncompatibilityReason::FutureReleaseError($crate::ReleaseFcw { issue_number: $issue_number })
771    };
772    (FutureReleaseSemanticsChange # $issue_number: literal) => {
773        $crate::FutureIncompatibilityReason::FutureReleaseSemanticsChange($crate::ReleaseFcw {
774            issue_number: $issue_number,
775        })
776    };
777    ($description: literal # $issue_number: literal) => {
778        $crate::FutureIncompatibilityReason::Custom($description, $crate::ReleaseFcw {
779            issue_number: $issue_number,
780        })
781    };
782    (EditionError $edition_name: tt $page_slug: literal) => {
783        $crate::FutureIncompatibilityReason::EditionError($crate::EditionFcw {
784            edition: fcw!(@edition $edition_name),
785            page_slug: $page_slug,
786        })
787    };
788    (EditionSemanticsChange $edition_name: tt $page_slug: literal) => {
789        $crate::FutureIncompatibilityReason::EditionSemanticsChange($crate::EditionFcw {
790            edition: fcw!(@edition $edition_name),
791            page_slug: $page_slug,
792        })
793    };
794    (EditionAndFutureReleaseSemanticsChange $edition_name: tt $page_slug: literal) => {
795        $crate::FutureIncompatibilityReason::EditionAndFutureReleaseSemanticsChange($crate::EditionFcw {
796            edition: fcw!(@edition $edition_name),
797            page_slug: $page_slug,
798        })
799    };
800    (EditionAndFutureReleaseError $edition_name: tt $page_slug: literal) => {
801        $crate::FutureIncompatibilityReason::EditionAndFutureReleaseError($crate::EditionFcw {
802            edition: fcw!(@edition $edition_name),
803            page_slug: $page_slug,
804        })
805    };
806    (@edition 2024) => {
807        rustc_span::edition::Edition::Edition2024
808    };
809    (@edition 2021) => {
810        rustc_span::edition::Edition::Edition2021
811    };
812    (@edition 2018) => {
813        rustc_span::edition::Edition::Edition2018
814    };
815}