Skip to main content

rustc_session/config/
externs.rs

1//! This module contains code to help parse and manipulate `--extern` arguments.
2
3use std::path::PathBuf;
4
5use rustc_errors::{Diag, FatalAbort};
6
7use super::UnstableOptions;
8use crate::EarlyDiagCtxt;
9
10#[cfg(test)]
11mod tests;
12
13/// Represents the pieces of an `--extern` argument.
14pub(crate) struct ExternOpt {
15    pub(crate) crate_name: String,
16    pub(crate) path: Option<PathBuf>,
17    pub(crate) options: Option<String>,
18}
19
20/// Breaks out the major components of an `--extern` argument.
21///
22/// The options field will be a string containing comma-separated options that will need further
23/// parsing and processing.
24pub(crate) fn split_extern_opt<'a>(
25    early_dcx: &'a EarlyDiagCtxt,
26    unstable_opts: &UnstableOptions,
27    extern_opt: &str,
28) -> Result<ExternOpt, Diag<'a, FatalAbort>> {
29    let (name, path) = match extern_opt.split_once('=') {
30        None => (extern_opt.to_string(), None),
31        Some((name, path)) => (name.to_string(), Some(PathBuf::from(path))),
32    };
33    let (options, crate_name) = match name.split_once(':') {
34        None => (None, name),
35        Some((opts, crate_name)) => {
36            if unstable_opts.namespaced_crates && crate_name.starts_with(':') {
37                // If the name starts with `:`, we know this was actually something like `foo::bar` and
38                // not a set of options. We can just use the original name as the crate name.
39                (None, name)
40            } else {
41                (Some(opts.to_string()), crate_name.to_string())
42            }
43        }
44    };
45
46    if !valid_crate_name(&crate_name, unstable_opts) {
47        let mut error = early_dcx.early_struct_fatal(::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("crate name `{0}` passed to `--extern` is not a valid ASCII identifier",
                crate_name))
    })format!(
48            "crate name `{crate_name}` passed to `--extern` is not a valid ASCII identifier"
49        ));
50        let adjusted_name = crate_name.replace('-', "_");
51        if is_ascii_ident(&adjusted_name) {
52            error
53                .help(::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("consider replacing the dashes with underscores: `{0}`",
                adjusted_name))
    })format!("consider replacing the dashes with underscores: `{adjusted_name}`"));
54        }
55        return Err(error);
56    }
57
58    Ok(ExternOpt { crate_name, path, options })
59}
60
61fn valid_crate_name(name: &str, unstable_opts: &UnstableOptions) -> bool {
62    match name.split_once("::") {
63        Some((a, b)) if unstable_opts.namespaced_crates => is_ascii_ident(a) && is_ascii_ident(b),
64        Some(_) => false,
65        None => is_ascii_ident(name),
66    }
67}
68
69fn is_ascii_ident(string: &str) -> bool {
70    let mut chars = string.chars();
71    if let Some(start) = chars.next()
72        && (start.is_ascii_alphabetic() || start == '_')
73    {
74        chars.all(|char| char.is_ascii_alphanumeric() || char == '_')
75    } else {
76        false
77    }
78}