| //! Contains code for selecting features |
| |
| #![deny(unused_extern_crates)] |
| #![deny(clippy::missing_docs_in_private_items)] |
| #![allow(deprecated)] |
| |
| use std::cmp::Ordering; |
| use std::io; |
| use std::str::FromStr; |
| |
| /// This macro defines the [`RustTarget`] and [`RustFeatures`] types. |
| macro_rules! define_rust_targets { |
| ( |
| Nightly => {$($nightly_feature:ident $(: #$issue:literal)?),* $(,)?} $(,)? |
| $( |
| $(#[$attrs:meta])* |
| $variant:ident($minor:literal) => {$($feature:ident $(: #$pull:literal)?),* $(,)?}, |
| )* |
| $(,)? |
| ) => { |
| /// Represents the version of the Rust language to target. |
| /// |
| /// To support a beta release, use the corresponding stable release. |
| /// |
| /// This enum will have more variants added as necessary. |
| #[allow(non_camel_case_types)] |
| #[derive(Clone, Copy, Debug, PartialEq, Eq, Hash)] |
| pub enum RustTarget { |
| /// Rust Nightly |
| $(#[doc = concat!( |
| "- [`", stringify!($nightly_feature), "`]", |
| "(", $("https://github.com/rust-lang/rust/pull/", stringify!($issue),)* ")", |
| )])* |
| Nightly, |
| $( |
| #[doc = concat!("Rust 1.", stringify!($minor))] |
| $(#[doc = concat!( |
| "- [`", stringify!($feature), "`]", |
| "(", $("https://github.com/rust-lang/rust/pull/", stringify!($pull),)* ")", |
| )])* |
| $(#[$attrs])* |
| $variant, |
| )* |
| } |
| |
| impl RustTarget { |
| fn minor(self) -> Option<u64> { |
| match self { |
| $( Self::$variant => Some($minor),)* |
| Self::Nightly => None |
| } |
| } |
| |
| const fn stable_releases() -> [(Self, u64); [$($minor,)*].len()] { |
| [$((Self::$variant, $minor),)*] |
| } |
| } |
| |
| #[cfg(feature = "__cli")] |
| /// Strings of allowed `RustTarget` values |
| pub const RUST_TARGET_STRINGS: &[&str] = &[$(concat!("1.", stringify!($minor)),)*]; |
| |
| #[derive(Debug, Copy, Clone, Eq, PartialEq, Hash)] |
| pub(crate) struct RustFeatures { |
| $($(pub(crate) $feature: bool,)*)* |
| $(pub(crate) $nightly_feature: bool,)* |
| } |
| |
| impl From<RustTarget> for RustFeatures { |
| fn from(target: RustTarget) -> Self { |
| if target == RustTarget::Nightly { |
| return Self { |
| $($($feature: true,)*)* |
| $($nightly_feature: true,)* |
| }; |
| } |
| |
| let mut features = Self { |
| $($($feature: false,)*)* |
| $($nightly_feature: false,)* |
| }; |
| |
| $(if target >= RustTarget::$variant { |
| $(features.$feature = true;)* |
| })* |
| |
| features |
| } |
| } |
| }; |
| } |
| |
| // NOTE: When adding or removing features here, make sure to add the stabilization PR |
| // number for the feature if it has been stabilized or the tracking issue number if the feature is |
| // not stable. |
| define_rust_targets! { |
| Nightly => { |
| vectorcall_abi, |
| }, |
| Stable_1_73(73) => { thiscall_abi: #42202 }, |
| Stable_1_71(71) => { c_unwind_abi: #106075 }, |
| Stable_1_68(68) => { abi_efiapi: #105795 }, |
| Stable_1_64(64) => { core_ffi_c: #94503 }, |
| Stable_1_59(59) => { const_cstr: #54745 }, |
| Stable_1_47(47) => { larger_arrays: #74060 }, |
| Stable_1_40(40) => { non_exhaustive: #44109 }, |
| Stable_1_36(36) => { maybe_uninit: #60445 }, |
| Stable_1_33(33) => { repr_packed_n: #57049 }, |
| #[deprecated] |
| Stable_1_30(30) => { |
| core_ffi_c_void: #53910, |
| min_const_fn: #54835, |
| }, |
| #[deprecated] |
| Stable_1_28(28) => { repr_transparent: #51562 }, |
| #[deprecated] |
| Stable_1_27(27) => { must_use_function: #48925 }, |
| #[deprecated] |
| Stable_1_26(26) => { i128_and_u128: #49101 }, |
| #[deprecated] |
| Stable_1_25(25) => { repr_align: #47006 }, |
| #[deprecated] |
| Stable_1_21(21) => { builtin_clone_impls: #43690 }, |
| #[deprecated] |
| Stable_1_20(20) => { associated_const: #42809 }, |
| #[deprecated] |
| Stable_1_19(19) => { untagged_union: #42068 }, |
| #[deprecated] |
| Stable_1_17(17) => { static_lifetime_elision: #39265 }, |
| #[deprecated] |
| Stable_1_0(0) => {}, |
| } |
| |
| /// Latest stable release of Rust |
| pub const LATEST_STABLE_RUST: RustTarget = { |
| // FIXME: replace all this code by |
| // ``` |
| // RustTarget::stable_releases() |
| // .into_iter() |
| // .max_by_key(|(_, m)| m) |
| // .map(|(t, _)| t) |
| // .unwrap_or(RustTarget::Nightly) |
| // ``` |
| // once those operations can be used in constants. |
| let targets = RustTarget::stable_releases(); |
| |
| let mut i = 0; |
| let mut latest_target = None; |
| let mut latest_minor = 0; |
| |
| while i < targets.len() { |
| let (target, minor) = targets[i]; |
| |
| if latest_minor < minor { |
| latest_minor = minor; |
| latest_target = Some(target); |
| } |
| |
| i += 1; |
| } |
| |
| match latest_target { |
| Some(target) => target, |
| None => unreachable!(), |
| } |
| }; |
| |
| impl Default for RustTarget { |
| fn default() -> Self { |
| LATEST_STABLE_RUST |
| } |
| } |
| |
| impl PartialOrd for RustTarget { |
| fn partial_cmp(&self, other: &Self) -> Option<Ordering> { |
| Some(self.cmp(other)) |
| } |
| } |
| |
| impl Ord for RustTarget { |
| fn cmp(&self, other: &Self) -> Ordering { |
| match (self.minor(), other.minor()) { |
| (Some(a), Some(b)) => a.cmp(&b), |
| (Some(_), None) => Ordering::Less, |
| (None, Some(_)) => Ordering::Greater, |
| (None, None) => Ordering::Equal, |
| } |
| } |
| } |
| |
| impl FromStr for RustTarget { |
| type Err = io::Error; |
| |
| fn from_str(s: &str) -> Result<Self, Self::Err> { |
| if s == "nightly" { |
| return Ok(Self::Nightly); |
| } |
| |
| if let Some(("1", str_minor)) = s.split_once('.') { |
| if let Ok(minor) = str_minor.parse::<u64>() { |
| for (target, target_minor) in Self::stable_releases() { |
| if minor == target_minor { |
| return Ok(target); |
| } |
| } |
| } |
| } |
| |
| Err(io::Error::new( |
| io::ErrorKind::InvalidInput, |
| "Got an invalid Rust target. Accepted values are of the form \"1.71\" or \"nightly\"." |
| )) |
| } |
| } |
| |
| impl std::fmt::Display for RustTarget { |
| fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { |
| match self.minor() { |
| Some(minor) => write!(f, "1.{}", minor), |
| None => "nightly".fmt(f), |
| } |
| } |
| } |
| |
| impl Default for RustFeatures { |
| fn default() -> Self { |
| RustTarget::default().into() |
| } |
| } |
| |
| #[cfg(test)] |
| mod test { |
| #![allow(unused_imports)] |
| use super::*; |
| |
| #[test] |
| fn target_features() { |
| let f_1_0 = RustFeatures::from(RustTarget::Stable_1_0); |
| assert!( |
| !f_1_0.static_lifetime_elision && |
| !f_1_0.core_ffi_c_void && |
| !f_1_0.untagged_union && |
| !f_1_0.associated_const && |
| !f_1_0.builtin_clone_impls && |
| !f_1_0.repr_align && |
| !f_1_0.thiscall_abi && |
| !f_1_0.vectorcall_abi |
| ); |
| let f_1_21 = RustFeatures::from(RustTarget::Stable_1_21); |
| assert!( |
| f_1_21.static_lifetime_elision && |
| !f_1_21.core_ffi_c_void && |
| f_1_21.untagged_union && |
| f_1_21.associated_const && |
| f_1_21.builtin_clone_impls && |
| !f_1_21.repr_align && |
| !f_1_21.thiscall_abi && |
| !f_1_21.vectorcall_abi |
| ); |
| let features = RustFeatures::from(RustTarget::Stable_1_71); |
| assert!( |
| features.c_unwind_abi && |
| features.abi_efiapi && |
| !features.thiscall_abi |
| ); |
| let f_nightly = RustFeatures::from(RustTarget::Nightly); |
| assert!( |
| f_nightly.static_lifetime_elision && |
| f_nightly.core_ffi_c_void && |
| f_nightly.untagged_union && |
| f_nightly.associated_const && |
| f_nightly.builtin_clone_impls && |
| f_nightly.maybe_uninit && |
| f_nightly.repr_align && |
| f_nightly.thiscall_abi && |
| f_nightly.vectorcall_abi |
| ); |
| } |
| |
| fn test_target(target_str: &str, target: RustTarget) { |
| let target_string = target.to_string(); |
| assert_eq!(target_str, target_string); |
| assert_eq!(target, RustTarget::from_str(target_str).unwrap()); |
| } |
| |
| #[test] |
| fn str_to_target() { |
| test_target("1.0", RustTarget::Stable_1_0); |
| test_target("1.17", RustTarget::Stable_1_17); |
| test_target("1.19", RustTarget::Stable_1_19); |
| test_target("1.21", RustTarget::Stable_1_21); |
| test_target("1.25", RustTarget::Stable_1_25); |
| test_target("1.71", RustTarget::Stable_1_71); |
| test_target("nightly", RustTarget::Nightly); |
| } |
| } |