1use std::hash::Hash;
24use std::iter;
25
26use rustc_abi::Align;
27use rustc_ast::ast;
28use rustc_data_structures::fx::{FxHashMap, FxHashSet, FxIndexSet};
29use rustc_lint_defs::builtin::EXPLICIT_BUILTIN_CFGS_IN_FLAGS;
30use rustc_span::{Symbol, sym};
31use rustc_target::spec::{PanicStrategy, RelocModel, SanitizerSet, Target};
32
33use crate::config::{CrateType, FmtDebug};
34use crate::{Session, errors};
35
36pub type Cfg = FxIndexSet<(Symbol, Option<Symbol>)>;
42
43#[derive(#[automatically_derived]
impl ::core::default::Default for CheckCfg {
#[inline]
fn default() -> CheckCfg {
CheckCfg {
exhaustive_names: ::core::default::Default::default(),
exhaustive_values: ::core::default::Default::default(),
expecteds: ::core::default::Default::default(),
well_known_names: ::core::default::Default::default(),
}
}
}Default)]
45pub struct CheckCfg {
46 pub exhaustive_names: bool,
48 pub exhaustive_values: bool,
50 pub expecteds: FxHashMap<Symbol, ExpectedValues<Symbol>>,
52 pub well_known_names: FxHashSet<Symbol>,
54}
55
56pub enum ExpectedValues<T> {
57 Some(FxHashSet<Option<T>>),
58 Any,
59}
60
61impl<T: Eq + Hash> ExpectedValues<T> {
62 fn insert(&mut self, value: T) -> bool {
63 match self {
64 ExpectedValues::Some(expecteds) => expecteds.insert(Some(value)),
65 ExpectedValues::Any => false,
66 }
67 }
68
69 pub fn contains(&self, value: &Option<T>) -> bool {
70 match self {
71 ExpectedValues::Some(expecteds) => expecteds.contains(value),
72 ExpectedValues::Any => false,
73 }
74 }
75}
76
77impl<T: Eq + Hash> Extend<T> for ExpectedValues<T> {
78 fn extend<I: IntoIterator<Item = T>>(&mut self, iter: I) {
79 match self {
80 ExpectedValues::Some(expecteds) => expecteds.extend(iter.into_iter().map(Some)),
81 ExpectedValues::Any => {}
82 }
83 }
84}
85
86impl<'a, T: Eq + Hash + Copy + 'a> Extend<&'a T> for ExpectedValues<T> {
87 fn extend<I: IntoIterator<Item = &'a T>>(&mut self, iter: I) {
88 match self {
89 ExpectedValues::Some(expecteds) => expecteds.extend(iter.into_iter().map(|a| Some(*a))),
90 ExpectedValues::Any => {}
91 }
92 }
93}
94
95pub(crate) fn disallow_cfgs(sess: &Session, user_cfgs: &Cfg) {
97 let disallow = |cfg: &(Symbol, Option<Symbol>), controlled_by| {
98 let cfg_name = cfg.0;
99 let cfg = if let Some(value) = cfg.1 {
100 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0}=\"{1}\"", cfg_name, value))
})format!(r#"{}="{}""#, cfg_name, value)
101 } else {
102 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0}", cfg_name))
})format!("{}", cfg_name)
103 };
104 sess.psess.opt_span_buffer_lint(
105 EXPLICIT_BUILTIN_CFGS_IN_FLAGS,
106 None,
107 ast::CRATE_NODE_ID,
108 errors::UnexpectedBuiltinCfg { cfg, cfg_name, controlled_by }.into(),
109 )
110 };
111
112 for cfg in user_cfgs {
124 match cfg {
125 (sym::overflow_checks, None) => disallow(cfg, "-C overflow-checks"),
126 (sym::debug_assertions, None) => disallow(cfg, "-C debug-assertions"),
127 (sym::ub_checks, None) => disallow(cfg, "-Z ub-checks"),
128 (sym::contract_checks, None) => disallow(cfg, "-Z contract-checks"),
129 (sym::sanitize, None | Some(_)) => disallow(cfg, "-Z sanitizer"),
130 (
131 sym::sanitizer_cfi_generalize_pointers | sym::sanitizer_cfi_normalize_integers,
132 None | Some(_),
133 ) => disallow(cfg, "-Z sanitizer=cfi"),
134 (sym::proc_macro, None) => disallow(cfg, "--crate-type proc-macro"),
135 (sym::panic, Some(sym::abort | sym::unwind | sym::immediate_abort)) => {
136 disallow(cfg, "-C panic")
137 }
138 (sym::target_feature, Some(_)) => disallow(cfg, "-C target-feature"),
139 (sym::unix, None)
140 | (sym::windows, None)
141 | (sym::relocation_model, Some(_))
142 | (sym::target_abi, None | Some(_))
143 | (sym::target_arch, Some(_))
144 | (sym::target_endian, Some(_))
145 | (sym::target_env, None | Some(_))
146 | (sym::target_family, Some(_))
147 | (sym::target_os, Some(_))
148 | (sym::target_pointer_width, Some(_))
149 | (sym::target_vendor, None | Some(_))
150 | (sym::target_has_atomic, Some(_))
151 | (sym::target_has_atomic_equal_alignment, Some(_))
152 | (sym::target_has_atomic_load_store, Some(_))
153 | (sym::target_has_reliable_f16, None | Some(_))
154 | (sym::target_has_reliable_f16_math, None | Some(_))
155 | (sym::target_has_reliable_f128, None | Some(_))
156 | (sym::target_has_reliable_f128_math, None | Some(_))
157 | (sym::target_thread_local, None) => disallow(cfg, "--target"),
158 (sym::fmt_debug, None | Some(_)) => disallow(cfg, "-Z fmt-debug"),
159 (sym::emscripten_wasm_eh, None | Some(_)) => disallow(cfg, "-Z emscripten_wasm_eh"),
160 _ => {}
161 }
162 }
163}
164
165pub(crate) fn default_configuration(sess: &Session) -> Cfg {
167 let mut ret = Cfg::default();
168
169 macro_rules! ins_none {
170 ($key:expr) => {
171 ret.insert(($key, None));
172 };
173 }
174 macro_rules! ins_str {
175 ($key:expr, $val_str:expr) => {
176 ret.insert(($key, Some(Symbol::intern($val_str))));
177 };
178 }
179 macro_rules! ins_sym {
180 ($key:expr, $val_sym:expr) => {
181 ret.insert(($key, Some($val_sym)));
182 };
183 }
184
185 if sess.opts.debug_assertions {
194 ret.insert((sym::debug_assertions, None));ins_none!(sym::debug_assertions);
195 }
196
197 if sess.is_nightly_build() {
198 match sess.opts.unstable_opts.fmt_debug {
199 FmtDebug::Full => {
200 ret.insert((sym::fmt_debug, Some(sym::full)));ins_sym!(sym::fmt_debug, sym::full);
201 }
202 FmtDebug::Shallow => {
203 ret.insert((sym::fmt_debug, Some(sym::shallow)));ins_sym!(sym::fmt_debug, sym::shallow);
204 }
205 FmtDebug::None => {
206 ret.insert((sym::fmt_debug, Some(sym::none)));ins_sym!(sym::fmt_debug, sym::none);
207 }
208 }
209 }
210
211 if sess.overflow_checks() {
212 ret.insert((sym::overflow_checks, None));ins_none!(sym::overflow_checks);
213 }
214
215 ret.insert((sym::panic, Some(sess.panic_strategy().desc_symbol())));ins_sym!(sym::panic, sess.panic_strategy().desc_symbol());
220 if sess.panic_strategy() == PanicStrategy::ImmediateAbort {
221 ret.insert((sym::panic, Some(PanicStrategy::Abort.desc_symbol())));ins_sym!(sym::panic, PanicStrategy::Abort.desc_symbol());
222 }
223
224 #[allow(rustc::bad_opt_access)]
226 if sess.opts.crate_types.contains(&CrateType::ProcMacro) {
227 ret.insert((sym::proc_macro, None));ins_none!(sym::proc_macro);
228 }
229
230 if sess.is_nightly_build() {
231 ret.insert((sym::relocation_model,
Some(sess.target.relocation_model.desc_symbol())));ins_sym!(sym::relocation_model, sess.target.relocation_model.desc_symbol());
232 }
233
234 for mut s in sess.sanitizers() {
235 if s == SanitizerSet::KERNELADDRESS {
237 s = SanitizerSet::ADDRESS;
238 }
239 if s == SanitizerSet::KERNELHWADDRESS {
241 s = SanitizerSet::HWADDRESS;
242 }
243 ret.insert((sym::sanitize, Some(Symbol::intern(&s.to_string()))));ins_str!(sym::sanitize, &s.to_string());
244 }
245
246 if sess.is_sanitizer_cfi_generalize_pointers_enabled() {
247 ret.insert((sym::sanitizer_cfi_generalize_pointers, None));ins_none!(sym::sanitizer_cfi_generalize_pointers);
248 }
249 if sess.is_sanitizer_cfi_normalize_integers_enabled() {
250 ret.insert((sym::sanitizer_cfi_normalize_integers, None));ins_none!(sym::sanitizer_cfi_normalize_integers);
251 }
252
253 ret.insert((sym::target_abi, Some(sess.target.cfg_abi.desc_symbol())));ins_sym!(sym::target_abi, sess.target.cfg_abi.desc_symbol());
254 ret.insert((sym::target_arch, Some(sess.target.arch.desc_symbol())));ins_sym!(sym::target_arch, sess.target.arch.desc_symbol());
255 ret.insert((sym::target_endian,
Some(Symbol::intern(sess.target.endian.as_str()))));ins_str!(sym::target_endian, sess.target.endian.as_str());
256 ret.insert((sym::target_env, Some(sess.target.env.desc_symbol())));ins_sym!(sym::target_env, sess.target.env.desc_symbol());
257
258 for family in sess.target.families.as_ref() {
259 ret.insert((sym::target_family, Some(Symbol::intern(family))));ins_str!(sym::target_family, family);
260 if family == "windows" {
261 ret.insert((sym::windows, None));ins_none!(sym::windows);
262 } else if family == "unix" {
263 ret.insert((sym::unix, None));ins_none!(sym::unix);
264 }
265 }
266
267 let layout = sess.target.parse_data_layout().unwrap_or_else(|err| {
269 sess.dcx().emit_fatal(err);
270 });
271 let mut has_atomic = false;
272 for (i, align) in [
273 (8, layout.i8_align),
274 (16, layout.i16_align),
275 (32, layout.i32_align),
276 (64, layout.i64_align),
277 (128, layout.i128_align),
278 ] {
279 if i >= sess.target.min_atomic_width() && i <= sess.target.max_atomic_width() {
280 if !has_atomic {
281 has_atomic = true;
282 if sess.is_nightly_build() {
283 if sess.target.atomic_cas {
284 ret.insert((sym::target_has_atomic, None));ins_none!(sym::target_has_atomic);
285 }
286 ret.insert((sym::target_has_atomic_load_store, None));ins_none!(sym::target_has_atomic_load_store);
287 }
288 }
289 let mut insert_atomic = |sym, align: Align| {
290 if sess.target.atomic_cas {
291 ret.insert((sym::target_has_atomic, Some(sym)));ins_sym!(sym::target_has_atomic, sym);
292 }
293 if align.bits() == i {
294 ret.insert((sym::target_has_atomic_equal_alignment, Some(sym)));ins_sym!(sym::target_has_atomic_equal_alignment, sym);
295 }
296 ret.insert((sym::target_has_atomic_load_store, Some(sym)));ins_sym!(sym::target_has_atomic_load_store, sym);
297 };
298 insert_atomic(sym::integer(i), align);
299 if sess.target.pointer_width as u64 == i {
300 insert_atomic(sym::ptr, layout.pointer_align().abi);
301 }
302 }
303 }
304
305 ret.insert((sym::target_os, Some(sess.target.os.desc_symbol())));ins_sym!(sym::target_os, sess.target.os.desc_symbol());
306 ret.insert((sym::target_pointer_width,
Some(sym::integer(sess.target.pointer_width))));ins_sym!(sym::target_pointer_width, sym::integer(sess.target.pointer_width));
307
308 if sess.opts.unstable_opts.has_thread_local.unwrap_or(sess.target.has_thread_local) {
309 ret.insert((sym::target_thread_local, None));ins_none!(sym::target_thread_local);
310 }
311
312 ret.insert((sym::target_vendor, Some(sess.target.vendor_symbol())));ins_sym!(sym::target_vendor, sess.target.vendor_symbol());
313
314 if sess.is_test_crate() {
316 ret.insert((sym::test, None));ins_none!(sym::test);
317 }
318
319 if sess.ub_checks() {
320 ret.insert((sym::ub_checks, None));ins_none!(sym::ub_checks);
321 }
322
323 if sess.is_nightly_build() && sess.opts.unstable_opts.emscripten_wasm_eh {
325 ret.insert((sym::emscripten_wasm_eh, None));ins_none!(sym::emscripten_wasm_eh);
326 }
327
328 if sess.contract_checks() {
329 ret.insert((sym::contract_checks, None));ins_none!(sym::contract_checks);
330 }
331
332 ret
333}
334
335impl CheckCfg {
336 pub fn fill_well_known(&mut self, current_target: &Target) {
338 if !self.exhaustive_values && !self.exhaustive_names {
339 return;
340 }
341
342 let no_values = || {
344 let mut values = FxHashSet::default();
345 values.insert(None);
346 ExpectedValues::Some(values)
347 };
348
349 let empty_values = || {
351 let values = FxHashSet::default();
352 ExpectedValues::Some(values)
353 };
354
355 macro_rules! ins {
356 ($name:expr, $values:expr) => {{
357 self.well_known_names.insert($name);
358 self.expecteds.entry($name).or_insert_with($values)
359 }};
360 }
361
362 {
self.well_known_names.insert(sym::debug_assertions);
self.expecteds.entry(sym::debug_assertions).or_insert_with(no_values)
};ins!(sym::debug_assertions, no_values);
382
383 {
self.well_known_names.insert(sym::fmt_debug);
self.expecteds.entry(sym::fmt_debug).or_insert_with(empty_values)
}ins!(sym::fmt_debug, empty_values).extend(FmtDebug::all());
384
385 {
self.well_known_names.insert(sym::clippy);
self.expecteds.entry(sym::clippy).or_insert_with(no_values)
};ins!(sym::clippy, no_values);
390 {
self.well_known_names.insert(sym::doc);
self.expecteds.entry(sym::doc).or_insert_with(no_values)
};ins!(sym::doc, no_values);
391 {
self.well_known_names.insert(sym::doctest);
self.expecteds.entry(sym::doctest).or_insert_with(no_values)
};ins!(sym::doctest, no_values);
392 {
self.well_known_names.insert(sym::miri);
self.expecteds.entry(sym::miri).or_insert_with(no_values)
};ins!(sym::miri, no_values);
393 {
self.well_known_names.insert(sym::rustfmt);
self.expecteds.entry(sym::rustfmt).or_insert_with(no_values)
};ins!(sym::rustfmt, no_values);
394
395 {
self.well_known_names.insert(sym::overflow_checks);
self.expecteds.entry(sym::overflow_checks).or_insert_with(no_values)
};ins!(sym::overflow_checks, no_values);
396
397 {
self.well_known_names.insert(sym::panic);
self.expecteds.entry(sym::panic).or_insert_with(empty_values)
}ins!(sym::panic, empty_values)
398 .extend(PanicStrategy::ALL.iter().map(PanicStrategy::desc_symbol));
399
400 {
self.well_known_names.insert(sym::proc_macro);
self.expecteds.entry(sym::proc_macro).or_insert_with(no_values)
};ins!(sym::proc_macro, no_values);
401
402 {
self.well_known_names.insert(sym::relocation_model);
self.expecteds.entry(sym::relocation_model).or_insert_with(empty_values)
}ins!(sym::relocation_model, empty_values)
403 .extend(RelocModel::ALL.iter().map(RelocModel::desc_symbol));
404
405 let sanitize_values = SanitizerSet::all()
406 .into_iter()
407 .map(|sanitizer| Symbol::intern(sanitizer.as_str().unwrap()));
408 {
self.well_known_names.insert(sym::sanitize);
self.expecteds.entry(sym::sanitize).or_insert_with(empty_values)
}ins!(sym::sanitize, empty_values).extend(sanitize_values);
409
410 {
self.well_known_names.insert(sym::sanitizer_cfi_generalize_pointers);
self.expecteds.entry(sym::sanitizer_cfi_generalize_pointers).or_insert_with(no_values)
};ins!(sym::sanitizer_cfi_generalize_pointers, no_values);
411 {
self.well_known_names.insert(sym::sanitizer_cfi_normalize_integers);
self.expecteds.entry(sym::sanitizer_cfi_normalize_integers).or_insert_with(no_values)
};ins!(sym::sanitizer_cfi_normalize_integers, no_values);
412
413 {
self.well_known_names.insert(sym::target_feature);
self.expecteds.entry(sym::target_feature).or_insert_with(empty_values)
}ins!(sym::target_feature, empty_values).extend(
414 rustc_target::target_features::all_rust_features()
415 .filter(|(_, s)| s.in_cfg())
416 .map(|(f, _s)| f)
417 .chain(rustc_target::target_features::RUSTC_SPECIFIC_FEATURES.iter().cloned())
418 .map(Symbol::intern),
419 );
420
421 {
423 const VALUES: [&Symbol; 8] = [
424 &sym::target_abi,
425 &sym::target_arch,
426 &sym::target_endian,
427 &sym::target_env,
428 &sym::target_family,
429 &sym::target_os,
430 &sym::target_pointer_width,
431 &sym::target_vendor,
432 ];
433
434 for &e in VALUES {
436 if !self.exhaustive_values {
437 {
self.well_known_names.insert(e);
self.expecteds.entry(e).or_insert_with(|| ExpectedValues::Any)
};ins!(e, || ExpectedValues::Any);
438 } else {
439 {
self.well_known_names.insert(e);
self.expecteds.entry(e).or_insert_with(empty_values)
};ins!(e, empty_values);
440 }
441 }
442
443 if self.exhaustive_values {
444 let [
447 Some(values_target_abi),
448 Some(values_target_arch),
449 Some(values_target_endian),
450 Some(values_target_env),
451 Some(values_target_family),
452 Some(values_target_os),
453 Some(values_target_pointer_width),
454 Some(values_target_vendor),
455 ] = self.expecteds.get_disjoint_mut(VALUES)
456 else {
457 {
::core::panicking::panic_fmt(format_args!("unable to get all the check-cfg values buckets"));
};panic!("unable to get all the check-cfg values buckets");
458 };
459
460 for target in Target::builtins().chain(iter::once(current_target.clone())) {
461 values_target_abi.insert(target.options.cfg_abi.desc_symbol());
462 values_target_arch.insert(target.arch.desc_symbol());
463 values_target_endian.insert(Symbol::intern(target.options.endian.as_str()));
464 values_target_env.insert(target.options.env.desc_symbol());
465 values_target_family.extend(
466 target.options.families.iter().map(|family| Symbol::intern(family)),
467 );
468 values_target_os.insert(target.options.os.desc_symbol());
469 values_target_pointer_width.insert(sym::integer(target.pointer_width));
470 values_target_vendor.insert(target.vendor_symbol());
471 }
472 }
473 }
474
475 let atomic_values = &[
476 sym::ptr,
477 sym::integer(8usize),
478 sym::integer(16usize),
479 sym::integer(32usize),
480 sym::integer(64usize),
481 sym::integer(128usize),
482 ];
483 for sym in [
484 sym::target_has_atomic,
485 sym::target_has_atomic_equal_alignment,
486 sym::target_has_atomic_load_store,
487 ] {
488 {
self.well_known_names.insert(sym);
self.expecteds.entry(sym).or_insert_with(no_values)
}ins!(sym, no_values).extend(atomic_values);
489 }
490
491 {
self.well_known_names.insert(sym::target_thread_local);
self.expecteds.entry(sym::target_thread_local).or_insert_with(no_values)
};ins!(sym::target_thread_local, no_values);
492
493 {
self.well_known_names.insert(sym::ub_checks);
self.expecteds.entry(sym::ub_checks).or_insert_with(no_values)
};ins!(sym::ub_checks, no_values);
494 {
self.well_known_names.insert(sym::contract_checks);
self.expecteds.entry(sym::contract_checks).or_insert_with(no_values)
};ins!(sym::contract_checks, no_values);
495
496 {
self.well_known_names.insert(sym::unix);
self.expecteds.entry(sym::unix).or_insert_with(no_values)
};ins!(sym::unix, no_values);
497 {
self.well_known_names.insert(sym::windows);
self.expecteds.entry(sym::windows).or_insert_with(no_values)
};ins!(sym::windows, no_values);
498 }
499}