1use proc_macro::TokenStream;
7use quote::{quote, ToTokens};
8
9mod attr_options;
10mod tokens;
11
12use attr_options::*;
13use syn::{Expr, FnArg};
14
15#[derive(Clone, Copy)]
16enum Macro<'a> {
17 Bench { fn_sig: &'a syn::Signature },
18 BenchGroup,
19}
20
21impl Macro<'_> {
22 fn name(&self) -> &'static str {
23 match self {
24 Self::Bench { .. } => "bench",
25 Self::BenchGroup => "bench_group",
26 }
27 }
28}
29
30mod systems {
32 use super::*;
33
34 pub fn elf() -> proc_macro2::TokenStream {
35 quote! {
36 target_os = "android",
37 target_os = "dragonfly",
38 target_os = "freebsd",
39 target_os = "fuchsia",
40 target_os = "haiku",
41 target_os = "illumos",
42 target_os = "linux",
43 target_os = "netbsd",
44 target_os = "openbsd",
45 target_os = "wasi",
46 target_os = "emscripten"
47 }
48 }
49
50 pub fn mach_o() -> proc_macro2::TokenStream {
51 quote! {
52 target_os = "ios",
53 target_os = "macos",
54 target_os = "tvos",
55 target_os = "watchos"
56 }
57 }
58}
59
60fn pre_main_attrs() -> proc_macro2::TokenStream {
63 let elf = systems::elf();
64 let mach_o = systems::mach_o();
65
66 quote! {
67 #[used]
68 #[cfg_attr(windows, link_section = ".CRT$XCU")]
69 #[cfg_attr(any(#elf), link_section = ".init_array")]
70 #[cfg_attr(any(#mach_o), link_section = "__DATA,__mod_init_func,mod_init_funcs")]
71 }
72}
73
74fn unsupported_error(attr_name: &str) -> proc_macro2::TokenStream {
75 let elf = systems::elf();
76 let mach_o = systems::mach_o();
77
78 let error = format!("Unsupported target OS for `#[divan::{attr_name}]`");
79
80 quote! {
81 #[cfg(not(any(windows, #elf, #mach_o)))]
82 ::std::compile_error!(#error);
83 }
84}
85
86#[proc_macro_attribute]
87pub fn bench(options: TokenStream, item: TokenStream) -> TokenStream {
88 let option_none = tokens::option_none();
89 let option_some = tokens::option_some();
90
91 let fn_item = item.clone();
92 let fn_item = syn::parse_macro_input!(fn_item as syn::ItemFn);
93 let fn_sig = &fn_item.sig;
94
95 let attr = Macro::Bench { fn_sig };
96 let attr_name = attr.name();
97
98 let options = match AttrOptions::parse(options, attr) {
99 Ok(options) => options,
100 Err(compile_error) => return compile_error,
101 };
102
103 let AttrOptions { private_mod, .. } = &options;
105
106 let fn_ident = &fn_sig.ident;
107 let fn_name = fn_ident.to_string();
108 let fn_name_pretty = fn_name.strip_prefix("r#").unwrap_or(&fn_name);
109
110 let ignore_attr_ident =
113 fn_item.attrs.iter().map(|attr| attr.meta.path()).find(|path| path.is_ident("ignore"));
114
115 let is_extern_abi = fn_sig.abi.is_some();
117
118 let fn_args = &fn_sig.inputs;
119
120 let type_param: Option<(usize, &syn::TypeParam)> = fn_sig
121 .generics
122 .params
123 .iter()
124 .enumerate()
125 .filter_map(|(i, param)| match param {
126 syn::GenericParam::Type(param) => Some((i, param)),
127 _ => None,
128 })
129 .next();
130
131 let const_param: Option<(usize, &syn::ConstParam)> = fn_sig
132 .generics
133 .params
134 .iter()
135 .enumerate()
136 .filter_map(|(i, param)| match param {
137 syn::GenericParam::Const(param) => Some((i, param)),
138 _ => None,
139 })
140 .next();
141
142 let is_type_before_const = match (type_param, const_param) {
143 (Some((t, _)), Some((c, _))) => t < c,
144 _ => false,
145 };
146
147 let static_ident = syn::Ident::new(
152 &format!("__DIVAN_BENCH_{}", fn_name_pretty.to_uppercase()),
153 fn_ident.span(),
154 );
155
156 let meta = entry_meta_expr(&fn_name, &options, ignore_attr_ident);
157
158 let bench_entry_runner = quote! { #private_mod::BenchEntryRunner };
159
160 let bench_args_global = if options.args_expr.is_some() {
162 quote! {
163 static __DIVAN_ARGS: #private_mod::BenchArgs = #private_mod::BenchArgs::new();
164 }
165 } else {
166 Default::default()
167 };
168
169 let last_arg_type = if options.args_expr.is_some() {
172 fn_args.last().map(|arg| match arg {
173 FnArg::Receiver(arg) => &*arg.ty,
174 FnArg::Typed(arg) => &*arg.ty,
175 })
176 } else {
177 None
178 };
179
180 let last_arg_type_tokens = last_arg_type
181 .map(|ty| match ty {
182 syn::Type::Reference(ty) if ty.lifetime.is_some() => {
186 let mut ty = ty.clone();
187 ty.lifetime = None;
188 ty.to_token_stream()
189 }
190
191 _ => ty.to_token_stream(),
192 })
193 .unwrap_or_default();
194
195 let arg_return_tokens = options
197 .args_expr
198 .as_ref()
199 .map(|args| match args {
200 Expr::Array(args) if args.elems.is_empty() => quote! {
202 -> [#last_arg_type_tokens; 0]
203 },
204
205 _ => Default::default(),
206 })
207 .unwrap_or_default();
208
209 let make_bench_fn = |generics: &[&dyn ToTokens]| {
212 let mut fn_expr = if generics.is_empty() {
213 fn_ident.to_token_stream()
215 } else {
216 quote! { #fn_ident::< #(#generics),* > }
218 };
219
220 match (fn_args.len(), &options.args_expr) {
222 (0, None) => {
224 if is_extern_abi {
226 fn_expr = quote! { || #fn_expr() };
227 }
228
229 quote! {
230 #bench_entry_runner::Plain(|divan | divan.bench(#fn_expr))
231 }
232 }
233
234 (0, Some(_)) => unreachable!(),
237
238 (1, None) => {
240 if is_extern_abi {
242 fn_expr = quote! { |divan | #fn_expr(divan) };
243 }
244
245 quote! { #bench_entry_runner::Plain(#fn_expr) }
246 }
247
248 (1, Some(args)) => quote! {
250 #bench_entry_runner::Args(|| __DIVAN_ARGS.runner(
251 || #arg_return_tokens { #args },
252
253 |arg| #private_mod::ToStringHelper(arg).to_string(),
254
255 |divan, __divan_arg| divan.bench(|| #fn_expr(
256 #private_mod::Arg::<#last_arg_type_tokens>::get(__divan_arg)
257 )),
258 ))
259 },
260
261 (2, Some(args)) => quote! {
263 #bench_entry_runner::Args(|| __DIVAN_ARGS.runner(
264 || #arg_return_tokens { #args },
265
266 |arg| #private_mod::ToStringHelper(arg).to_string(),
267
268 |divan, __divan_arg| #fn_expr(
269 divan,
270 #private_mod::Arg::<#last_arg_type_tokens>::get(__divan_arg),
271 ),
272 ))
273 },
274
275 (_, None) => quote! {
277 ::std::compile_error!(::std::concat!(
278 "expected 'args' option containing '",
279 ::std::stringify!(#last_arg_type_tokens),
280 "'",
281 ))
282 },
283
284 (_, Some(_)) => unreachable!(),
287 }
288 };
289
290 let pre_main_attrs = pre_main_attrs();
291 let unsupported_error = unsupported_error(attr_name);
292
293 let make_generic_group = |generic_benches: proc_macro2::TokenStream| {
295 let entry = quote! {
296 #private_mod::GroupEntry {
297 meta: #meta,
298 generic_benches: #option_some({ #generic_benches }),
299 }
300 };
301
302 quote! {
303 #unsupported_error
304
305 static #static_ident: #private_mod::GroupEntry = {
307 {
308 #pre_main_attrs
310 static PUSH: extern "C" fn() = push;
311
312 extern "C" fn push() {
313 static NODE: #private_mod::EntryList<#private_mod::GroupEntry>
314 = #private_mod::EntryList::new(&#static_ident);
315
316 #private_mod::GROUP_ENTRIES.push(&NODE);
317 }
318 }
319
320 #bench_args_global
324
325 #entry
326 };
327 }
328 };
329
330 let make_generic_bench_entry =
332 |ty: Option<&dyn ToTokens>, const_value: Option<&dyn ToTokens>| {
333 let generic_const_value = const_value.map(|const_value| quote!({ #const_value }));
334
335 let generics: Vec<&dyn ToTokens> = {
336 let mut generics = Vec::new();
337
338 generics.extend(generic_const_value.as_ref().map(|t| t as &dyn ToTokens));
339 generics.extend(ty);
340
341 if is_type_before_const {
342 generics.reverse();
343 }
344
345 generics
346 };
347
348 let bench_fn = make_bench_fn(&generics);
349
350 let type_value = match ty {
351 Some(ty) => quote! {
352 #option_some(#private_mod::EntryType::new::<#ty>())
353 },
354 None => option_none.clone(),
355 };
356
357 let const_value = match const_value {
358 Some(const_value) => quote! {
359 #option_some(#private_mod::EntryConst::new(&#const_value))
360 },
361 None => option_none.clone(),
362 };
363
364 quote! {
365 #private_mod::GenericBenchEntry {
366 group: &#static_ident,
367 bench: #bench_fn,
368 ty: #type_value,
369 const_value: #const_value,
370 }
371 }
372 };
373
374 let generated_items: proc_macro2::TokenStream = match &options.generic.consts {
375 _ if options.generic.is_empty() => Default::default(),
377
378 None => match &options.generic.types {
379 None => {
381 let bench_fn = make_bench_fn(&[]);
382
383 let entry = quote! {
384 #private_mod::BenchEntry {
385 meta: #meta,
386 bench: #bench_fn,
387 }
388 };
389
390 quote! {
391 static #static_ident: #private_mod::BenchEntry = {
394 {
395 #pre_main_attrs
397 static PUSH: extern "C" fn() = push;
398
399 extern "C" fn push() {
400 static NODE: #private_mod::EntryList<#private_mod::BenchEntry>
401 = #private_mod::EntryList::new(&#static_ident);
402
403 #private_mod::BENCH_ENTRIES.push(&NODE);
404 }
405 }
406
407 #bench_args_global
408
409 #entry
410 };
411 }
412 }
413
414 Some(GenericTypes::List(generic_types)) => {
416 let generic_benches =
417 generic_types.iter().map(|ty| make_generic_bench_entry(Some(&ty), None));
418
419 make_generic_group(quote! {
420 &[&[#(#generic_benches),*]]
421 })
422 }
423 },
424
425 Some(Expr::Array(generic_consts)) => {
427 let consts_count = generic_consts.elems.len();
428 let const_type = &const_param.unwrap().1.ty;
429
430 let generic_benches = options.generic.types_iter().map(|ty| {
431 let generic_benches = (0..consts_count).map(move |i| {
432 let const_value = quote! { __DIVAN_CONSTS[#i] };
433 make_generic_bench_entry(ty, Some(&const_value))
434 });
435
436 quote! {
439 static __DIVAN_GENERIC_BENCHES: [#private_mod::GenericBenchEntry; #consts_count] = [#(#generic_benches),*];
440 &__DIVAN_GENERIC_BENCHES
441 }
442 });
443
444 make_generic_group(quote! {
445 const __DIVAN_CONSTS: &[#const_type] = &#generic_consts;
451
452 &[#({ #generic_benches }),*]
453 })
454 }
455
456 Some(generic_consts) => {
462 const MAX_EXTERN_COUNT: usize = 20;
464
465 let const_type = &const_param.unwrap().1.ty;
466
467 let generic_benches = options.generic.types_iter().map(|ty| {
468 let generic_benches = (0..MAX_EXTERN_COUNT).map(move |i| {
469 let const_value = quote! {
470 __DIVAN_CONSTS[if #i < __DIVAN_CONST_COUNT { #i } else { 0 }]
472 };
473 make_generic_bench_entry(ty, Some(&const_value))
474 });
475
476 quote! {
479 static __DIVAN_GENERIC_BENCHES: [#private_mod::GenericBenchEntry; __DIVAN_CONST_COUNT]
480 = match #private_mod::shrink_array([#(#generic_benches),*]) {
481 Some(array) => array,
482 _ => panic!("external 'consts' cannot contain more than 20 values"),
483 };
484
485 &__DIVAN_GENERIC_BENCHES
486 }
487 });
488
489 make_generic_group(quote! {
490 const __DIVAN_CONST_COUNT: usize = __DIVAN_CONSTS.len();
491 const __DIVAN_CONSTS: &[#const_type] = &#generic_consts;
492
493 &[#({ #generic_benches }),*]
494 })
495 }
496 };
497
498 let mut result = item;
500 result.extend(TokenStream::from(generated_items));
501 result
502}
503
504#[proc_macro_attribute]
505pub fn bench_group(options: TokenStream, item: TokenStream) -> TokenStream {
506 let attr = Macro::BenchGroup;
507 let attr_name = attr.name();
508
509 let options = match AttrOptions::parse(options, attr) {
510 Ok(options) => options,
511 Err(compile_error) => return compile_error,
512 };
513
514 let AttrOptions { private_mod, .. } = &options;
516
517 let option_none = tokens::option_none();
518
519 let mod_item = item.clone();
521 let mod_item = syn::parse_macro_input!(mod_item as syn::ItemMod);
522
523 let mod_ident = &mod_item.ident;
524 let mod_name = mod_ident.to_string();
525 let mod_name_pretty = mod_name.strip_prefix("r#").unwrap_or(&mod_name);
526
527 let ignore_attr_ident =
532 mod_item.attrs.iter().map(|attr| attr.meta.path()).find(|path| path.is_ident("ignore"));
533
534 let static_ident = syn::Ident::new(
539 &format!("__DIVAN_GROUP_{}", mod_name_pretty.to_uppercase()),
540 mod_ident.span(),
541 );
542
543 let meta = entry_meta_expr(&mod_name, &options, ignore_attr_ident);
544
545 let pre_main_attrs = pre_main_attrs();
546 let unsupported_error = unsupported_error(attr_name);
547
548 let generated_items = quote! {
549 #unsupported_error
550
551 static #static_ident: #private_mod::EntryList<#private_mod::GroupEntry> = {
553 {
554 #pre_main_attrs
556 static PUSH: extern "C" fn() = push;
557
558 extern "C" fn push() {
559 #private_mod::GROUP_ENTRIES.push(&#static_ident);
560 }
561 }
562
563 #private_mod::EntryList::new({
564 static #static_ident: #private_mod::GroupEntry = #private_mod::GroupEntry {
565 meta: #meta,
566 generic_benches: #option_none,
567 };
568
569 &#static_ident
570 })
571 };
572 };
573
574 let mut result = item;
576 result.extend(TokenStream::from(generated_items));
577 result
578}
579
580fn entry_meta_expr(
582 raw_name: &str,
583 options: &AttrOptions,
584 ignore_attr_ident: Option<&syn::Path>,
585) -> proc_macro2::TokenStream {
586 let AttrOptions { private_mod, .. } = &options;
587
588 let raw_name_pretty = raw_name.strip_prefix("r#").unwrap_or(raw_name);
589
590 let display_name: &dyn ToTokens = match &options.name_expr {
591 Some(name) => name,
592 None => &raw_name_pretty,
593 };
594
595 let bench_options = options.bench_options_fn(ignore_attr_ident);
596
597 quote! {
598 #private_mod::EntryMeta {
599 raw_name: #raw_name,
600 display_name: #display_name,
601 bench_options: #bench_options,
602 module_path: ::std::module_path!(),
603
604 location: #private_mod::EntryLocation {
606 file: ::std::file!(),
607 line: ::std::line!(),
608 col: ::std::column!(),
609 },
610 }
611 }
612}