nom_tracable_macros/
lib.rs

1#![recursion_limit = "128"]
2
3extern crate proc_macro;
4
5use crate::proc_macro::TokenStream;
6use quote::ToTokens;
7use syn::{self, parse_macro_input, parse_quote, AttributeArgs, FnArg, ItemFn, Stmt};
8
9#[proc_macro_attribute]
10pub fn tracable_parser(attr: TokenStream, item: TokenStream) -> TokenStream {
11    let attr = parse_macro_input!(attr as AttributeArgs);
12    let item = parse_macro_input!(item as ItemFn);
13    impl_tracable_parser(&attr, &item)
14}
15
16fn impl_tracable_parser(_attr: &AttributeArgs, item: &ItemFn) -> TokenStream {
17    let default = impl_tracable_parser_default(&item);
18    let trace = impl_tracable_parser_trace(&item);
19
20    let mut item = item.clone();
21
22    item.block.stmts.clear();
23    item.block.stmts.push(default);
24    item.block.stmts.push(trace);
25
26    item.into_token_stream().into()
27}
28
29fn impl_tracable_parser_default(item: &ItemFn) -> Stmt {
30    let body = item.block.as_ref();
31    parse_quote! {
32        #[cfg(not(feature = "trace"))]
33        {
34            #body
35        }
36    }
37}
38
39fn impl_tracable_parser_trace(item: &ItemFn) -> Stmt {
40    let ident = &item.sig.ident;
41
42    let input = if let Some(x) = &item.sig.inputs.first() {
43        match x {
44            FnArg::Typed(arg) => &arg.pat,
45            _ => panic!("function with #[tracable_parser] must have an argument"),
46        }
47    } else {
48        panic!("function with #[tracable_parser] must have an argument");
49    };
50
51    let body = item.block.as_ref();
52
53    parse_quote! {
54        #[cfg(feature = "trace")]
55        {
56            let (depth, #input) = nom_tracable::forward_trace(#input, stringify!(#ident));
57
58            let body_ret = {
59                let body = || { #body };
60                body()
61            };
62
63            nom_tracable::backward_trace(body_ret, stringify!(#ident), depth)
64        }
65    }
66}