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
extern crate proc_macro;
use proc_macro::TokenStream;
use std::error::Error;
mod generate_impl;
mod type_parser;
use generate_impl::{escape, size_of};
use syn::{parse_macro_input, LitStr};
use type_parser::{one, GVariantType};
#[proc_macro]
pub fn gv_type(input: TokenStream) -> TokenStream {
let typestr = parse_macro_input!(input as LitStr).value();
type_for_typestr(typestr.as_ref()).unwrap().parse().unwrap()
}
#[proc_macro]
pub fn define_gv(input: TokenStream) -> TokenStream {
let typestr = parse_macro_input!(input as LitStr).value();
generate_impl::generate_types(&one(typestr.as_ref()).unwrap())
.unwrap()
.parse()
.unwrap()
}
fn type_for_typestr(gv_typestr: &[u8]) -> Result<String, Box<dyn Error>> {
let spec = type_parser::one(gv_typestr)?;
Ok(marker_type(&spec))
}
pub(crate) fn marker_type(t: &GVariantType) -> String {
match t {
GVariantType::B => "Bool".to_string(),
GVariantType::Y => "u8".to_string(),
GVariantType::N => "i16".to_string(),
GVariantType::Q => "u16".to_string(),
GVariantType::I => "i32".to_string(),
GVariantType::U => "u32".to_string(),
GVariantType::X => "i64".to_string(),
GVariantType::T => "u64".to_string(),
GVariantType::D => "f64".to_string(),
GVariantType::S => "Str".to_string(),
GVariantType::O => "Str".to_string(),
GVariantType::G => "Str".to_string(),
GVariantType::V => "Variant".to_string(),
GVariantType::A(t) => match size_of(t) {
None => format!("NonFixedWidthArray::<{}>", marker_type(t)),
Some(_) => format!("[{}]", marker_type(t)),
},
GVariantType::M(t) => match size_of(t) {
None => format!("MaybeNonFixedSize::<{}>", marker_type(t)),
Some(_) => format!("MaybeFixedSize::<{}>", marker_type(t)),
},
GVariantType::Tuple(_) | GVariantType::DictItem(_) => {
format!("Structure{name}", name = escape(t.to_string()))
}
}
}