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
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
extern crate proc_macro;
use pmutil::synom_ext::FromSpan;
#[cfg(procmacro2_semver_exempt)]
use pmutil::SpanExt;
use proc_macro2::Span;
use syn::*;
pub mod binder;
pub mod derive;
pub mod prelude;
mod syn_ext;
pub fn call_site<T: FromSpan>() -> T {
T::from_span(Span::call_site())
}
#[cfg(not(procmacro2_semver_exempt))]
pub fn def_site<T: FromSpan>() -> T {
call_site()
}
#[cfg(procmacro2_semver_exempt)]
pub fn def_site<T: FromSpan>() -> T {
Span::def_site().located_at(Span::call_site()).as_token()
}
pub fn print<T: Into<proc_macro2::TokenStream>>(
attr: &'static str,
t: T,
) -> proc_macro::TokenStream {
use std::env;
let tokens = t.into();
match env::var("PRINT_GENERATED") {
Ok(ref s) if s == "1" || attr == s => {}
_ => return tokens.into(),
}
println!("\n\tOutput of #[{}]:\n\t {}", attr, tokens);
tokens.into()
}
pub fn is_attr_name(attr: &Attribute, name: &str) -> bool {
match *attr {
Attribute {
path:
Path {
leading_colon: None,
ref segments,
},
..
} if segments.len() == 1 => segments.first().unwrap().ident == name,
_ => false,
}
}
pub fn doc_str(attr: &Attribute) -> Option<String> {
fn parse_tts(attr: &Attribute) -> String {
let meta = attr.parse_meta().ok();
match meta {
Some(Meta::NameValue(MetaNameValue {
lit: Lit::Str(s), ..
})) => s.value(),
_ => panic!("failed to parse {}", attr.tokens),
}
}
match *attr {
Attribute { .. } => {
if !is_attr_name(attr, "doc") {
return None;
}
Some(parse_tts(attr))
}
}
}
#[macro_export]
macro_rules! fail {
($($args:tt)+) => {{
panic!("{}\n --> {}:{}:{}", format_args!($($args)*), file!(), line!(), column!());
}};
}
#[macro_export]
macro_rules! unimplemented {
($($args:tt)+) => {{
fail!("not yet implemented: {}", format_args!($($args)*));
}};
}
#[macro_export]
macro_rules! unreachable {
() => {{
fail!("internal error: unreachable");
}};
($($args:tt)+) => {{
fail!("internal error: unreachable\n{}", format_args!($($args)*));
}};
}