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
#![feature(arbitrary_enum_discriminant)]
#![crate_type = "proc-macro"]
#[macro_use]
extern crate quote;
extern crate proc_macro;
extern crate proc_macro2;
use proc_macro::TokenStream;
use syn::{Data, Meta};
mod enum_from_bitreader;
mod struct_from_bitreader;
#[proc_macro_derive(FromBitReader, attributes(size_in_bits))]
pub fn from_bit_reader(input: TokenStream) -> TokenStream {
let ast: syn::DeriveInput = syn::parse(input).unwrap();
let name = &ast.ident;
match ast.data {
Data::Enum(ref data_enum) =>
{
let size_in_bits : Option<syn::Lit> = get_attribute_value(&ast, "size_in_bits");
let size_in_bits_value = match size_in_bits
{
Some(sib) =>
{
sib
},
None => panic!("No size_in_bits attribute found!")
};
enum_from_bitreader::enum_from_bitreader(&data_enum.variants, name, size_in_bits_value)
},
Data::Struct(ref data_struct) =>
{
struct_from_bitreader::struct_from_bitreader(&data_struct.fields, name)
},
_ => panic!(
"deriving `FromByte` can be applied only to enums, {} is neither",
name
),
}
}
fn get_attribute_value(ast: &syn::DeriveInput, token : &str) -> Option<syn::Lit>
{
for attr in ast.attrs.iter()
{
match attr.parse_meta()
{
Ok(meta_attribute) =>
{
match meta_attribute
{
Meta::NameValue(meta_name_value) =>
{
let path_to_print = &meta_name_value.path;
match &*quote!{#path_to_print}.to_string() == token
{
true => return Some(meta_name_value.lit),
false => return None
}
},
Meta::Path(_path) => {},
Meta::List(_meta_list) => {}
}
},
_ => return None
}
}
return None
}