• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // rustc-cfg emitted by the build script:
2 //
3 // "wrap_proc_macro"
4 //     Wrap types from libproc_macro rather than polyfilling the whole API.
5 //     Enabled on rustc 1.29+ as long as procmacro2_semver_exempt is not set,
6 //     because we can't emulate the unstable API without emulating everything
7 //     else. Also enabled unconditionally on nightly, in which case the
8 //     procmacro2_semver_exempt surface area is implemented by using the
9 //     nightly-only proc_macro API.
10 //
11 // "hygiene"
12 //    Enable Span::mixed_site() and non-dummy behavior of Span::resolved_at
13 //    and Span::located_at. Enabled on Rust 1.45+.
14 //
15 // "proc_macro_span"
16 //     Enable non-dummy behavior of Span::start and Span::end methods which
17 //     requires an unstable compiler feature. Enabled when building with
18 //     nightly, unless `-Z allow-feature` in RUSTFLAGS disallows unstable
19 //     features.
20 //
21 // "super_unstable"
22 //     Implement the semver exempt API in terms of the nightly-only proc_macro
23 //     API. Enabled when using procmacro2_semver_exempt on a nightly compiler.
24 //
25 // "span_locations"
26 //     Provide methods Span::start and Span::end which give the line/column
27 //     location of a token. Enabled by procmacro2_semver_exempt or the
28 //     "span-locations" Cargo cfg. This is behind a cfg because tracking
29 //     location inside spans is a performance hit.
30 //
31 // "is_available"
32 //     Use proc_macro::is_available() to detect if the proc macro API is
33 //     available or needs to be polyfilled instead of trying to use the proc
34 //     macro API and catching a panic if it isn't available. Enabled on Rust
35 //     1.57+.
36 
37 use std::env;
38 use std::ffi::OsString;
39 use std::path::Path;
40 use std::process::{self, Command, Stdio};
41 use std::str;
42 use std::u32;
43 
main()44 fn main() {
45     let rustc = rustc_minor_version().unwrap_or(u32::MAX);
46 
47     let docs_rs = env::var_os("DOCS_RS").is_some();
48     let semver_exempt = cfg!(procmacro2_semver_exempt) || docs_rs;
49     if semver_exempt {
50         // https://github.com/dtolnay/proc-macro2/issues/147
51         println!("cargo:rustc-cfg=procmacro2_semver_exempt");
52     }
53 
54     if semver_exempt || cfg!(feature = "span-locations") {
55         println!("cargo:rustc-cfg=span_locations");
56     }
57 
58     if rustc < 57 {
59         println!("cargo:rustc-cfg=no_is_available");
60     }
61 
62     if rustc < 66 {
63         println!("cargo:rustc-cfg=no_source_text");
64     }
65 
66     if !cfg!(feature = "proc-macro") {
67         println!("cargo:rerun-if-changed=build.rs");
68         return;
69     }
70 
71     println!("cargo:rerun-if-changed=build/probe.rs");
72 
73     let proc_macro_span;
74     let consider_rustc_bootstrap;
75     if compile_probe(false) {
76         // This is a nightly or dev compiler, so it supports unstable features
77         // regardless of RUSTC_BOOTSTRAP. No need to rerun build script if
78         // RUSTC_BOOTSTRAP is changed.
79         proc_macro_span = true;
80         consider_rustc_bootstrap = false;
81     } else if let Some(rustc_bootstrap) = env::var_os("RUSTC_BOOTSTRAP") {
82         if compile_probe(true) {
83             // This is a stable or beta compiler for which the user has set
84             // RUSTC_BOOTSTRAP to turn on unstable features. Rerun build script
85             // if they change it.
86             proc_macro_span = true;
87             consider_rustc_bootstrap = true;
88         } else if rustc_bootstrap == "1" {
89             // This compiler does not support the proc macro Span API in the
90             // form that proc-macro2 expects. No need to pay attention to
91             // RUSTC_BOOTSTRAP.
92             proc_macro_span = false;
93             consider_rustc_bootstrap = false;
94         } else {
95             // This is a stable or beta compiler for which RUSTC_BOOTSTRAP is
96             // set to restrict the use of unstable features by this crate.
97             proc_macro_span = false;
98             consider_rustc_bootstrap = true;
99         }
100     } else {
101         // Without RUSTC_BOOTSTRAP, this compiler does not support the proc
102         // macro Span API in the form that proc-macro2 expects, but try again if
103         // the user turns on unstable features.
104         proc_macro_span = false;
105         consider_rustc_bootstrap = true;
106     }
107 
108     if proc_macro_span || !semver_exempt {
109         println!("cargo:rustc-cfg=wrap_proc_macro");
110     }
111 
112     if proc_macro_span {
113         println!("cargo:rustc-cfg=proc_macro_span");
114     }
115 
116     if semver_exempt && proc_macro_span {
117         println!("cargo:rustc-cfg=super_unstable");
118     }
119 
120     if consider_rustc_bootstrap {
121         println!("cargo:rerun-if-env-changed=RUSTC_BOOTSTRAP");
122     }
123 }
124 
compile_probe(rustc_bootstrap: bool) -> bool125 fn compile_probe(rustc_bootstrap: bool) -> bool {
126     if env::var_os("RUSTC_STAGE").is_some() {
127         // We are running inside rustc bootstrap. This is a highly non-standard
128         // environment with issues such as:
129         //
130         //     https://github.com/rust-lang/cargo/issues/11138
131         //     https://github.com/rust-lang/rust/issues/114839
132         //
133         // Let's just not use nightly features here.
134         return false;
135     }
136 
137     let rustc = cargo_env_var("RUSTC");
138     let out_dir = cargo_env_var("OUT_DIR");
139     let probefile = Path::new("build").join("probe.rs");
140 
141     // Make sure to pick up Cargo rustc configuration.
142     let mut cmd = if let Some(wrapper) = env::var_os("RUSTC_WRAPPER") {
143         let mut cmd = Command::new(wrapper);
144         // The wrapper's first argument is supposed to be the path to rustc.
145         cmd.arg(rustc);
146         cmd
147     } else {
148         Command::new(rustc)
149     };
150 
151     if !rustc_bootstrap {
152         cmd.env_remove("RUSTC_BOOTSTRAP");
153     }
154 
155     cmd.stderr(Stdio::null())
156         .arg("--edition=2021")
157         .arg("--crate-name=proc_macro2")
158         .arg("--crate-type=lib")
159         .arg("--emit=dep-info,metadata")
160         .arg("--out-dir")
161         .arg(out_dir)
162         .arg(probefile);
163 
164     if let Some(target) = env::var_os("TARGET") {
165         cmd.arg("--target").arg(target);
166     }
167 
168     // If Cargo wants to set RUSTFLAGS, use that.
169     if let Ok(rustflags) = env::var("CARGO_ENCODED_RUSTFLAGS") {
170         if !rustflags.is_empty() {
171             for arg in rustflags.split('\x1f') {
172                 cmd.arg(arg);
173             }
174         }
175     }
176 
177     match cmd.status() {
178         Ok(status) => status.success(),
179         Err(_) => false,
180     }
181 }
182 
rustc_minor_version() -> Option<u32>183 fn rustc_minor_version() -> Option<u32> {
184     let rustc = cargo_env_var("RUSTC");
185     let output = Command::new(rustc).arg("--version").output().ok()?;
186     let version = str::from_utf8(&output.stdout).ok()?;
187     let mut pieces = version.split('.');
188     if pieces.next() != Some("rustc 1") {
189         return None;
190     }
191     pieces.next()?.parse().ok()
192 }
193 
cargo_env_var(key: &str) -> OsString194 fn cargo_env_var(key: &str) -> OsString {
195     env::var_os(key).unwrap_or_else(|| {
196         eprintln!(
197             "Environment variable ${} is not set during execution of build script",
198             key,
199         );
200         process::exit(1);
201     })
202 }
203