#![forbid(unsafe_code, future_incompatible, rust_2018_idioms)]
#![deny(missing_debug_implementations, nonstandard_style)]
#![recursion_limit = "512"]
extern crate proc_macro;
use proc_macro::TokenStream;
use quote::{quote, quote_spanned};
use syn::spanned::Spanned;
#[cfg(not(test))] #[proc_macro_attribute]
pub fn main(attr: TokenStream, item: TokenStream) -> TokenStream {
let rt = if attr.is_empty() {
if cfg!(feature = "native") {
syn::parse_str("runtime::native::Native").unwrap()
} else {
let tokens = quote_spanned! { proc_macro2::Span::call_site() =>
compile_error!("async runtime needs to be specified if no default runtime is set");
};
return TokenStream::from(tokens);
}
} else {
syn::parse_macro_input!(attr as syn::Expr)
};
let input = syn::parse_macro_input!(item as syn::ItemFn);
let ret = &input.sig.output;
let args = input.sig.inputs.iter();
let name = &input.sig.ident;
let body = &input.block;
let attrs = &input.attrs;
if name != "main" {
let tokens = quote_spanned! { name.span() =>
compile_error!("only the main function can be tagged with #[runtime::main]");
};
return TokenStream::from(tokens);
}
if input.sig.asyncness.is_none() {
let tokens = quote_spanned! { input.span() =>
compile_error!("the async keyword is missing from the function declaration");
};
return TokenStream::from(tokens);
}
let result = quote! {
fn main() #ret {
#(#attrs)*
async fn main(#(#args),*) #ret {
#body
}
runtime::raw::enter(#rt, async {
main().await
})
}
};
result.into()
}
#[proc_macro_attribute]
pub fn test(attr: TokenStream, item: TokenStream) -> TokenStream {
let rt = if attr.is_empty() {
if cfg!(feature = "native") {
syn::parse_str("runtime::native::Native").unwrap()
} else {
let tokens = quote_spanned! { proc_macro2::Span::call_site() =>
compile_error!("async runtime needs to be specified if no default runtime is set");
};
return TokenStream::from(tokens);
}
} else {
syn::parse_macro_input!(attr as syn::Expr)
};
let input = syn::parse_macro_input!(item as syn::ItemFn);
let ret = &input.sig.output;
let name = &input.sig.ident;
let body = &input.block;
let attrs = &input.attrs;
if input.sig.asyncness.is_none() {
let tokens = quote_spanned! { input.span() =>
compile_error!("the async keyword is missing from the function declaration");
};
return TokenStream::from(tokens);
}
let result = quote! {
#[test]
#(#attrs)*
fn #name() #ret {
runtime::raw::enter(#rt, async { #body })
}
};
result.into()
}
#[proc_macro_attribute]
pub fn bench(attr: TokenStream, item: TokenStream) -> TokenStream {
let rt = if attr.is_empty() {
if cfg!(feature = "native") {
syn::parse_str("runtime::native::Native").unwrap()
} else {
let tokens = quote_spanned! { proc_macro2::Span::call_site() =>
compile_error!("async runtime needs to be specified if no default runtime is set");
};
return TokenStream::from(tokens);
}
} else {
syn::parse_macro_input!(attr as syn::Expr)
};
let input = syn::parse_macro_input!(item as syn::ItemFn);
let args = &input.sig.inputs;
let name = &input.sig.ident;
let body = &input.block;
let attrs = &input.attrs;
if input.sig.asyncness.is_none() {
let tokens = quote_spanned! { input.span() =>
compile_error!("the async keyword is missing from the function declaration");
};
return TokenStream::from(tokens);
}
if !args.is_empty() {
let tokens = quote_spanned! { args.span() =>
compile_error!("async benchmarks don't take any arguments");
};
return TokenStream::from(tokens);
}
let result = quote! {
#[bench]
#(#attrs)*
fn #name(b: &mut test::Bencher) {
b.iter(|| {
let _ = runtime::raw::enter(#rt, async { #body });
});
}
};
result.into()
}