1
Fork 0

Life's too short not to use cfg_if

This commit is contained in:
gnzlbg 2019-03-26 16:14:32 +01:00
parent 3b6b4899bf
commit 0c127e8494
2 changed files with 101 additions and 23 deletions

View file

@ -97,36 +97,33 @@ pub fn spin_loop() {
/// elimination. /// elimination.
/// ///
/// This function is a no-op, and does not even read from `dummy`. /// This function is a no-op, and does not even read from `dummy`.
#[inline]
#[unstable(feature = "test", issue = "27812")] #[unstable(feature = "test", issue = "27812")]
pub fn black_box<T>(dummy: T) -> T { pub fn black_box<T>(dummy: T) -> T {
#[cfg(not( cfg_if! {
any( if #[cfg(any(
target_arch = "asmjs", target_arch = "asmjs",
all( all(
target_arch = "wasm32", target_arch = "wasm32",
target_os = "emscripten" target_os = "emscripten"
) )
) ))] {
))] { #[inline]
// we need to "use" the argument in some way LLVM can't unsafe fn black_box_impl<T>(d: T) -> T {
// introspect. // these targets do not support inline assembly
unsafe { asm!("" : : "r"(&dummy)) } let ret = crate::ptr::read_volatile(&d);
dummy crate::mem::forget(d);
} ret
#[cfg( }
any( } else {
target_arch = "asmjs", #[inline]
all( unsafe fn black_box_impl<T>(d: T) -> T {
target_arch = "wasm32", // we need to "use" the argument in some way LLVM can't
target_os = "emscripten" // introspect.
) asm!("" : : "r"(&d));
) d
)] { }
// asm.js and emscripten do not support inline assembly
unsafe {
let ret = crate::ptr::read_volatile(&dummy);
crate::mem::forget(dummy);
ret
} }
} }
unsafe { black_box_impl(dummy) }
} }

View file

@ -119,3 +119,84 @@ macro_rules! impl_fn_for_zst {
)+ )+
} }
} }
/// A macro for defining `#[cfg]` if-else statements.
///
/// The macro provided by this crate, `cfg_if`, is similar to the `if/elif` C
/// preprocessor macro by allowing definition of a cascade of `#[cfg]` cases,
/// emitting the implementation which matches first.
///
/// This allows you to conveniently provide a long list `#[cfg]`'d blocks of code
/// without having to rewrite each clause multiple times.
///
/// # Example
///
/// ```
/// #[macro_use]
/// extern crate cfg_if;
///
/// cfg_if! {
/// if #[cfg(unix)] {
/// fn foo() { /* unix specific functionality */ }
/// } else if #[cfg(target_pointer_width = "32")] {
/// fn foo() { /* non-unix, 32-bit functionality */ }
/// } else {
/// fn foo() { /* fallback implementation */ }
/// }
/// }
///
/// # fn main() {}
/// ```
macro_rules! cfg_if {
// match if/else chains with a final `else`
($(
if #[cfg($($meta:meta),*)] { $($it:item)* }
) else * else {
$($it2:item)*
}) => {
cfg_if! {
@__items
() ;
$( ( ($($meta),*) ($($it)*) ), )*
( () ($($it2)*) ),
}
};
// match if/else chains lacking a final `else`
(
if #[cfg($($i_met:meta),*)] { $($i_it:item)* }
$(
else if #[cfg($($e_met:meta),*)] { $($e_it:item)* }
)*
) => {
cfg_if! {
@__items
() ;
( ($($i_met),*) ($($i_it)*) ),
$( ( ($($e_met),*) ($($e_it)*) ), )*
( () () ),
}
};
// Internal and recursive macro to emit all the items
//
// Collects all the negated cfgs in a list at the beginning and after the
// semicolon is all the remaining items
(@__items ($($not:meta,)*) ; ) => {};
(@__items ($($not:meta,)*) ; ( ($($m:meta),*) ($($it:item)*) ), $($rest:tt)*) => {
// Emit all items within one block, applying an approprate #[cfg]. The
// #[cfg] will require all `$m` matchers specified and must also negate
// all previous matchers.
cfg_if! { @__apply cfg(all($($m,)* not(any($($not),*)))), $($it)* }
// Recurse to emit all other items in `$rest`, and when we do so add all
// our `$m` matchers to the list of `$not` matchers as future emissions
// will have to negate everything we just matched as well.
cfg_if! { @__items ($($not,)* $($m,)*) ; $($rest)* }
};
// Internal macro to Apply a cfg attribute to a list of items
(@__apply $m:meta, $($it:item)*) => {
$(#[$m] $it)*
};
}