Rust 函数名称(调用者)或宏内的任何其他上下文

Tec*_*dar 0 macros rust

Rust 有没有办法获取“调用”函数名称或宏内的任何其他上下文信息?

例子:

#[macro_export]
macro_rules! somemacro {
    ( $x:expr ) => {
        {
          // access function name (etc..) that called this macro
        }
    };
}
Run Code Online (Sandbox Code Playgroud)

mca*_*ton 7

这可以使用过程宏来完成:

extern crate proc_macro;
use proc_macro::TokenStream;

#[proc_macro_attribute]
pub fn with_name(_: TokenStream, item: TokenStream) -> TokenStream {
    let mut input = syn::parse_macro_input!(item as syn::ItemFn);

    let fn_name = input.ident.to_string();
    let const_decl = quote::quote! {
        const THIS_FN: &str = #fn_name;
    };

    input.block.stmts.insert(0, syn::parse(const_decl.into()).unwrap());

    let output = quote::quote! {
        #input
    };

    output.into()
}
Run Code Online (Sandbox Code Playgroud)

货物.toml:

[package]
name = "with_name"
version = "0.1.0"
edition = "2018"

[lib]
proc-macro = true

[dependencies]
quote = "0.6.12"
syn = { version = "0.15.37", features = ["full"] }
Run Code Online (Sandbox Code Playgroud)

可以用作:

[package]
name = "with_name"
version = "0.1.0"
edition = "2018"

[lib]
proc-macro = true

[dependencies]
quote = "0.6.12"
syn = { version = "0.15.37", features = ["full"] }
Run Code Online (Sandbox Code Playgroud)

另请注意,如果您只关心模块,则有一个内置宏

#[with_name::with_name]
fn foo() {
    println!("Name: {}", THIS_FN);
}

fn main() {
    foo();
}
Run Code Online (Sandbox Code Playgroud)

(链接到游乐场)