47 lines
1.2 KiB
Rust

use proc_macro2::TokenStream;
use quote::{quote, ToTokens};
use syn::parse::{Parse, ParseStream};
use syn::{parse_quote, Result};
use crate::util::{kw, parse_key_value, parse_string};
/// Expand the `#[time(..)]` macro.
pub fn time(stream: TokenStream, item: syn::ItemFn) -> Result<TokenStream> {
let meta: Meta = syn::parse2(stream)?;
Ok(create(meta, item))
}
/// The `..` in `#[time(..)]`.
pub struct Meta {
pub span: Option<syn::Expr>,
pub name: Option<String>,
}
impl Parse for Meta {
fn parse(input: ParseStream) -> Result<Self> {
Ok(Self {
name: parse_string::<kw::name>(input)?,
span: parse_key_value::<kw::span, syn::Expr>(input)?,
})
}
}
fn create(meta: Meta, mut item: syn::ItemFn) -> TokenStream {
let name = meta.name.unwrap_or_else(|| item.sig.ident.to_string());
let span = meta
.span
.as_ref()
.map(|span| quote! { Some(#span) })
.unwrap_or_else(|| quote! { None });
item.block.stmts.insert(
0,
parse_quote! {
#[cfg(not(target_arch = "wasm32"))]
let __scope = ::typst_timing::TimingScope::new(#name, #span);
},
);
item.into_token_stream()
}