aboutsummaryrefslogtreecommitdiff
path: root/macros/src/codegen/locals.rs
blob: 9663563730fb589e4b30deb88148f4011588c7f9 (plain) (blame)
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
use proc_macro2::TokenStream as TokenStream2;
use quote::quote;
use rtfm_syntax::{
    ast::{App, Local},
    Context, Map,
};

use crate::codegen::util;

pub fn codegen(
    ctxt: Context,
    locals: &Map<Local>,
    app: &App,
) -> (
    // locals
    TokenStream2,
    // pat
    TokenStream2,
) {
    assert!(!locals.is_empty());

    let runs_once = ctxt.runs_once();
    let ident = util::locals_ident(ctxt, app);

    let mut lt = None;
    let mut fields = vec![];
    let mut items = vec![];
    let mut names = vec![];
    let mut values = vec![];
    let mut pats = vec![];
    let mut has_cfgs = false;

    for (name, local) in locals {
        let lt = if runs_once {
            quote!('static)
        } else {
            lt = Some(quote!('a));
            quote!('a)
        };

        let cfgs = &local.cfgs;
        has_cfgs |= !cfgs.is_empty();

        let expr = &local.expr;
        let ty = &local.ty;
        fields.push(quote!(
            #(#cfgs)*
            #name: &#lt mut #ty
        ));
        items.push(quote!(
            #(#cfgs)*
            static mut #name: #ty = #expr
        ));
        values.push(quote!(
            #(#cfgs)*
            #name: &mut #name
        ));
        names.push(name);
        pats.push(quote!(
            #(#cfgs)*
            #name
        ));
    }

    if lt.is_some() && has_cfgs {
        fields.push(quote!(__marker__: core::marker::PhantomData<&'a mut ()>));
        values.push(quote!(__marker__: core::marker::PhantomData));
    }

    let locals = quote!(
        #[allow(non_snake_case)]
        #[doc(hidden)]
        pub struct #ident<#lt> {
            #(#fields),*
        }

        impl<#lt> #ident<#lt> {
            #[inline(always)]
            unsafe fn new() -> Self {
                #(#items;)*

                #ident {
                    #(#values),*
                }
            }
        }
    );

    let ident = ctxt.ident(app);
    (
        locals,
        quote!(#ident::Locals { #(#pats,)* .. }: #ident::Locals),
    )
}