aboutsummaryrefslogtreecommitdiff
path: root/zkvms/nexus/wrapper_macro/src/lib.rs
blob: 324ace835d1bc8ebbea37b864141d0a227c3b4de (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
use proc_macro::TokenStream;

#[path = "../../../../guests_macro/src/parse_fn.rs"]
mod parse_fn;
use crate::parse_fn::{args_divide_grouped, args_divide_public, args_split, split_fn};

/// Creates a body, which reads all inputs, stores them in variables, then
/// writes the ones, defined as public in `default_public_input.toml` to the
/// journal and finally executes the guest entrypoint function with those
/// arguments, committing its output.
///
/// # Usage
///
/// Inside Nexus' guest (excluding the `entrypoint_expr` call):
///
/// ```rust
/// make_wrapper!{fn main(...) -> ...}
/// ```
///
/// # Example output
///
/// ```rust
/// {
///     let ... = read_private_input::<...>().unwrap();
///     let ... = read_private_input::<...>().unwrap();
///     ...
///     write_output::<...>(&...);
///     write_output::<...>(&...);
///     ...
///     write_output::<...>(&zkp::main(..., ..., ...));
/// }
/// ```
#[proc_macro]
pub fn make_wrapper(item: TokenStream) -> TokenStream {
    let (name, args, ret) = split_fn(&item);

    let (ts_patterns, ts_types) = args_divide_grouped(&args);

    let mut out = TokenStream::new();
    out.extend(
        format!(
            "let {} = read_private_input::<{}>().unwrap();",
            ts_patterns, ts_types
        )
        .parse::<TokenStream>(),
    );

    let public_inputs = toml::from_str::<toml::Table>(include_str!(concat!(
        env!("INPUTS_DIR"),
        "/default_public_input.toml"
    )))
    .unwrap();
    let (public_patterns, public_types) =
        args_divide_public(&args, &public_inputs.keys().collect()).0;
    let public_patterns: Vec<(TokenStream, TokenStream)> = public_patterns
        .into_iter()
        .zip(public_types.into_iter())
        .collect();
    for (pattern, ptype) in public_patterns {
        out.extend(format!("write_output::<{}>(&{});", ptype, pattern).parse::<TokenStream>());
    }

    out.extend(
        format!("write_output::<{}>(&zkp::{}{});", ret, name, ts_patterns).parse::<TokenStream>(),
    );

    let mut block = TokenStream::new();
    block.extend(format!("{{ {} }}", out).parse::<TokenStream>());
    block
}