core_extensions

Macro gen_ident_range

Source
gen_ident_range!() { /* proc-macro */ }
Expand description

Generates identifiers. passing them to a callback macro.

§Repetition Syntax

The syntax for describing the generated identifiers:

for <ident> * in <range>

Where <ident> is any valid identifier.

Where <range> can be either <number> .. <number> or <number> ..= <number>.

Where <number> can be any of:

  • An integer literal

  • count(....): Which counts the amount of token trees in (....). Macro parameters (eg: $foo) are one token tree, and matched pairs of []/()/{} count as one token tree regardless of the tokens inside.

§Version compatibility

This macro requires Rust 1.45.0 to be invoked inside of a function.

§Example

§Basic

Basic examples of using this macro, and what it passes to a callback macro.

For a more realistic example you can look at the one below

use core_extensions::gen_ident_range;
 
fn main() {
    assert_eq!(hello(), "world");
    assert_eq!(foo(), "bar");
}
 
// Calls the `expected_0_to_2` macro.
gen_ident_range!{
    crate::expected_0_to_2!{hello "world"}
    for stuff_* in 0..3
}
 
// Calls the `expected_1_to_4` macro.
gen_ident_range!{
    crate::expected_1_to_4!{foo "bar" baz}
    // `count(....)` here counts 4 token trees
    for pre_* in 1..=count(a (b c) [d e f] {g h i j})
}

#[macro_export]
macro_rules! expected_0_to_2{
    ($func:ident $lit:literal  (stuff_0 stuff_1 stuff_2)) => {
        fn $func() -> &'static str {
            $lit
        }
    }
}
 
#[macro_export]
macro_rules! expected_1_to_4{
    ($func:ident $lit:literal baz  (pre_1 pre_2 pre_3 pre_4)) => {
        fn $func() -> &'static str {
            $lit
        }
    }
}

§More Realistic Example

use core_extensions::gen_ident_range;
 
fn main() {
    assert_eq!(add_unsigned(3, 5, 8), 16);
    assert_eq!(add_signed(-3, 8), 5);
 
}
 
adder_fn!{ pub fn add_unsigned(u16, u32, u64) -> u64 }
adder_fn!{ fn add_signed(i8, i16) -> i64 }
 
 
 
#[macro_export]
macro_rules! adder_fn {
    ($vis:vis fn $func:ident ($($arg_ty:ty),* $(,)?) -> $ret_ty:ty) => {
        gen_ident_range!{
            $crate::__priv_adder_fn!{
                ($vis fn $func ($($arg_ty,)*) -> $ret_ty)
            }
            for arg_* in 0..count($($arg_ty)*)
        }
    }
}
 
#[macro_export]
macro_rules! __priv_adder_fn {
    (
        ($vis:vis fn $func:ident ($($arg_ty:ty,)*) -> $ret_ty:ty)
        ($($arg:ident)*)
    ) => {
        $vis fn $func($($arg: $arg_ty,)*) -> $ret_ty {
            // assuming that Default::default is zero or empty
            <$ret_ty as $crate::__::Default>::default()
            $(
                + <$ret_ty as $crate::__::From<_>>::from($arg)
            )*
        }
    }
}
 
 
#[doc(hidden)]
pub mod __ {
    pub use core_extensions::gen_ident_range;
     
    pub use std::convert::From;
    pub use std::default::Default;
}