mirror of
https://github.com/rust-lang/rust.git
synced 2026-05-16 21:15:18 +03:00
260 lines
9.3 KiB
Rust
260 lines
9.3 KiB
Rust
// tidy-alphabetical-start
|
|
#![allow(rustc::default_hash_types)]
|
|
#![feature(never_type)]
|
|
#![feature(proc_macro_diagnostic)]
|
|
#![feature(proc_macro_tracked_env)]
|
|
// tidy-alphabetical-end
|
|
|
|
use proc_macro::TokenStream;
|
|
use synstructure::decl_derive;
|
|
|
|
mod current_version;
|
|
mod diagnostics;
|
|
mod extension;
|
|
mod hash_stable;
|
|
mod lift;
|
|
mod print_attribute;
|
|
mod query;
|
|
mod serialize;
|
|
mod symbols;
|
|
mod type_foldable;
|
|
mod type_visitable;
|
|
mod visitable;
|
|
|
|
// Reads the rust version (e.g. "1.75.0") from the CFG_RELEASE env var and
|
|
// produces a `RustcVersion` literal containing that version (e.g.
|
|
// `RustcVersion { major: 1, minor: 75, patch: 0 }`).
|
|
#[proc_macro]
|
|
pub fn current_rustc_version(input: TokenStream) -> TokenStream {
|
|
current_version::current_version(input)
|
|
}
|
|
|
|
#[proc_macro]
|
|
pub fn rustc_queries(input: TokenStream) -> TokenStream {
|
|
query::rustc_queries(input)
|
|
}
|
|
|
|
#[proc_macro]
|
|
pub fn symbols(input: TokenStream) -> TokenStream {
|
|
symbols::symbols(input.into()).into()
|
|
}
|
|
|
|
/// Derive an extension trait for a given impl block. The trait name
|
|
/// goes into the parenthesized args of the macro, for greppability.
|
|
/// For example:
|
|
/// ```
|
|
/// use rustc_macros::extension;
|
|
/// #[extension(pub trait Foo)]
|
|
/// impl i32 { fn hello() {} }
|
|
/// ```
|
|
///
|
|
/// expands to:
|
|
/// ```
|
|
/// pub trait Foo { fn hello(); }
|
|
/// impl Foo for i32 { fn hello() {} }
|
|
/// ```
|
|
#[proc_macro_attribute]
|
|
pub fn extension(attr: TokenStream, input: TokenStream) -> TokenStream {
|
|
extension::extension(attr, input)
|
|
}
|
|
|
|
decl_derive!([HashStable, attributes(stable_hasher)] => hash_stable::hash_stable_derive);
|
|
decl_derive!(
|
|
[HashStable_Generic, attributes(stable_hasher)] =>
|
|
hash_stable::hash_stable_generic_derive
|
|
);
|
|
decl_derive!(
|
|
[HashStable_NoContext] =>
|
|
/// `HashStable` implementation that has no `HashStableContext` bound and
|
|
/// which adds `where` bounds for `HashStable` based off of fields and not
|
|
/// generics. This is suitable for use in crates like `rustc_type_ir`.
|
|
hash_stable::hash_stable_no_context_derive
|
|
);
|
|
|
|
// Encoding and Decoding derives
|
|
decl_derive!([Decodable_NoContext] =>
|
|
/// See docs on derive [`Decodable`].
|
|
///
|
|
/// Derives `Decodable<D> for T where D: Decoder`.
|
|
serialize::decodable_nocontext_derive
|
|
);
|
|
decl_derive!([Encodable_NoContext] => serialize::encodable_nocontext_derive);
|
|
decl_derive!([Decodable] =>
|
|
/// Derives `Decodable<D> for T where D: SpanDecoder`
|
|
///
|
|
/// # Deriving decoding traits
|
|
///
|
|
/// > Some shared docs about decoding traits, since this is likely the first trait you find
|
|
///
|
|
/// The difference between these derives can be subtle!
|
|
/// At a high level, there's the `T: Decodable<D>` trait that says some type `T`
|
|
/// can be decoded using a decoder `D`. There are various decoders!
|
|
/// The different derives place different *trait* bounds on this type `D`.
|
|
///
|
|
/// Even though this derive, based on its name, seems like the most vanilla one,
|
|
/// it actually places a pretty strict bound on `D`: `SpanDecoder`.
|
|
/// It means that types that derive this can contain spans, among other things,
|
|
/// and still be decoded. The reason this is hard is that at least in metadata,
|
|
/// spans can only be decoded later, once some information from the header
|
|
/// is already decoded to properly deal with spans.
|
|
///
|
|
/// The hierarchy is roughly:
|
|
///
|
|
/// - derive [`Decodable_NoContext`] is the most relaxed bounds that could be placed on `D`,
|
|
/// and is only really suited for structs and enums containing primitive types.
|
|
/// - derive [`BlobDecodable`] may be a better default, than deriving `Decodable`:
|
|
/// it places fewer requirements on `D`, while still allowing some complex types to be decoded.
|
|
/// - derive [`LazyDecodable`]: Only for types containing `Lazy{Array,Table,Value}`.
|
|
/// - derive [`Decodable`] for structures containing spans. Requires `D: SpanDecoder`
|
|
/// - derive [`TyDecodable`] for types that require access to the `TyCtxt` while decoding.
|
|
/// For example: arena allocated types.
|
|
serialize::decodable_derive
|
|
);
|
|
decl_derive!([Encodable] => serialize::encodable_derive);
|
|
decl_derive!([TyDecodable] =>
|
|
/// See docs on derive [`Decodable`].
|
|
///
|
|
/// Derives `Decodable<D> for T where D: TyDecoder`.
|
|
serialize::type_decodable_derive
|
|
);
|
|
decl_derive!([TyEncodable] => serialize::type_encodable_derive);
|
|
decl_derive!([LazyDecodable] =>
|
|
/// See docs on derive [`Decodable`].
|
|
///
|
|
/// Derives `Decodable<D> for T where D: LazyDecoder`.
|
|
/// This constrains the decoder to be specifically the decoder that can decode
|
|
/// `LazyArray`s, `LazyValue`s amd `LazyTable`s in metadata.
|
|
/// Therefore, we only need this on things containing LazyArray really.
|
|
///
|
|
/// Most decodable derives mirror an encodable derive.
|
|
/// [`LazyDecodable`] and [`BlobDecodable`] together roughly mirror [`MetadataEncodable`]
|
|
serialize::lazy_decodable_derive
|
|
);
|
|
decl_derive!([BlobDecodable] =>
|
|
/// See docs on derive [`Decodable`].
|
|
///
|
|
/// Derives `Decodable<D> for T where D: BlobDecoder`.
|
|
///
|
|
/// Most decodable derives mirror an encodable derive.
|
|
/// [`LazyDecodable`] and [`BlobDecodable`] together roughly mirror [`MetadataEncodable`]
|
|
serialize::blob_decodable_derive
|
|
);
|
|
decl_derive!([MetadataEncodable] =>
|
|
/// Most encodable derives mirror a decodable derive.
|
|
/// [`MetadataEncodable`] is roughly mirrored by the combination of [`LazyDecodable`] and [`BlobDecodable`]
|
|
serialize::meta_encodable_derive
|
|
);
|
|
|
|
decl_derive!(
|
|
[TypeFoldable, attributes(type_foldable)] =>
|
|
/// Derives `TypeFoldable` for the annotated `struct` or `enum` (`union` is not supported).
|
|
///
|
|
/// The fold will produce a value of the same struct or enum variant as the input, with
|
|
/// each field respectively folded using the `TypeFoldable` implementation for its type.
|
|
/// However, if a field of a struct or an enum variant is annotated with
|
|
/// `#[type_foldable(identity)]` then that field will retain its incumbent value (and its
|
|
/// type is not required to implement `TypeFoldable`).
|
|
type_foldable::type_foldable_derive
|
|
);
|
|
decl_derive!(
|
|
[TypeVisitable, attributes(type_visitable)] =>
|
|
/// Derives `TypeVisitable` for the annotated `struct` or `enum` (`union` is not supported).
|
|
///
|
|
/// Each field of the struct or enum variant will be visited in definition order, using the
|
|
/// `TypeVisitable` implementation for its type. However, if a field of a struct or an enum
|
|
/// variant is annotated with `#[type_visitable(ignore)]` then that field will not be
|
|
/// visited (and its type is not required to implement `TypeVisitable`).
|
|
type_visitable::type_visitable_derive
|
|
);
|
|
decl_derive!(
|
|
[Walkable, attributes(visitable)] =>
|
|
/// Derives `Walkable` for the annotated `struct` or `enum` (`union` is not supported).
|
|
///
|
|
/// Each field of the struct or enum variant will be visited in definition order, using the
|
|
/// `Walkable` implementation for its type. However, if a field of a struct or an enum
|
|
/// variant is annotated with `#[visitable(ignore)]` then that field will not be
|
|
/// visited (and its type is not required to implement `Walkable`).
|
|
visitable::visitable_derive
|
|
);
|
|
decl_derive!([Lift, attributes(lift)] => lift::lift_derive);
|
|
decl_derive!(
|
|
[Diagnostic, attributes(
|
|
// struct attributes
|
|
diag,
|
|
help,
|
|
help_once,
|
|
note,
|
|
note_once,
|
|
warning,
|
|
// field attributes
|
|
skip_arg,
|
|
primary_span,
|
|
label,
|
|
subdiagnostic,
|
|
suggestion,
|
|
suggestion_short,
|
|
suggestion_hidden,
|
|
suggestion_verbose)] => diagnostics::diagnostic_derive
|
|
);
|
|
decl_derive!(
|
|
[LintDiagnostic, attributes(
|
|
// struct attributes
|
|
diag,
|
|
help,
|
|
help_once,
|
|
note,
|
|
note_once,
|
|
warning,
|
|
// field attributes
|
|
skip_arg,
|
|
primary_span,
|
|
label,
|
|
subdiagnostic,
|
|
suggestion,
|
|
suggestion_short,
|
|
suggestion_hidden,
|
|
suggestion_verbose)] => diagnostics::lint_diagnostic_derive
|
|
);
|
|
decl_derive!(
|
|
[Subdiagnostic, attributes(
|
|
// struct/variant attributes
|
|
label,
|
|
help,
|
|
help_once,
|
|
note,
|
|
note_once,
|
|
warning,
|
|
subdiagnostic,
|
|
suggestion,
|
|
suggestion_short,
|
|
suggestion_hidden,
|
|
suggestion_verbose,
|
|
multipart_suggestion,
|
|
multipart_suggestion_short,
|
|
multipart_suggestion_hidden,
|
|
multipart_suggestion_verbose,
|
|
// field attributes
|
|
skip_arg,
|
|
primary_span,
|
|
suggestion_part,
|
|
applicability)] => diagnostics::subdiagnostic_derive
|
|
);
|
|
|
|
/// This macro creates a translatable `DiagMessage` from a fluent format string.
|
|
/// It should be used in places where a translatable message is needed, but struct diagnostics are undesired.
|
|
///
|
|
/// This macro statically checks that the message is valid Fluent, but not that variables in the Fluent message actually exist.
|
|
#[proc_macro]
|
|
pub fn msg(input: TokenStream) -> TokenStream {
|
|
diagnostics::msg_macro(input)
|
|
}
|
|
|
|
decl_derive! {
|
|
[PrintAttribute] =>
|
|
/// Derives `PrintAttribute` for `AttributeKind`.
|
|
/// This macro is pretty specific to `rustc_hir::attrs` and likely not that useful in
|
|
/// other places. It's deriving something close to `Debug` without printing some extraneous
|
|
/// things like spans.
|
|
print_attribute::print_attribute
|
|
}
|