1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
use std::ops::Range;
use proc_macro::TokenStream;
use quote::{format_ident, quote};
use syn::{Index, Result};
pub fn expand(range: Range<usize>) -> Result<TokenStream> {
let mut ts = TokenStream::new();
for n in range {
let generic_params_argument = (1..=n).map(|n| format_ident!("A{}", n));
let matches_argument = generic_params_argument
.clone()
.enumerate()
.map(|(n, gp)| quote!(<#gp>::matches(params.get_unchecked(#n).ty())));
let invokables = (0..n).map(Index::from).map(|n| quote!(self.#n.invokable()));
let generic_params_parameter = (1..=n).map(|n| format_ident!("P{}", n));
let matches_parameter = generic_params_parameter
.clone()
.enumerate()
.map(|(n, gp)| quote!(<#gp>::matches(params.get_unchecked(#n).ty())));
let generic_params_argument_tuple = generic_params_argument.clone();
let generic_params_argument_where = generic_params_argument.clone();
let generic_params_argument_type = generic_params_argument.clone();
let generic_params_parameter_tuple = generic_params_parameter.clone();
let generic_params_parameter_where = generic_params_parameter.clone();
let impl_ts = quote! {
unsafe impl<#(#generic_params_argument),*> Arguments<#n> for (#(#generic_params_argument_tuple,)*)
where
#(#generic_params_argument_where: Argument),*
{
type Type = (#(#generic_params_argument_type::Type,)*);
fn matches(method: &MethodInfo) -> bool {
let params = method.parameters();
params.len() == #n && unsafe { #(#matches_argument) && * }
}
fn invokable(&mut self) -> [*mut c_void; #n] {
[#(#invokables),*]
}
}
unsafe impl<#(#generic_params_parameter),*> Parameters for (#(#generic_params_parameter_tuple,)*)
where
#(#generic_params_parameter_where: Parameter),*
{
const COUNT: usize = #n;
fn matches(method: &MethodInfo) -> bool {
let params = method.parameters();
params.len() == #n && unsafe { #(#matches_parameter) && * }
}
}
};
ts.extend(TokenStream::from(impl_ts));
}
Ok(ts)
}