mirror of
				https://github.com/rust-lang/rust.git
				synced 2025-10-25 06:32:04 +00:00 
			
		
		
		
	 d1fcf61117
			
		
	
	
		d1fcf61117
		
	
	
	
	
		
			
			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
 | |
| );
 |