genco_macros/lib.rs
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
//! [<img alt="github" src="https://img.shields.io/badge/github-udoprog/genco-8da0cb?style=for-the-badge&logo=github" height="20">](https://github.com/udoprog/genco)
//! [<img alt="crates.io" src="https://img.shields.io/crates/v/genco-macros.svg?style=for-the-badge&color=fc8d62&logo=rust" height="20">](https://crates.io/crates/genco-macros)
//! [<img alt="docs.rs" src="https://img.shields.io/badge/docs.rs-genco--macros-66c2a5?style=for-the-badge&logoColor=white&logo=" height="20">](https://docs.rs/genco-macros)
#![recursion_limit = "256"]
#![allow(clippy::type_complexity)]
#![cfg_attr(proc_macro_span, feature(proc_macro_span))]
extern crate proc_macro;
use proc_macro2::Span;
use syn::parse::{ParseStream, Parser as _};
struct Ctxt {
receiver: syn::Ident,
module: syn::Path,
}
impl Default for Ctxt {
fn default() -> Self {
let mut module = syn::Path {
leading_colon: None,
segments: syn::punctuated::Punctuated::default(),
};
module
.segments
.push(syn::Ident::new("genco", Span::call_site()).into());
Self {
receiver: syn::Ident::new("__genco_macros_toks", Span::call_site()),
module,
}
}
}
mod ast;
mod cursor;
mod encoder;
mod fake;
mod quote;
mod quote_fn;
mod quote_in;
mod requirements;
mod static_buffer;
mod string_parser;
#[proc_macro]
pub fn quote(input: proc_macro::TokenStream) -> proc_macro::TokenStream {
let cx = Ctxt::default();
let parser = crate::quote::Quote::new(&cx);
let parser = move |stream: ParseStream| parser.parse(stream);
let (req, output) = match parser.parse(input) {
Ok(data) => data,
Err(e) => return proc_macro::TokenStream::from(e.to_compile_error()),
};
let check = req.into_check(&cx.receiver);
let Ctxt { receiver, module } = &cx;
let gen = q::quote! {{
let mut #receiver = #module::tokens::Tokens::new();
{
let mut #receiver = &mut #receiver;
#output
}
#check
#receiver
}};
gen.into()
}
#[proc_macro]
pub fn quote_in(input: proc_macro::TokenStream) -> proc_macro::TokenStream {
let quote_in = syn::parse_macro_input!(input as quote_in::QuoteIn);
quote_in.stream.into()
}
#[proc_macro]
pub fn quote_fn(input: proc_macro::TokenStream) -> proc_macro::TokenStream {
let quote_fn = syn::parse_macro_input!(input as quote_fn::QuoteFn);
quote_fn.stream.into()
}