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
//! A crate holding derive macros for the repr-trait crate.
//!
//! In general, prefer to use repr-trait instead of this.

use proc_macro::TokenStream;
use quote::quote;
use syn::{parse_macro_input, AttrStyle, Attribute, DeriveInput, Path};

macro_rules! repr_derive {
    ($tr:ident : $fn:ident($inner:expr) ) => {
        #[proc_macro_derive($tr)]
        pub fn $fn(input: TokenStream) -> TokenStream {
            let input = parse_macro_input!(input as DeriveInput);

            let ident = input.ident;

            if has_repr(&input.attrs, $inner) {
                quote!(
                    unsafe impl $tr for #ident {}
                ).into()
            } else {
                panic!("Can't derive {} on a struct without #[repr({})]", stringify!($tr), $inner);
            }
        }
    }
}

repr_derive!(Packed: repr_packed("packed"));
repr_derive!(Transparent: repr_transparent("transparent"));
repr_derive!(C: repr_c("C"));

fn has_repr(attrs: &[Attribute], repr: &str) -> bool {
    for attr in attrs {
        // If the style isn't outer, reject it
        if !matches!(attr.style, AttrStyle::Outer) {
            continue;
        }

        // If the path doesn't match, reject it
        if let Path {
            leading_colon: None,
            ref segments,
        } = attr.path
        {
            // If there's more than one, reject it
            if segments.len() != 1 {
                continue;
            }

            let seg = segments.first().unwrap();

            // If there are arguments, reject it
            if !seg.arguments.is_empty() {
                continue;
            }

            // If the ident isn't "repr", reject it
            if seg.ident.to_string() != "repr" {
                continue;
            }
        } else {
            // If we don't match, reject if
            continue;
        }

        // If it doesn't match, reject it
        if format!("{}", attr.tokens) != format!("({})", repr) {
            continue;
        }

        return true;
    }

    false
}