mirror of
https://github.com/rust-lang/rust.git
synced 2025-04-29 19:47:38 +00:00

Instead of loading the Fluent resources for every crate in `rustc_error_messages`, each crate generates typed identifiers for its own diagnostics and creates a static which are pulled together in the `rustc_driver` crate and provided to the diagnostic emitter. Signed-off-by: David Wood <david.wood@huawei.com>
200 lines
6.6 KiB
Rust
200 lines
6.6 KiB
Rust
#![feature(allow_internal_unstable)]
|
|
#![feature(if_let_guard)]
|
|
#![feature(let_chains)]
|
|
#![feature(never_type)]
|
|
#![feature(proc_macro_diagnostic)]
|
|
#![feature(proc_macro_span)]
|
|
#![allow(rustc::default_hash_types)]
|
|
#![deny(rustc::untranslatable_diagnostic)]
|
|
#![deny(rustc::diagnostic_outside_of_impl)]
|
|
#![recursion_limit = "128"]
|
|
|
|
use synstructure::decl_derive;
|
|
|
|
use proc_macro::TokenStream;
|
|
|
|
mod diagnostics;
|
|
mod hash_stable;
|
|
mod lift;
|
|
mod newtype;
|
|
mod query;
|
|
mod serialize;
|
|
mod symbols;
|
|
mod type_foldable;
|
|
mod type_visitable;
|
|
|
|
#[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()
|
|
}
|
|
|
|
/// Creates a struct type `S` that can be used as an index with
|
|
/// `IndexVec` and so on.
|
|
///
|
|
/// There are two ways of interacting with these indices:
|
|
///
|
|
/// - The `From` impls are the preferred way. So you can do
|
|
/// `S::from(v)` with a `usize` or `u32`. And you can convert back
|
|
/// to an integer with `u32::from(s)`.
|
|
///
|
|
/// - Alternatively, you can use the methods `S::new(v)` and `s.index()`
|
|
/// to create/return a value.
|
|
///
|
|
/// Internally, the index uses a u32, so the index must not exceed
|
|
/// `u32::MAX`. You can also customize things like the `Debug` impl,
|
|
/// what traits are derived, and so forth via the macro.
|
|
#[proc_macro]
|
|
#[allow_internal_unstable(step_trait, rustc_attrs, trusted_step, spec_option_partial_eq)]
|
|
pub fn newtype_index(input: TokenStream) -> TokenStream {
|
|
newtype::newtype(input)
|
|
}
|
|
|
|
/// Implements the `fluent_messages` macro, which performs compile-time validation of the
|
|
/// compiler's Fluent resources (i.e. that the resources parse and don't multiply define the same
|
|
/// messages) and generates constants that make using those messages in diagnostics more ergonomic.
|
|
///
|
|
/// For example, given the following invocation of the macro..
|
|
///
|
|
/// ```ignore (rust)
|
|
/// fluent_messages! { "./typeck.ftl" }
|
|
/// ```
|
|
/// ..where `typeck.ftl` has the following contents..
|
|
///
|
|
/// ```fluent
|
|
/// typeck_field_multiply_specified_in_initializer =
|
|
/// field `{$ident}` specified more than once
|
|
/// .label = used more than once
|
|
/// .label_previous_use = first use of `{$ident}`
|
|
/// ```
|
|
/// ...then the macro parse the Fluent resource, emitting a diagnostic if it fails to do so, and
|
|
/// will generate the following code:
|
|
///
|
|
/// ```ignore (rust)
|
|
/// pub static DEFAULT_LOCALE_RESOURCE: &'static [&'static str] = include_str!("./typeck.ftl");
|
|
///
|
|
/// mod fluent_generated {
|
|
/// mod typeck {
|
|
/// pub const field_multiply_specified_in_initializer: DiagnosticMessage =
|
|
/// DiagnosticMessage::fluent("typeck_field_multiply_specified_in_initializer");
|
|
/// pub const field_multiply_specified_in_initializer_label_previous_use: DiagnosticMessage =
|
|
/// DiagnosticMessage::fluent_attr(
|
|
/// "typeck_field_multiply_specified_in_initializer",
|
|
/// "previous_use_label"
|
|
/// );
|
|
/// }
|
|
/// }
|
|
/// ```
|
|
/// When emitting a diagnostic, the generated constants can be used as follows:
|
|
///
|
|
/// ```ignore (rust)
|
|
/// let mut err = sess.struct_span_err(
|
|
/// span,
|
|
/// fluent::typeck::field_multiply_specified_in_initializer
|
|
/// );
|
|
/// err.span_default_label(span);
|
|
/// err.span_label(
|
|
/// previous_use_span,
|
|
/// fluent::typeck::field_multiply_specified_in_initializer_label_previous_use
|
|
/// );
|
|
/// err.emit();
|
|
/// ```
|
|
#[proc_macro]
|
|
pub fn fluent_messages(input: TokenStream) -> TokenStream {
|
|
diagnostics::fluent_messages(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!([Decodable] => serialize::decodable_derive);
|
|
decl_derive!([Encodable] => serialize::encodable_derive);
|
|
decl_derive!([TyDecodable] => serialize::type_decodable_derive);
|
|
decl_derive!([TyEncodable] => serialize::type_encodable_derive);
|
|
decl_derive!([MetadataDecodable] => serialize::meta_decodable_derive);
|
|
decl_derive!([MetadataEncodable] => 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!([Lift, attributes(lift)] => lift::lift_derive);
|
|
decl_derive!(
|
|
[Diagnostic, attributes(
|
|
// struct attributes
|
|
diag,
|
|
help,
|
|
note,
|
|
warning,
|
|
// field attributes
|
|
skip_arg,
|
|
primary_span,
|
|
label,
|
|
subdiagnostic,
|
|
suggestion,
|
|
suggestion_short,
|
|
suggestion_hidden,
|
|
suggestion_verbose)] => diagnostics::session_diagnostic_derive
|
|
);
|
|
decl_derive!(
|
|
[LintDiagnostic, attributes(
|
|
// struct attributes
|
|
diag,
|
|
help,
|
|
note,
|
|
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,
|
|
note,
|
|
warning,
|
|
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::session_subdiagnostic_derive
|
|
);
|