diff --git a/compiler/rustc_attr/src/builtin.rs b/compiler/rustc_attr/src/builtin.rs index 34c24a26f7b13..3f99616494519 100644 --- a/compiler/rustc_attr/src/builtin.rs +++ b/compiler/rustc_attr/src/builtin.rs @@ -503,10 +503,6 @@ fn parse_unstability(sess: &Session, attr: &Attribute) -> Option<(Symbol, Stabil } } -pub fn find_crate_name(attrs: &[Attribute]) -> Option { - attr::first_attr_value_str_by_name(attrs, sym::crate_name) -} - #[derive(Clone, Debug)] pub struct Condition { pub name: Symbol, diff --git a/compiler/rustc_driver_impl/src/lib.rs b/compiler/rustc_driver_impl/src/lib.rs index bbe9741bf444e..98baf95084c9d 100644 --- a/compiler/rustc_driver_impl/src/lib.rs +++ b/compiler/rustc_driver_impl/src/lib.rs @@ -759,12 +759,17 @@ fn print_crate_info( // no crate attributes, print out an error and exit return Compilation::Continue; }; - let t_outputs = rustc_interface::util::build_output_filenames(attrs, sess); - let id = rustc_session::output::find_crate_name(sess, attrs); + let (crate_name, crate_name_origin) = passes::get_crate_name(sess, attrs); + let t_outputs = rustc_interface::util::build_output_filenames( + sess, + crate_name, + crate_name_origin, + ); let crate_types = collect_crate_types(sess, attrs); for &style in &crate_types { - let fname = - rustc_session::output::filename_for_input(sess, style, id, &t_outputs); + let fname = rustc_session::output::filename_for_input( + sess, style, crate_name, &t_outputs, + ); println_info!("{}", fname.as_path().file_name().unwrap().to_string_lossy()); } } @@ -773,8 +778,8 @@ fn print_crate_info( // no crate attributes, print out an error and exit return Compilation::Continue; }; - let id = rustc_session::output::find_crate_name(sess, attrs); - println_info!("{id}"); + let (crate_name, _) = passes::get_crate_name(sess, attrs); + println_info!("{crate_name}"); } Cfg => { let mut cfgs = sess diff --git a/compiler/rustc_expand/src/module.rs b/compiler/rustc_expand/src/module.rs index 506bd445be35f..e6b49e7e397a0 100644 --- a/compiler/rustc_expand/src/module.rs +++ b/compiler/rustc_expand/src/module.rs @@ -179,12 +179,12 @@ fn mod_file_path_from_attr( let first_path = attrs.iter().find(|at| at.has_name(sym::path))?; let Some(path_sym) = first_path.value_str() else { // This check is here mainly to catch attempting to use a macro, - // such as #[path = concat!(...)]. This isn't currently supported + // such as `#[path = concat!(...)]`. This isn't currently supported // because otherwise the InvocationCollector would need to defer - // loading a module until the #[path] attribute was expanded, and - // it doesn't support that (and would likely add a bit of - // complexity). Usually bad forms are checked in AstValidator (via - // `check_builtin_attribute`), but by the time that runs the macro + // loading a module until the `#[path]` attribute was expanded, and + // it doesn't support that (and would likely add a bit of complexity). + // Usually bad forms are checked during semantic analysis via + // `TyCtxt::check_mod_attrs`), but by the time that runs the macro // is expanded, and it doesn't give an error. validate_attr::emit_fatal_malformed_builtin_attribute(&sess.psess, first_path, sym::path); }; diff --git a/compiler/rustc_incremental/src/persist/fs.rs b/compiler/rustc_incremental/src/persist/fs.rs index 9afea3d66b0de..139b19723f54e 100644 --- a/compiler/rustc_incremental/src/persist/fs.rs +++ b/compiler/rustc_incremental/src/persist/fs.rs @@ -116,8 +116,9 @@ use rustc_errors::ErrorGuaranteed; use rustc_fs_util::{link_or_copy, try_canonicalize, LinkOrCopy}; use rustc_middle::bug; use rustc_session::config::CrateType; -use rustc_session::output::{collect_crate_types, find_crate_name}; +use rustc_session::output::collect_crate_types; use rustc_session::{Session, StableCrateId}; +use rustc_span::Symbol; use std::fs as std_fs; use std::io::{self, ErrorKind}; @@ -215,7 +216,10 @@ pub fn in_incr_comp_dir(incr_comp_session_dir: &Path, file_name: &str) -> PathBu /// The garbage collection will take care of it. /// /// [`rustc_interface::queries::dep_graph`]: ../../rustc_interface/struct.Queries.html#structfield.dep_graph -pub(crate) fn prepare_session_directory(sess: &Session) -> Result<(), ErrorGuaranteed> { +pub(crate) fn prepare_session_directory( + sess: &Session, + crate_name: Symbol, +) -> Result<(), ErrorGuaranteed> { if sess.opts.incremental.is_none() { return Ok(()); } @@ -225,7 +229,7 @@ pub(crate) fn prepare_session_directory(sess: &Session) -> Result<(), ErrorGuara debug!("prepare_session_directory"); // {incr-comp-dir}/{crate-name-and-disambiguator} - let crate_dir = crate_path(sess); + let crate_dir = crate_path(sess, crate_name); debug!("crate-dir: {}", crate_dir.display()); create_dir(sess, &crate_dir, "crate")?; @@ -608,10 +612,9 @@ fn string_to_timestamp(s: &str) -> Result { Ok(UNIX_EPOCH + duration) } -fn crate_path(sess: &Session) -> PathBuf { +fn crate_path(sess: &Session, crate_name: Symbol) -> PathBuf { let incr_dir = sess.opts.incremental.as_ref().unwrap().clone(); - let crate_name = find_crate_name(sess, &[]); let crate_types = collect_crate_types(sess, &[]); let stable_crate_id = StableCrateId::new( crate_name, diff --git a/compiler/rustc_incremental/src/persist/load.rs b/compiler/rustc_incremental/src/persist/load.rs index af667a57ce123..7aab960ed123f 100644 --- a/compiler/rustc_incremental/src/persist/load.rs +++ b/compiler/rustc_incremental/src/persist/load.rs @@ -9,7 +9,7 @@ use rustc_serialize::opaque::MemDecoder; use rustc_serialize::Decodable; use rustc_session::config::IncrementalStateAssertion; use rustc_session::Session; -use rustc_span::ErrorGuaranteed; +use rustc_span::{ErrorGuaranteed, Symbol}; use std::path::{Path, PathBuf}; use std::sync::Arc; use tracing::{debug, warn}; @@ -204,9 +204,9 @@ pub fn load_query_result_cache(sess: &Session) -> Option> { /// Setups the dependency graph by loading an existing graph from disk and set up streaming of a /// new graph to an incremental session directory. -pub fn setup_dep_graph(sess: &Session) -> Result { +pub fn setup_dep_graph(sess: &Session, crate_name: Symbol) -> Result { // `load_dep_graph` can only be called after `prepare_session_directory`. - prepare_session_directory(sess)?; + prepare_session_directory(sess, crate_name)?; let res = sess.opts.build_dep_graph().then(|| load_dep_graph(sess)); diff --git a/compiler/rustc_interface/messages.ftl b/compiler/rustc_interface/messages.ftl index 47dfbc1d7fbf1..87ed9e2461058 100644 --- a/compiler/rustc_interface/messages.ftl +++ b/compiler/rustc_interface/messages.ftl @@ -1,6 +1,10 @@ interface_cant_emit_mir = could not emit MIR: {$error} +interface_crate_name_does_not_match = `--crate-name` and `#[crate_name]` are required to match, but `{$crate_name}` != `{$attr_crate_name}` + +interface_crate_name_invalid = crate names cannot start with a `-`, but `{$crate_name}` has a leading hyphen + interface_emoji_identifier = identifiers cannot contain emoji: `{$ident}` diff --git a/compiler/rustc_interface/src/errors.rs b/compiler/rustc_interface/src/errors.rs index 29294003b8f29..cf4c87c1ab69d 100644 --- a/compiler/rustc_interface/src/errors.rs +++ b/compiler/rustc_interface/src/errors.rs @@ -4,6 +4,21 @@ use rustc_span::{Span, Symbol}; use std::io; use std::path::Path; +#[derive(Diagnostic)] +#[diag(interface_crate_name_does_not_match)] +pub(crate) struct CrateNameDoesNotMatch { + #[primary_span] + pub(crate) span: Span, + pub(crate) crate_name: Symbol, + pub(crate) attr_crate_name: Symbol, +} + +#[derive(Diagnostic)] +#[diag(interface_crate_name_invalid)] +pub(crate) struct CrateNameInvalid<'a> { + pub(crate) crate_name: &'a str, +} + #[derive(Diagnostic)] #[diag(interface_ferris_identifier)] pub struct FerrisIdentifier { diff --git a/compiler/rustc_interface/src/passes.rs b/compiler/rustc_interface/src/passes.rs index 2951f50b1f598..c5bd606bcc05f 100644 --- a/compiler/rustc_interface/src/passes.rs +++ b/compiler/rustc_interface/src/passes.rs @@ -27,12 +27,12 @@ use rustc_resolve::Resolver; use rustc_session::code_stats::VTableSizeInfo; use rustc_session::config::{CrateType, Input, OutFileName, OutputFilenames, OutputType}; use rustc_session::cstore::Untracked; +use rustc_session::output::collect_crate_types; use rustc_session::output::filename_for_input; -use rustc_session::output::{collect_crate_types, find_crate_name}; use rustc_session::search_paths::PathKind; use rustc_session::{Limit, Session}; -use rustc_span::symbol::{sym, Symbol}; use rustc_span::FileName; +use rustc_span::{sym, Span, Symbol}; use rustc_target::spec::PanicStrategy; use rustc_trait_selection::traits; @@ -660,8 +660,7 @@ pub(crate) fn create_global_ctxt<'tcx>( let pre_configured_attrs = rustc_expand::config::pre_configure_attrs(sess, &krate.attrs); - // parse `#[crate_name]` even if `--crate-name` was passed, to make sure it matches. - let crate_name = find_crate_name(sess, &pre_configured_attrs); + let (crate_name, crate_name_origin) = get_crate_name(sess, &pre_configured_attrs); let crate_types = collect_crate_types(sess, &pre_configured_attrs); let stable_crate_id = StableCrateId::new( crate_name, @@ -669,8 +668,9 @@ pub(crate) fn create_global_ctxt<'tcx>( sess.opts.cg.metadata.clone(), sess.cfg_version, ); - let outputs = util::build_output_filenames(&pre_configured_attrs, sess); - let dep_graph = setup_dep_graph(sess)?; + let outputs = util::build_output_filenames(sess, crate_name, crate_name_origin); + + let dep_graph = setup_dep_graph(sess, crate_name)?; let cstore = FreezeLock::new(Box::new(CStore::new(compiler.codegen_backend.metadata_loader())) as _); @@ -1044,23 +1044,91 @@ pub(crate) fn start_codegen<'tcx>( Ok(codegen) } -fn get_recursion_limit(krate_attrs: &[ast::Attribute], sess: &Session) -> Limit { - if let Some(attr) = krate_attrs - .iter() - .find(|attr| attr.has_name(sym::recursion_limit) && attr.value_str().is_none()) +/// Compute and validate the crate name. +pub fn get_crate_name(sess: &Session, krate_attrs: &[ast::Attribute]) -> (Symbol, CrateNameOrigin) { + // We unconditionally validate all `#![crate_name]`s even if a crate name was + // set on the command line via `--crate-name` which we prioritize over the + // crate attributes. We perform the validation here instead of later to ensure + // it gets run in all code paths requiring the crate name very early on. + // Namely, print requests (`--print`). + let attr_crate_name = + validate_and_find_value_str_builtin_attr(sym::crate_name, sess, krate_attrs); + + let validate = |name, span, origin| { + rustc_session::output::validate_crate_name(sess, name, span); + (name, origin) + }; + + if let Some(crate_name) = &sess.opts.crate_name { + let crate_name = Symbol::intern(crate_name); + if let Some((attr_crate_name, span)) = attr_crate_name + && attr_crate_name != crate_name + { + sess.dcx().emit_err(errors::CrateNameDoesNotMatch { + span, + crate_name, + attr_crate_name, + }); + } + return validate(crate_name, None, CrateNameOrigin::CliOpt); + } + + if let Some((crate_name, span)) = attr_crate_name { + return validate(crate_name, Some(span), CrateNameOrigin::CrateAttr); + } + + if let Input::File(ref path) = sess.io.input + && let Some(file_stem) = path.file_stem().and_then(|s| s.to_str()) { - // This is here mainly to check for using a macro, such as - // #![recursion_limit = foo!()]. That is not supported since that - // would require expanding this while in the middle of expansion, - // which needs to know the limit before expanding. Otherwise, - // validation would normally be caught in AstValidator (via - // `check_builtin_attribute`), but by the time that runs the macro - // is expanded, and it doesn't give an error. - validate_attr::emit_fatal_malformed_builtin_attribute( - &sess.psess, - attr, - sym::recursion_limit, - ); + if file_stem.starts_with('-') { + sess.dcx().emit_err(errors::CrateNameInvalid { crate_name: file_stem }); + } else { + let crate_name = file_stem.replace('-', "_"); + return validate(Symbol::intern(&crate_name), None, CrateNameOrigin::InputFile); + } } + + (Symbol::intern("rust_out"), CrateNameOrigin::Fallback) +} + +pub enum CrateNameOrigin { + CliOpt, + CrateAttr, + InputFile, + Fallback, +} + +fn get_recursion_limit(krate_attrs: &[ast::Attribute], sess: &Session) -> Limit { + // We don't permit macro calls inside of the attribute (e.g., #![recursion_limit = `expand!()`]) + // because that would require expanding this while in the middle of expansion, which needs to + // know the limit before expanding. + let _ = validate_and_find_value_str_builtin_attr(sym::recursion_limit, sess, krate_attrs); rustc_middle::middle::limits::get_recursion_limit(krate_attrs, sess) } + +/// Validate *all* occurrences of the given "[value-str]" built-in attribute and return the first find. +/// +/// This validator is intended for built-in attributes whose value needs to be known very early +/// during compilation (namely, before macro expansion) and it mainly exists to reject macro calls +/// inside of the attributes, such as in `#![name = expand!()]`. Normal attribute validation happens +/// during semantic analysis via [`TyCtxt::check_mod_attrs`] which happens *after* macro expansion +/// when such macro calls (here: `expand`) have already been expanded and we can no longer check for +/// their presence. +/// +/// [value-str]: ast::Attribute::value_str +fn validate_and_find_value_str_builtin_attr( + name: Symbol, + sess: &Session, + krate_attrs: &[ast::Attribute], +) -> Option<(Symbol, Span)> { + let mut result = None; + // Validate *all* relevant attributes, not just the first occurrence. + for attr in ast::attr::filter_by_name(krate_attrs, name) { + let Some(value) = attr.value_str() else { + validate_attr::emit_fatal_malformed_builtin_attribute(&sess.psess, attr, name) + }; + // Choose the first occurrence as our result. + result.get_or_insert((value, attr.span)); + } + result +} diff --git a/compiler/rustc_interface/src/util.rs b/compiler/rustc_interface/src/util.rs index 8dac524bb5bfd..558c6a86ac6a1 100644 --- a/compiler/rustc_interface/src/util.rs +++ b/compiler/rustc_interface/src/util.rs @@ -1,4 +1,5 @@ use crate::errors; +use crate::passes::CrateNameOrigin; use rustc_ast as ast; use rustc_codegen_ssa::traits::CodegenBackend; #[cfg(parallel_compiler)] @@ -14,8 +15,9 @@ use rustc_session::{filesearch, EarlyDiagCtxt, Session}; use rustc_span::edit_distance::find_best_match_for_name; use rustc_span::edition::Edition; use rustc_span::source_map::SourceMapInputs; -use rustc_span::symbol::sym; +use rustc_span::{sym, Symbol}; use rustc_target::spec::Target; + use std::env::consts::{DLL_PREFIX, DLL_SUFFIX}; use std::path::{Path, PathBuf}; use std::sync::atomic::{AtomicBool, Ordering}; @@ -413,11 +415,11 @@ pub(crate) fn check_attr_crate_type( } } else { // This is here mainly to check for using a macro, such as - // #![crate_type = foo!()]. That is not supported since the + // `#![crate_type = foo!()]`. That is not supported since the // crate type needs to be known very early in compilation long // before expansion. Otherwise, validation would normally be - // caught in AstValidator (via `check_builtin_attribute`), but - // by the time that runs the macro is expanded, and it doesn't + // caught during semantic analysis via `TyCtxt::check_mod_attrs`, + // but by the time that runs the macro is expanded, and it doesn't // give an error. validate_attr::emit_fatal_malformed_builtin_attribute( &sess.psess, @@ -453,7 +455,11 @@ fn multiple_output_types_to_stdout( } } -pub fn build_output_filenames(attrs: &[ast::Attribute], sess: &Session) -> OutputFilenames { +pub fn build_output_filenames( + sess: &Session, + crate_name: Symbol, + crate_name_origin: CrateNameOrigin, +) -> OutputFilenames { if multiple_output_types_to_stdout( &sess.opts.output_types, sess.io.output_file == Some(OutFileName::Stdout), @@ -461,12 +467,6 @@ pub fn build_output_filenames(attrs: &[ast::Attribute], sess: &Session) -> Outpu sess.dcx().emit_fatal(errors::MultipleOutputTypesToStdout); } - let crate_name = sess - .opts - .crate_name - .clone() - .or_else(|| rustc_attr::find_crate_name(attrs).map(|n| n.to_string())); - match sess.io.output_file { None => { // "-" as input file will cause the parser to read from stdin so we @@ -474,16 +474,22 @@ pub fn build_output_filenames(attrs: &[ast::Attribute], sess: &Session) -> Outpu // We want to toss everything after the final '.' let dirpath = sess.io.output_dir.clone().unwrap_or_default(); - // If a crate name is present, we use it as the link name - let stem = crate_name.clone().unwrap_or_else(|| sess.io.input.filestem().to_owned()); + // FIXME(fmease): Figure out a nicer way to do this. `stem` is almost + // identical to `crate_name` except when it was derived + // from a real(!) input file in which case they may differ + // by `-`/`_` only. + let stem = match crate_name_origin { + CrateNameOrigin::InputFile => sess.io.input.filestem(), + _ => crate_name.as_str(), + }; OutputFilenames::new( dirpath, - crate_name.unwrap_or_else(|| stem.replace('-', "_")), + crate_name, stem, None, sess.io.temps_dir.clone(), - sess.opts.cg.extra_filename.clone(), + &sess.opts.cg.extra_filename, sess.opts.output_types.clone(), ) } @@ -504,15 +510,14 @@ pub fn build_output_filenames(attrs: &[ast::Attribute], sess: &Session) -> Outpu sess.dcx().emit_warn(errors::IgnoringOutDir); } - let out_filestem = - out_file.filestem().unwrap_or_default().to_str().unwrap().to_string(); + let out_filestem = out_file.filestem().unwrap_or_default().to_str().unwrap(); OutputFilenames::new( out_file.parent().unwrap_or_else(|| Path::new("")).to_path_buf(), - crate_name.unwrap_or_else(|| out_filestem.replace('-', "_")), + crate_name, out_filestem, ofile, sess.io.temps_dir.clone(), - sess.opts.cg.extra_filename.clone(), + &sess.opts.cg.extra_filename, sess.opts.output_types.clone(), ) } diff --git a/compiler/rustc_session/messages.ftl b/compiler/rustc_session/messages.ftl index b84280a3ccf3f..994c7b61f0d80 100644 --- a/compiler/rustc_session/messages.ftl +++ b/compiler/rustc_session/messages.ftl @@ -8,12 +8,8 @@ session_cannot_mix_and_match_sanitizers = `-Zsanitizer={$first}` is incompatible session_cli_feature_diagnostic_help = add `-Zcrate-attr="feature({$feature})"` to the command-line options to enable -session_crate_name_does_not_match = `--crate-name` and `#[crate_name]` are required to match, but `{$s}` != `{$name}` - session_crate_name_empty = crate name must not be empty -session_crate_name_invalid = crate names cannot start with a `-`, but `{$s}` has a leading hyphen - session_expr_parentheses_needed = parentheses are required to parse this as an expression session_failed_to_create_profiler = failed to create profiler: {$err} @@ -48,8 +44,8 @@ session_instrumentation_not_supported = {$us} instrumentation is not supported f session_int_literal_too_large = integer literal is too large .note = value exceeds limit of `{$limit}` -session_invalid_character_in_create_name = invalid character `{$character}` in crate name: `{$crate_name}` -session_invalid_character_in_create_name_help = you can either pass `--crate-name` on the command line or add `#![crate_name="…"]` to set the crate name +session_invalid_character_in_crate_name = invalid character {$character} in crate name: `{$crate_name}` + .help = you can either pass `--crate-name` on the command line or add `#![crate_name = "…"]` to set the crate name session_invalid_float_literal_suffix = invalid suffix `{$suffix}` for float literal .label = invalid suffix `{$suffix}` diff --git a/compiler/rustc_session/src/config.rs b/compiler/rustc_session/src/config.rs index 41c99f7edeeff..46386937d0e24 100644 --- a/compiler/rustc_session/src/config.rs +++ b/compiler/rustc_session/src/config.rs @@ -18,7 +18,9 @@ use rustc_feature::UnstableFeatures; use rustc_macros::{Decodable, Encodable, HashStable_Generic}; use rustc_span::edition::{Edition, DEFAULT_EDITION, EDITION_NAME_LIST, LATEST_STABLE_EDITION}; use rustc_span::source_map::FilePathMapping; -use rustc_span::{FileName, FileNameDisplayPreference, RealFileName, SourceFileHashAlgorithm}; +use rustc_span::{ + FileName, FileNameDisplayPreference, RealFileName, SourceFileHashAlgorithm, Symbol, +}; use rustc_target::spec::{FramePointer, LinkSelfContainedComponents, LinkerFeatures}; use rustc_target::spec::{SplitDebuginfo, Target, TargetTriple}; use std::collections::btree_map::{ @@ -945,7 +947,7 @@ impl OutFileName { #[derive(Clone, Hash, Debug, HashStable_Generic, Encodable, Decodable)] pub struct OutputFilenames { pub(crate) out_directory: PathBuf, - /// Crate name. Never contains '-'. + /// Crate name (must not contain `-`) plus extra (`-Cextra-filename`). crate_stem: String, /// Typically based on `.rs` input file name. Any '-' is preserved. filestem: String, @@ -961,11 +963,11 @@ pub const DWARF_OBJECT_EXT: &str = "dwo"; impl OutputFilenames { pub fn new( out_directory: PathBuf, - out_crate_name: String, - out_filestem: String, + crate_name: Symbol, + out_filestem: &str, single_output_file: Option, temps_directory: Option, - extra: String, + extra: &str, outputs: OutputTypes, ) -> Self { OutputFilenames { @@ -973,7 +975,7 @@ impl OutputFilenames { single_output_file, temps_directory, outputs, - crate_stem: format!("{out_crate_name}{extra}"), + crate_stem: format!("{crate_name}{extra}"), filestem: format!("{out_filestem}{extra}"), } } diff --git a/compiler/rustc_session/src/errors.rs b/compiler/rustc_session/src/errors.rs index 4cbc1b570225b..fe6b9437f1ced 100644 --- a/compiler/rustc_session/src/errors.rs +++ b/compiler/rustc_session/src/errors.rs @@ -193,21 +193,6 @@ pub(crate) struct FileWriteFail<'a> { pub(crate) err: String, } -#[derive(Diagnostic)] -#[diag(session_crate_name_does_not_match)] -pub(crate) struct CrateNameDoesNotMatch { - #[primary_span] - pub(crate) span: Span, - pub(crate) s: Symbol, - pub(crate) name: Symbol, -} - -#[derive(Diagnostic)] -#[diag(session_crate_name_invalid)] -pub(crate) struct CrateNameInvalid<'a> { - pub(crate) s: &'a str, -} - #[derive(Diagnostic)] #[diag(session_crate_name_empty)] pub(crate) struct CrateNameEmpty { @@ -216,20 +201,14 @@ pub(crate) struct CrateNameEmpty { } #[derive(Diagnostic)] -#[diag(session_invalid_character_in_create_name)] +#[diag(session_invalid_character_in_crate_name)] pub(crate) struct InvalidCharacterInCrateName { #[primary_span] pub(crate) span: Option, pub(crate) character: char, pub(crate) crate_name: Symbol, - #[subdiagnostic] - pub(crate) crate_name_help: Option, -} - -#[derive(Subdiagnostic)] -pub(crate) enum InvalidCrateNameHelp { - #[help(session_invalid_character_in_create_name_help)] - AddCrateName, + #[help] + pub(crate) help: Option<()>, } #[derive(Subdiagnostic)] diff --git a/compiler/rustc_session/src/output.rs b/compiler/rustc_session/src/output.rs index 9a5314312e567..f2d1e14ace4cb 100644 --- a/compiler/rustc_session/src/output.rs +++ b/compiler/rustc_session/src/output.rs @@ -1,12 +1,9 @@ //! Related to out filenames of compilation (e.g. binaries). -use crate::config::{self, CrateType, Input, OutFileName, OutputFilenames, OutputType}; -use crate::errors::{ - self, CrateNameDoesNotMatch, CrateNameEmpty, CrateNameInvalid, FileIsNotWriteable, - InvalidCharacterInCrateName, InvalidCrateNameHelp, -}; +use crate::config::{self, CrateType, OutFileName, OutputFilenames, OutputType}; +use crate::errors::{self, CrateNameEmpty, FileIsNotWriteable, InvalidCharacterInCrateName}; use crate::Session; -use rustc_ast::{self as ast, attr}; +use rustc_ast as ast; use rustc_errors::FatalError; use rustc_span::symbol::sym; use rustc_span::{Span, Symbol}; @@ -49,74 +46,34 @@ fn is_writeable(p: &Path) -> bool { } } -pub fn find_crate_name(sess: &Session, attrs: &[ast::Attribute]) -> Symbol { - let validate = |s: Symbol, span: Option| { - validate_crate_name(sess, s, span); - s - }; - - // Look in attributes 100% of the time to make sure the attribute is marked - // as used. After doing this, however, we still prioritize a crate name from - // the command line over one found in the #[crate_name] attribute. If we - // find both we ensure that they're the same later on as well. - let attr_crate_name = - attr::find_by_name(attrs, sym::crate_name).and_then(|at| at.value_str().map(|s| (at, s))); +/// Validate the given crate name. +/// +/// Note that this validation is more permissive than identifier parsing. It considers +/// non-empty sequences of alphanumeric and underscore characters to be valid crate names. +/// Most notably, it accepts names starting with a numeric character like `0`! +/// +/// Furthermore, this shouldn't be taken as the canonical crate name validator. +/// Other places may use a more restrictive grammar (e.g., identifier or ASCII identifier). +pub fn validate_crate_name(sess: &Session, crate_name: Symbol, span: Option) { + let mut result = Ok(()); - if let Some(ref s) = sess.opts.crate_name { - let s = Symbol::intern(s); - if let Some((attr, name)) = attr_crate_name { - if name != s { - sess.dcx().emit_err(CrateNameDoesNotMatch { span: attr.span, s, name }); - } - } - return validate(s, None); + if crate_name.is_empty() { + result = Err(sess.dcx().emit_err(CrateNameEmpty { span })); } - if let Some((attr, s)) = attr_crate_name { - return validate(s, Some(attr.span)); - } - if let Input::File(ref path) = sess.io.input { - if let Some(s) = path.file_stem().and_then(|s| s.to_str()) { - if s.starts_with('-') { - sess.dcx().emit_err(CrateNameInvalid { s }); - } else { - return validate(Symbol::intern(&s.replace('-', "_")), None); - } - } - } - - Symbol::intern("rust_out") -} - -pub fn validate_crate_name(sess: &Session, s: Symbol, sp: Option) { - let mut err_count = 0; - { - if s.is_empty() { - err_count += 1; - sess.dcx().emit_err(CrateNameEmpty { span: sp }); - } - for c in s.as_str().chars() { - if c.is_alphanumeric() { - continue; - } - if c == '_' { - continue; - } - err_count += 1; - sess.dcx().emit_err(InvalidCharacterInCrateName { - span: sp, - character: c, - crate_name: s, - crate_name_help: if sp.is_none() { - Some(InvalidCrateNameHelp::AddCrateName) - } else { - None - }, - }); + for c in crate_name.as_str().chars() { + if c.is_alphanumeric() || c == '_' { + continue; } + result = Err(sess.dcx().emit_err(InvalidCharacterInCrateName { + span, + character: c, + crate_name, + help: span.is_none().then_some(()), + })); } - if err_count > 0 { + if result.is_err() { FatalError.raise(); } } diff --git a/tests/ui/attributes/crate-name-macro-call.rs b/tests/ui/attributes/crate-name-macro-call.rs new file mode 100644 index 0000000000000..1aae2e506e29b --- /dev/null +++ b/tests/ui/attributes/crate-name-macro-call.rs @@ -0,0 +1,6 @@ +// issue: rust-lang/rust#122001 +// Ensure we reject macro calls inside `#![crate_name]` as their result wouldn't get honored anyway. + +#![crate_name = concat!("my", "crate")] //~ ERROR malformed `crate_name` attribute input + +fn main() {} diff --git a/tests/ui/attributes/crate-name-macro-call.stderr b/tests/ui/attributes/crate-name-macro-call.stderr new file mode 100644 index 0000000000000..ab562b41a31f4 --- /dev/null +++ b/tests/ui/attributes/crate-name-macro-call.stderr @@ -0,0 +1,8 @@ +error: malformed `crate_name` attribute input + --> $DIR/crate-name-macro-call.rs:4:1 + | +LL | #![crate_name = concat!("my", "crate")] + | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ help: must be of the form: `#![crate_name = "name"]` + +error: aborting due to 1 previous error + diff --git a/tests/ui/invalid_crate_type_syntax.rs b/tests/ui/attributes/crate-type-delimited.rs similarity index 100% rename from tests/ui/invalid_crate_type_syntax.rs rename to tests/ui/attributes/crate-type-delimited.rs diff --git a/tests/ui/invalid_crate_type_syntax.stderr b/tests/ui/attributes/crate-type-delimited.stderr similarity index 82% rename from tests/ui/invalid_crate_type_syntax.stderr rename to tests/ui/attributes/crate-type-delimited.stderr index a5563720f6223..0bbbe07b19810 100644 --- a/tests/ui/invalid_crate_type_syntax.stderr +++ b/tests/ui/attributes/crate-type-delimited.stderr @@ -1,5 +1,5 @@ error: malformed `crate_type` attribute input - --> $DIR/invalid_crate_type_syntax.rs:2:1 + --> $DIR/crate-type-delimited.rs:2:1 | LL | #![crate_type(lib)] | ^^^^^^^^^^^^^^^^^^^ help: must be of the form: `#![crate_type = "bin|lib|..."]` diff --git a/tests/ui/no_crate_type.rs b/tests/ui/attributes/crate-type-empty.rs similarity index 100% rename from tests/ui/no_crate_type.rs rename to tests/ui/attributes/crate-type-empty.rs diff --git a/tests/ui/no_crate_type.stderr b/tests/ui/attributes/crate-type-empty.stderr similarity index 84% rename from tests/ui/no_crate_type.stderr rename to tests/ui/attributes/crate-type-empty.stderr index 85d8f87afa673..b9279d961eef4 100644 --- a/tests/ui/no_crate_type.stderr +++ b/tests/ui/attributes/crate-type-empty.stderr @@ -1,5 +1,5 @@ error: malformed `crate_type` attribute input - --> $DIR/no_crate_type.rs:2:1 + --> $DIR/crate-type-empty.rs:2:1 | LL | #![crate_type] | ^^^^^^^^^^^^^^ help: must be of the form: `#![crate_type = "bin|lib|..."]` diff --git a/tests/ui/invalid/invalid-crate-type-macro.rs b/tests/ui/attributes/crate-type-macro-call.rs similarity index 100% rename from tests/ui/invalid/invalid-crate-type-macro.rs rename to tests/ui/attributes/crate-type-macro-call.rs diff --git a/tests/ui/invalid/invalid-crate-type-macro.stderr b/tests/ui/attributes/crate-type-macro-call.stderr similarity index 83% rename from tests/ui/invalid/invalid-crate-type-macro.stderr rename to tests/ui/attributes/crate-type-macro-call.stderr index 1cf77d4086bb0..6ccc3edf885e3 100644 --- a/tests/ui/invalid/invalid-crate-type-macro.stderr +++ b/tests/ui/attributes/crate-type-macro-call.stderr @@ -1,5 +1,5 @@ error: malformed `crate_type` attribute input - --> $DIR/invalid-crate-type-macro.rs:1:1 + --> $DIR/crate-type-macro-call.rs:1:1 | LL | #![crate_type = foo!()] | ^^^^^^^^^^^^^^^^^^^^^^^ help: must be of the form: `#![crate_type = "bin|lib|..."]` diff --git a/tests/ui/invalid-compile-flags/branch-protection-missing-pac-ret.BADFLAGS.stderr b/tests/ui/cli/branch-protection-missing-pac-ret.BADFLAGS.stderr similarity index 100% rename from tests/ui/invalid-compile-flags/branch-protection-missing-pac-ret.BADFLAGS.stderr rename to tests/ui/cli/branch-protection-missing-pac-ret.BADFLAGS.stderr diff --git a/tests/ui/invalid-compile-flags/branch-protection-missing-pac-ret.BADTARGET.stderr b/tests/ui/cli/branch-protection-missing-pac-ret.BADTARGET.stderr similarity index 100% rename from tests/ui/invalid-compile-flags/branch-protection-missing-pac-ret.BADTARGET.stderr rename to tests/ui/cli/branch-protection-missing-pac-ret.BADTARGET.stderr diff --git a/tests/ui/invalid-compile-flags/branch-protection-missing-pac-ret.rs b/tests/ui/cli/branch-protection-missing-pac-ret.rs similarity index 100% rename from tests/ui/invalid-compile-flags/branch-protection-missing-pac-ret.rs rename to tests/ui/cli/branch-protection-missing-pac-ret.rs diff --git a/tests/ui/invalid-compile-flags/codegen-option-without-group.rs b/tests/ui/cli/codegen-option-without-group.rs similarity index 100% rename from tests/ui/invalid-compile-flags/codegen-option-without-group.rs rename to tests/ui/cli/codegen-option-without-group.rs diff --git a/tests/ui/invalid-compile-flags/codegen-option-without-group.stderr b/tests/ui/cli/codegen-option-without-group.stderr similarity index 100% rename from tests/ui/invalid-compile-flags/codegen-option-without-group.stderr rename to tests/ui/cli/codegen-option-without-group.stderr diff --git a/tests/ui/invalid-compile-flags/debug-option-without-group.rs b/tests/ui/cli/debug-option-without-group.rs similarity index 100% rename from tests/ui/invalid-compile-flags/debug-option-without-group.rs rename to tests/ui/cli/debug-option-without-group.rs diff --git a/tests/ui/invalid-compile-flags/debug-option-without-group.stderr b/tests/ui/cli/debug-option-without-group.stderr similarity index 100% rename from tests/ui/invalid-compile-flags/debug-option-without-group.stderr rename to tests/ui/cli/debug-option-without-group.stderr diff --git a/tests/ui/extern-flag/auxiliary/panic_handler.rs b/tests/ui/cli/extern-flag/auxiliary/panic_handler.rs similarity index 100% rename from tests/ui/extern-flag/auxiliary/panic_handler.rs rename to tests/ui/cli/extern-flag/auxiliary/panic_handler.rs diff --git a/tests/ui/extern-flag/auxiliary/somedep.rs b/tests/ui/cli/extern-flag/auxiliary/somedep.rs similarity index 100% rename from tests/ui/extern-flag/auxiliary/somedep.rs rename to tests/ui/cli/extern-flag/auxiliary/somedep.rs diff --git a/tests/ui/extern-flag/empty-extern-arg.rs b/tests/ui/cli/extern-flag/empty-extern-arg.rs similarity index 100% rename from tests/ui/extern-flag/empty-extern-arg.rs rename to tests/ui/cli/extern-flag/empty-extern-arg.rs diff --git a/tests/ui/extern-flag/empty-extern-arg.stderr b/tests/ui/cli/extern-flag/empty-extern-arg.stderr similarity index 100% rename from tests/ui/extern-flag/empty-extern-arg.stderr rename to tests/ui/cli/extern-flag/empty-extern-arg.stderr diff --git a/tests/ui/extern-flag/force-extern.rs b/tests/ui/cli/extern-flag/force-extern.rs similarity index 100% rename from tests/ui/extern-flag/force-extern.rs rename to tests/ui/cli/extern-flag/force-extern.rs diff --git a/tests/ui/extern-flag/invalid-crate-name-dashed.rs b/tests/ui/cli/extern-flag/invalid-crate-name-dashed.rs similarity index 100% rename from tests/ui/extern-flag/invalid-crate-name-dashed.rs rename to tests/ui/cli/extern-flag/invalid-crate-name-dashed.rs diff --git a/tests/ui/extern-flag/invalid-crate-name-dashed.stderr b/tests/ui/cli/extern-flag/invalid-crate-name-dashed.stderr similarity index 100% rename from tests/ui/extern-flag/invalid-crate-name-dashed.stderr rename to tests/ui/cli/extern-flag/invalid-crate-name-dashed.stderr diff --git a/tests/ui/extern-flag/invalid-crate-name-non-ascii.rs b/tests/ui/cli/extern-flag/invalid-crate-name-non-ascii.rs similarity index 100% rename from tests/ui/extern-flag/invalid-crate-name-non-ascii.rs rename to tests/ui/cli/extern-flag/invalid-crate-name-non-ascii.rs diff --git a/tests/ui/extern-flag/invalid-crate-name-non-ascii.stderr b/tests/ui/cli/extern-flag/invalid-crate-name-non-ascii.stderr similarity index 100% rename from tests/ui/extern-flag/invalid-crate-name-non-ascii.stderr rename to tests/ui/cli/extern-flag/invalid-crate-name-non-ascii.stderr diff --git a/tests/ui/extern-flag/invalid-crate-name.rs b/tests/ui/cli/extern-flag/invalid-crate-name.rs similarity index 100% rename from tests/ui/extern-flag/invalid-crate-name.rs rename to tests/ui/cli/extern-flag/invalid-crate-name.rs diff --git a/tests/ui/extern-flag/invalid-crate-name.stderr b/tests/ui/cli/extern-flag/invalid-crate-name.stderr similarity index 100% rename from tests/ui/extern-flag/invalid-crate-name.stderr rename to tests/ui/cli/extern-flag/invalid-crate-name.stderr diff --git a/tests/ui/extern-flag/multiple-opts.rs b/tests/ui/cli/extern-flag/multiple-opts.rs similarity index 100% rename from tests/ui/extern-flag/multiple-opts.rs rename to tests/ui/cli/extern-flag/multiple-opts.rs diff --git a/tests/ui/extern-flag/multiple-opts.stderr b/tests/ui/cli/extern-flag/multiple-opts.stderr similarity index 100% rename from tests/ui/extern-flag/multiple-opts.stderr rename to tests/ui/cli/extern-flag/multiple-opts.stderr diff --git a/tests/ui/extern-flag/no-force-extern.rs b/tests/ui/cli/extern-flag/no-force-extern.rs similarity index 100% rename from tests/ui/extern-flag/no-force-extern.rs rename to tests/ui/cli/extern-flag/no-force-extern.rs diff --git a/tests/ui/extern-flag/no-nounused.rs b/tests/ui/cli/extern-flag/no-nounused.rs similarity index 100% rename from tests/ui/extern-flag/no-nounused.rs rename to tests/ui/cli/extern-flag/no-nounused.rs diff --git a/tests/ui/extern-flag/no-nounused.stderr b/tests/ui/cli/extern-flag/no-nounused.stderr similarity index 100% rename from tests/ui/extern-flag/no-nounused.stderr rename to tests/ui/cli/extern-flag/no-nounused.stderr diff --git a/tests/ui/extern-flag/noprelude-and-prelude.rs b/tests/ui/cli/extern-flag/noprelude-and-prelude.rs similarity index 100% rename from tests/ui/extern-flag/noprelude-and-prelude.rs rename to tests/ui/cli/extern-flag/noprelude-and-prelude.rs diff --git a/tests/ui/extern-flag/noprelude-resolves.rs b/tests/ui/cli/extern-flag/noprelude-resolves.rs similarity index 100% rename from tests/ui/extern-flag/noprelude-resolves.rs rename to tests/ui/cli/extern-flag/noprelude-resolves.rs diff --git a/tests/ui/extern-flag/noprelude.rs b/tests/ui/cli/extern-flag/noprelude.rs similarity index 100% rename from tests/ui/extern-flag/noprelude.rs rename to tests/ui/cli/extern-flag/noprelude.rs diff --git a/tests/ui/extern-flag/noprelude.stderr b/tests/ui/cli/extern-flag/noprelude.stderr similarity index 100% rename from tests/ui/extern-flag/noprelude.stderr rename to tests/ui/cli/extern-flag/noprelude.stderr diff --git a/tests/ui/extern-flag/nounused.rs b/tests/ui/cli/extern-flag/nounused.rs similarity index 100% rename from tests/ui/extern-flag/nounused.rs rename to tests/ui/cli/extern-flag/nounused.rs diff --git a/tests/ui/extern-flag/public-and-private.rs b/tests/ui/cli/extern-flag/public-and-private.rs similarity index 100% rename from tests/ui/extern-flag/public-and-private.rs rename to tests/ui/cli/extern-flag/public-and-private.rs diff --git a/tests/ui/extern-flag/public-and-private.stderr b/tests/ui/cli/extern-flag/public-and-private.stderr similarity index 100% rename from tests/ui/extern-flag/public-and-private.stderr rename to tests/ui/cli/extern-flag/public-and-private.stderr diff --git a/tests/ui/extern-flag/redundant-force-extern.rs b/tests/ui/cli/extern-flag/redundant-force-extern.rs similarity index 100% rename from tests/ui/extern-flag/redundant-force-extern.rs rename to tests/ui/cli/extern-flag/redundant-force-extern.rs diff --git a/tests/ui/invalid-compile-flags/fuel.rs b/tests/ui/cli/fuel.rs similarity index 100% rename from tests/ui/invalid-compile-flags/fuel.rs rename to tests/ui/cli/fuel.rs diff --git a/tests/ui/invalid-compile-flags/function-return/requires-x86-or-x86_64.aarch64.stderr b/tests/ui/cli/function-return/requires-x86-or-x86_64.aarch64.stderr similarity index 100% rename from tests/ui/invalid-compile-flags/function-return/requires-x86-or-x86_64.aarch64.stderr rename to tests/ui/cli/function-return/requires-x86-or-x86_64.aarch64.stderr diff --git a/tests/ui/invalid-compile-flags/function-return/requires-x86-or-x86_64.rs b/tests/ui/cli/function-return/requires-x86-or-x86_64.rs similarity index 100% rename from tests/ui/invalid-compile-flags/function-return/requires-x86-or-x86_64.rs rename to tests/ui/cli/function-return/requires-x86-or-x86_64.rs diff --git a/tests/ui/invalid-compile-flags/function-return/thunk-extern-requires-non-large-code-model.large.stderr b/tests/ui/cli/function-return/thunk-extern-requires-non-large-code-model.large.stderr similarity index 100% rename from tests/ui/invalid-compile-flags/function-return/thunk-extern-requires-non-large-code-model.large.stderr rename to tests/ui/cli/function-return/thunk-extern-requires-non-large-code-model.large.stderr diff --git a/tests/ui/invalid-compile-flags/function-return/thunk-extern-requires-non-large-code-model.rs b/tests/ui/cli/function-return/thunk-extern-requires-non-large-code-model.rs similarity index 100% rename from tests/ui/invalid-compile-flags/function-return/thunk-extern-requires-non-large-code-model.rs rename to tests/ui/cli/function-return/thunk-extern-requires-non-large-code-model.rs diff --git a/tests/ui/invalid-compile-flags/invalid-llvm-passes.rs b/tests/ui/cli/invalid-llvm-passes.rs similarity index 100% rename from tests/ui/invalid-compile-flags/invalid-llvm-passes.rs rename to tests/ui/cli/invalid-llvm-passes.rs diff --git a/tests/ui/invalid-compile-flags/invalid-llvm-passes.stderr b/tests/ui/cli/invalid-llvm-passes.stderr similarity index 100% rename from tests/ui/invalid-compile-flags/invalid-llvm-passes.stderr rename to tests/ui/cli/invalid-llvm-passes.stderr diff --git a/tests/ui/command/need-crate-arg-ignore-tidy$x.rs b/tests/ui/cli/need-crate-arg-ignore-tidy$x.rs similarity index 100% rename from tests/ui/command/need-crate-arg-ignore-tidy$x.rs rename to tests/ui/cli/need-crate-arg-ignore-tidy$x.rs diff --git a/tests/ui/cli/need-crate-arg-ignore-tidy$x.stderr b/tests/ui/cli/need-crate-arg-ignore-tidy$x.stderr new file mode 100644 index 0000000000000..861510212f920 --- /dev/null +++ b/tests/ui/cli/need-crate-arg-ignore-tidy$x.stderr @@ -0,0 +1,6 @@ +error: invalid character '$' in crate name: `need_crate_arg_ignore_tidy$x` + | + = help: you can either pass `--crate-name` on the command line or add `#![crate_name = "…"]` to set the crate name + +error: aborting due to 1 previous error + diff --git a/tests/ui/cli/print-crate-name-request-malformed-crate-name.rs b/tests/ui/cli/print-crate-name-request-malformed-crate-name.rs new file mode 100644 index 0000000000000..c7f3c2c5403d1 --- /dev/null +++ b/tests/ui/cli/print-crate-name-request-malformed-crate-name.rs @@ -0,0 +1,5 @@ +// Ensure we validate `#![crate_name]` on print requests and reject macro calls inside of it. +// See also . + +//@ compile-flags: --print=crate-name +#![crate_name = concat!("wrapped")] //~ ERROR malformed `crate_name` attribute input diff --git a/tests/ui/cli/print-crate-name-request-malformed-crate-name.stderr b/tests/ui/cli/print-crate-name-request-malformed-crate-name.stderr new file mode 100644 index 0000000000000..6bf09a2b13171 --- /dev/null +++ b/tests/ui/cli/print-crate-name-request-malformed-crate-name.stderr @@ -0,0 +1,8 @@ +error: malformed `crate_name` attribute input + --> $DIR/print-crate-name-request-malformed-crate-name.rs:5:1 + | +LL | #![crate_name = concat!("wrapped")] + | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ help: must be of the form: `#![crate_name = "name"]` + +error: aborting due to 1 previous error + diff --git a/tests/ui/cli/print-file-names-request-malformed-crate-name-1.rs b/tests/ui/cli/print-file-names-request-malformed-crate-name-1.rs new file mode 100644 index 0000000000000..5cf1ae36b42b7 --- /dev/null +++ b/tests/ui/cli/print-file-names-request-malformed-crate-name-1.rs @@ -0,0 +1,4 @@ +// Ensure we validate `#![crate_name]` on print requests. + +//@ compile-flags: --print=file-names +#![crate_name] //~ ERROR malformed `crate_name` attribute input diff --git a/tests/ui/cli/print-file-names-request-malformed-crate-name-1.stderr b/tests/ui/cli/print-file-names-request-malformed-crate-name-1.stderr new file mode 100644 index 0000000000000..de62aed79fce5 --- /dev/null +++ b/tests/ui/cli/print-file-names-request-malformed-crate-name-1.stderr @@ -0,0 +1,8 @@ +error: malformed `crate_name` attribute input + --> $DIR/print-file-names-request-malformed-crate-name-1.rs:4:1 + | +LL | #![crate_name] + | ^^^^^^^^^^^^^^ help: must be of the form: `#![crate_name = "name"]` + +error: aborting due to 1 previous error + diff --git a/tests/ui/cli/print-file-names-request-malformed-crate-name-2.rs b/tests/ui/cli/print-file-names-request-malformed-crate-name-2.rs new file mode 100644 index 0000000000000..13c9d1e002739 --- /dev/null +++ b/tests/ui/cli/print-file-names-request-malformed-crate-name-2.rs @@ -0,0 +1,7 @@ +// Ensure that we validate *all* `#![crate_name]`s on print requests, not just the first, +// and that we reject macro calls inside of them. +// See also . + +//@ compile-flags: --print=file-names +#![crate_name = "this_one_is_okay"] +#![crate_name = concat!("this_one_is_not")] //~ ERROR malformed `crate_name` attribute input diff --git a/tests/ui/cli/print-file-names-request-malformed-crate-name-2.stderr b/tests/ui/cli/print-file-names-request-malformed-crate-name-2.stderr new file mode 100644 index 0000000000000..42c33de12210d --- /dev/null +++ b/tests/ui/cli/print-file-names-request-malformed-crate-name-2.stderr @@ -0,0 +1,8 @@ +error: malformed `crate_name` attribute input + --> $DIR/print-file-names-request-malformed-crate-name-2.rs:7:1 + | +LL | #![crate_name = concat!("this_one_is_not")] + | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ help: must be of the form: `#![crate_name = "name"]` + +error: aborting due to 1 previous error + diff --git a/tests/ui/cli/print-file-names-request-malformed-crate-name.rs b/tests/ui/cli/print-file-names-request-malformed-crate-name.rs new file mode 100644 index 0000000000000..5fe8bd7945f4a --- /dev/null +++ b/tests/ui/cli/print-file-names-request-malformed-crate-name.rs @@ -0,0 +1,5 @@ +// Ensure we validate `#![crate_name]` on print requests and reject macro calls inside of it. +// See also . + +//@ compile-flags: --print=file-names +#![crate_name = concat!("wrapped")] //~ ERROR malformed `crate_name` attribute input diff --git a/tests/ui/cli/print-file-names-request-malformed-crate-name.stderr b/tests/ui/cli/print-file-names-request-malformed-crate-name.stderr new file mode 100644 index 0000000000000..cb5ffaab9ca20 --- /dev/null +++ b/tests/ui/cli/print-file-names-request-malformed-crate-name.stderr @@ -0,0 +1,8 @@ +error: malformed `crate_name` attribute input + --> $DIR/print-file-names-request-malformed-crate-name.rs:5:1 + | +LL | #![crate_name = concat!("wrapped")] + | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ help: must be of the form: `#![crate_name = "name"]` + +error: aborting due to 1 previous error + diff --git a/tests/ui/invalid-compile-flags/print.rs b/tests/ui/cli/print.rs similarity index 100% rename from tests/ui/invalid-compile-flags/print.rs rename to tests/ui/cli/print.rs diff --git a/tests/ui/invalid-compile-flags/print.stderr b/tests/ui/cli/print.stderr similarity index 100% rename from tests/ui/invalid-compile-flags/print.stderr rename to tests/ui/cli/print.stderr diff --git a/tests/ui/command/need-crate-arg-ignore-tidy$x.stderr b/tests/ui/command/need-crate-arg-ignore-tidy$x.stderr deleted file mode 100644 index 28f6d31b1ce59..0000000000000 --- a/tests/ui/command/need-crate-arg-ignore-tidy$x.stderr +++ /dev/null @@ -1,6 +0,0 @@ -error: invalid character `'$'` in crate name: `need_crate_arg_ignore_tidy$x` - | - = help: you can either pass `--crate-name` on the command line or add `#![crate_name="…"]` to set the crate name - -error: aborting due to 1 previous error -