关于rust宏-过程宏(补充)

一、 rust编译过程

rust编译过程

从上面的编译过程图,可以看到声明宏过程宏被编译到AST中过程是不同的:

  • 声明宏:通过macro_rule 定义的宏最终只是被解析为TokenStream;
  • 过程宏: 定义的过程宏首先被解析为proc_macro::TokenStream,接着再被解析为proc_macro2::TokenStream,再使用Syn库解析为AST(这里的不同于编译生成的AST),最后再通过Quote解析为TokenStream,最终会被扩展到编译过程TokenStream中,进而再被编译AST;

二、过程宏

1、样例


属性过程宏样例

此处定义一个属性过程宏的样例
2、定义过程宏

  • 准备工作
    主要是Cargo.toml文件中
[lib]
proc-macro=true # 开启过程宏

# 引入如下三个crate的原因 参考上面编译过程图:在生成最终的TokenStream前,需要依赖这三个crate
[dependencies]
proc-macro2 = "1.0.34"
syn = {version="1.0.82", feature="full"}
quote ="1.0.10"
  • 项目结构
    过程宏需要定义在一个单独的crate中,主要是因为过程宏是一段在编译crate前,对其代码进行加工的代码,而这段是需要在编译后执行的。若是将定义过程宏和使用过程宏放到同一个crate中,就会陷入编译“死锁”:
    1、要编译的代码需要运行过程宏进行展开,否则代码是不完整的,无法编译crate;
    2、不能编译crate,那么在crate中的过程宏就无法执行,就不能展开被过程宏“修饰”的代码

  • 定义过程宏

use proc_macro::TokenStream;

/// 定义一个属性过程宏
/// ???输入参数:TokenStream 输出参数: TokenStream【解释见`项目结构`】
/// 不做任何加工,直接输出item
#[proc_macro_attribute]
pub fn my_first_attr_proc_macro(attr: TokenStream, item: TokenStream) -> TokenStream {
    eprintln!("===输出attr");
    eprintln!(" {:#?}", attr);
    eprintln!("===输出item");
    eprintln!("{:#?}", item);
    item
}
  • 使用过程宏
    1、Cargo.toml文件引入过程宏crate
[dependencies]
produceral_macro={path= "../procedural_macro_demos/procedural_macro" } # 本地路径引用

2、使用过程宏
通过#[crate_name::proc_macro_func("过程宏名字-任意")]
类似 #[produceral_macro::my_first_attr_proc_macro("my_first_procedural_macro")]

#[produceral_macro::my_first_attr_proc_macro("my_first_procedural_macro")]
fn add(a: u64, b: u64) -> () {
    eprintln!("a={:?}, b={:?}, a+b={:?}", a, b , (a+b));
}

rust过程宏本质就是一个编译环节的“过滤器”或者说是一个“中间件”,接收一段用户编写的源代码,再做一通“转换”操作,然后返回给编译器一段经过修改的代码。

目前过程宏代码的调试,最好通过print来进行

最终生成的TokenStream是没有任何语义信息的,是通过树形结构的数据组织,表达了用户源代码各个语言元素的类型及其相互之间的关系;每个语言元素都有一个span属性,记录该元素在用户源代码中的位置。同时不同类型的节点有各自独有的属性。
而Rust过程宏就是自己能够手动修改输入变量中的值,比如样例中的attr,item,换句话说等价于加工原始输入代码,最后将加工后的代码返回给编译器即可.

三、proc-macro2\syn\quote三个包在过程宏中的应用

使用syn、quote等crate模拟过程宏“修改”源代码,并将结果以TokenStream给到编译器:

#[proc_macro_attribute]
pub fn my_first_attr_proc_macro_parse(attr: TokenStream, input: TokenStream) -> TokenStream {
    eprintln!("===attr: {:#?}", attr);
    eprintln!("===item: {:#?}", input);
    // 通过syn来解析输入的TokenStream
    let input_fn = syn::parse_macro_input!(input as syn::ItemFn);
    // 获取该token对应的类型:ident
    let name = input_fn.sig.ident.clone();
    //  输出TokenStream给到编译器
    TokenStream::from(quote! { // 使用quote库来构建编译器需要的TokenStream
        fn #name () {
            #input_fn

            for i in 0..3 {
                println!("loop time {}", i);

                let r = std::panic::catch_unwind(|| {
                    #name();
                });

                if r.is_ok() {
                    return;
                }
                if i == 2 {
                    std::panic::resume_unwind(r.unwrap_err());
                }
            }
        }
    })
}

测试代码

#[procedural_macro::my_first_attr_proc_macro_parse]
fn test_proc_macro() {
    assert_eq!(1,1);  // 运行正常
    assert_eq!(1,22);  // 触发panic
}

四、编译过程宏代码

#输出未进行宏扩展的ast树
$ cargo rustc -- -Z ast-json-noexpand=yes
#输出宏扩展后的ast树
$ cargo rustc -- -Z ast-json=yes 
#输出hir格式的中间描述
$ cargo rustc -- -Z unpretty=hir 
#输出hir格式并带有类型信息的中间描述
$ cargo rustc -- -Z unpretty=hir,typed 
#输出hir格式并带有完整树结构的中间描述
$ cargo rustc -- -Z unpretty=hir-tree
#输出mir格式的中间描述
$ cargo rustc -- -Z unpretty=mir

#### 代码编译中间代码 ####
#输出llvm ir格式的中间描述
$ rustc --emit llvm-ir lrfrc.rs
#输出汇编格式的中间描述
$ rustc --emit asm lrfrc.rs
#分析中间汇编输出
$ rustc -C opt-level=3 --emit=obj lrfrc.rs
$ size -A lrfrc.o
$ objdump -d lrfrc.o
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,793评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,567评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,342评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,825评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,814评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,680评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,033评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,687评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,175评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,668评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,775评论 1 332
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,419评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,020评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,978评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,206评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,092评论 2 351
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,510评论 2 343

推荐阅读更多精彩内容