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
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
#![feature(proc_macro_diagnostic)]
use proc_macro::TokenStream;
use syn::{ItemFn, parse::Nothing, Path};
use std::sync::RwLock;
use quote::ToTokens;
mod item_fn;
lazy_static::lazy_static! {
static ref EXPORTNUM: RwLock<usize> = RwLock::new(0);
}
#[proc_macro_attribute]
pub fn patchable(attr: TokenStream, input: TokenStream) -> TokenStream {
let modpath = get_modpath(attr);
if modpath.is_err() {
return TokenStream::new();
}
if let Ok(fn_item) = syn::parse::<ItemFn>(input) {
item_fn::patchable(fn_item, modpath.unwrap())
} else {
panic!("I can't hotpatch this yet!");
}
}
#[proc_macro_attribute]
pub fn patch(attr: TokenStream, input: TokenStream) -> TokenStream {
let modpath = get_modpath(attr);
if modpath.is_err() {
return TokenStream::new();
}
if let Ok(fn_item) = syn::parse::<ItemFn>(input) {
item_fn::patch(fn_item, modpath.unwrap())
} else {
panic!("I can't patch this yet!");
}
}
fn get_modpath(attr: TokenStream) -> Result<Option<String>, ()> {
if syn::parse::<Nothing>(attr.clone()).is_ok() {
Ok(None)
} else {
let path = syn::parse::<Path>(attr.clone());
if path.is_err() {
proc_macro::Span::call_site().error("Expected module path")
.help("Just use #[patchable]; it's already module aware.")
.help("If you're trying to spoof a module path, the supplied arguement is an invalid path")
.emit();
return Err(());
}
let mut ts = syn::export::TokenStream2::new();
path.unwrap().to_tokens(&mut ts);
Ok(Some(ts.to_string().replace(" ", "")))
}
}