blob: 901a719cdc4e27e82d29a7c2cd918be78fe76cd5 [file] [log] [blame]
Haibo Huangf5ffad32020-05-25 23:19:03 -07001//! [![github]](https://github.com/dtolnay/proc-macro-hack) [![crates-io]](https://crates.io/crates/proc-macro-hack) [![docs-rs]](https://docs.rs/proc-macro-hack)
2//!
3//! [github]: https://img.shields.io/badge/github-8da0cb?style=for-the-badge&labelColor=555555&logo=github
4//! [crates-io]: https://img.shields.io/badge/crates.io-fc8d62?style=for-the-badge&labelColor=555555&logo=rust
5//! [docs-rs]: https://img.shields.io/badge/docs.rs-66c2a5?style=for-the-badge&labelColor=555555&logoColor=white&logo=
6//!
7//! <br>
8//!
Haibo Huange23d8e22020-07-26 23:48:13 -07009//! <table><tr><td><hr>
10//! <b>Note:</b> <i>As of Rust 1.45 this crate is superseded by native support
11//! for #[proc_macro] in expression position. Only consider using this crate if
12//! you care about supporting compilers between 1.31 and 1.45.</i>
13//! <hr></td></tr></table>
14//!
15//! Since Rust 1.30, the language supports user-defined function-like procedural
Jason Macnake62a2eb2020-03-19 20:41:17 +000016//! macros. However these can only be invoked in item position, not in
17//! statements or expressions.
18//!
19//! This crate implements an alternative type of procedural macro that can be
20//! invoked in statement or expression position.
21//!
22//! # Defining procedural macros
23//!
24//! Two crates are required to define a procedural macro.
25//!
26//! ## The implementation crate
27//!
28//! This crate must contain nothing but procedural macros. Private helper
29//! functions and private modules are fine but nothing can be public.
30//!
Haibo Huangf5ffad32020-05-25 23:19:03 -070031//! [&raquo; example of an implementation crate][demo-hack-impl]
Jason Macnake62a2eb2020-03-19 20:41:17 +000032//!
33//! Just like you would use a #\[proc_macro\] attribute to define a natively
34//! supported procedural macro, use proc-macro-hack's #\[proc_macro_hack\]
35//! attribute to define a procedural macro that works in expression position.
36//! The function signature is the same as for ordinary function-like procedural
37//! macros.
38//!
39//! ```
Haibo Huange23d8e22020-07-26 23:48:13 -070040//! # extern crate proc_macro;
41//! #
Jason Macnake62a2eb2020-03-19 20:41:17 +000042//! use proc_macro::TokenStream;
43//! use proc_macro_hack::proc_macro_hack;
44//! use quote::quote;
45//! use syn::{parse_macro_input, Expr};
46//!
47//! # const IGNORE: &str = stringify! {
48//! #[proc_macro_hack]
49//! # };
50//! pub fn add_one(input: TokenStream) -> TokenStream {
51//! let expr = parse_macro_input!(input as Expr);
52//! TokenStream::from(quote! {
53//! 1 + (#expr)
54//! })
55//! }
56//! #
57//! # fn main() {}
58//! ```
59//!
60//! ## The declaration crate
61//!
62//! This crate is allowed to contain other public things if you need, for
63//! example traits or functions or ordinary macros.
64//!
Haibo Huangf5ffad32020-05-25 23:19:03 -070065//! [&raquo; example of a declaration crate][demo-hack]
Jason Macnake62a2eb2020-03-19 20:41:17 +000066//!
67//! Within the declaration crate there needs to be a re-export of your
68//! procedural macro from the implementation crate. The re-export also carries a
69//! \#\[proc_macro_hack\] attribute.
70//!
71//! ```
72//! use proc_macro_hack::proc_macro_hack;
73//!
74//! /// Add one to an expression.
75//! ///
76//! /// (Documentation goes here on the re-export, not in the other crate.)
77//! #[proc_macro_hack]
78//! pub use demo_hack_impl::add_one;
79//! #
80//! # fn main() {}
81//! ```
82//!
83//! Both crates depend on `proc-macro-hack`:
84//!
85//! ```toml
86//! [dependencies]
87//! proc-macro-hack = "0.5"
88//! ```
89//!
90//! Additionally, your implementation crate (but not your declaration crate) is
91//! a proc macro crate:
92//!
93//! ```toml
94//! [lib]
95//! proc-macro = true
96//! ```
97//!
98//! # Using procedural macros
99//!
100//! Users of your crate depend on your declaration crate (not your
101//! implementation crate), then use your procedural macros as usual.
102//!
Haibo Huangf5ffad32020-05-25 23:19:03 -0700103//! [&raquo; example of a downstream crate][example]
Jason Macnake62a2eb2020-03-19 20:41:17 +0000104//!
105//! ```
106//! use demo_hack::add_one;
107//!
108//! fn main() {
109//! let two = 2;
110//! let nine = add_one!(two) + add_one!(2 + 3);
111//! println!("nine = {}", nine);
112//! }
113//! ```
114//!
115//! [demo-hack-impl]: https://github.com/dtolnay/proc-macro-hack/tree/master/demo-hack-impl
116//! [demo-hack]: https://github.com/dtolnay/proc-macro-hack/tree/master/demo-hack
117//! [example]: https://github.com/dtolnay/proc-macro-hack/tree/master/example
118//!
119//! # Limitations
120//!
121//! - Only proc macros in expression position are supported. Proc macros in
122//! pattern position ([#20]) are not supported.
123//!
124//! - By default, nested invocations are not supported i.e. the code emitted by
125//! a proc-macro-hack macro invocation cannot contain recursive calls to the
126//! same proc-macro-hack macro nor calls to any other proc-macro-hack macros.
127//! Use [`proc-macro-nested`] if you require support for nested invocations.
128//!
129//! - By default, hygiene is structured such that the expanded code can't refer
130//! to local variables other than those passed by name somewhere in the macro
131//! input. If your macro must refer to *local* variables that don't get named
132//! in the macro input, use `#[proc_macro_hack(fake_call_site)]` on the
133//! re-export in your declaration crate. *Most macros won't need this.*
134//!
135//! [#10]: https://github.com/dtolnay/proc-macro-hack/issues/10
136//! [#20]: https://github.com/dtolnay/proc-macro-hack/issues/20
137//! [`proc-macro-nested`]: https://docs.rs/proc-macro-nested
138
139#![recursion_limit = "512"]
Chih-Hung Hsiehd5ad55b2020-04-17 14:32:18 -0700140#![allow(clippy::needless_doctest_main, clippy::toplevel_ref_arg)]
Jason Macnake62a2eb2020-03-19 20:41:17 +0000141
142extern crate proc_macro;
143
Chih-Hung Hsiehd5ad55b2020-04-17 14:32:18 -0700144#[macro_use]
145mod quote;
Jason Macnake62a2eb2020-03-19 20:41:17 +0000146
Chih-Hung Hsiehd5ad55b2020-04-17 14:32:18 -0700147mod error;
Haibo Huangf5ffad32020-05-25 23:19:03 -0700148mod iter;
Chih-Hung Hsiehd5ad55b2020-04-17 14:32:18 -0700149mod parse;
150
151use crate::error::{compile_error, Error};
Haibo Huangf5ffad32020-05-25 23:19:03 -0700152use crate::iter::Iter;
153use crate::parse::{
154 parse_define_args, parse_enum_hack, parse_export_args, parse_fake_call_site, parse_input,
155};
156use proc_macro::{Ident, Punct, Spacing, Span, TokenStream, TokenTree};
Chih-Hung Hsiehd5ad55b2020-04-17 14:32:18 -0700157use std::fmt::Write;
Chih-Hung Hsiehd5ad55b2020-04-17 14:32:18 -0700158
Chih-Hung Hsiehd5ad55b2020-04-17 14:32:18 -0700159type Visibility = Option<Span>;
Jason Macnake62a2eb2020-03-19 20:41:17 +0000160
161enum Input {
162 Export(Export),
163 Define(Define),
164}
165
166// pub use demo_hack_impl::{m1, m2 as qrst};
167struct Export {
168 attrs: TokenStream,
169 vis: Visibility,
170 from: Ident,
171 macros: Vec<Macro>,
172}
173
174// pub fn m1(input: TokenStream) -> TokenStream { ... }
175struct Define {
176 attrs: TokenStream,
177 name: Ident,
178 body: TokenStream,
179}
180
181struct Macro {
182 name: Ident,
183 export_as: Ident,
184}
185
Jason Macnake62a2eb2020-03-19 20:41:17 +0000186#[proc_macro_attribute]
Chih-Hung Hsiehd5ad55b2020-04-17 14:32:18 -0700187pub fn proc_macro_hack(args: TokenStream, input: TokenStream) -> TokenStream {
Haibo Huangf5ffad32020-05-25 23:19:03 -0700188 let ref mut args = iter::new(args);
189 let ref mut input = iter::new(input);
Chih-Hung Hsiehd5ad55b2020-04-17 14:32:18 -0700190 expand_proc_macro_hack(args, input).unwrap_or_else(compile_error)
191}
192
193fn expand_proc_macro_hack(args: Iter, input: Iter) -> Result<TokenStream, Error> {
194 match parse_input(input)? {
Jason Macnake62a2eb2020-03-19 20:41:17 +0000195 Input::Export(export) => {
Chih-Hung Hsiehd5ad55b2020-04-17 14:32:18 -0700196 let args = parse_export_args(args)?;
197 Ok(expand_export(export, args))
Jason Macnake62a2eb2020-03-19 20:41:17 +0000198 }
199 Input::Define(define) => {
Chih-Hung Hsiehd5ad55b2020-04-17 14:32:18 -0700200 parse_define_args(args)?;
201 Ok(expand_define(define))
Jason Macnake62a2eb2020-03-19 20:41:17 +0000202 }
Chih-Hung Hsiehd5ad55b2020-04-17 14:32:18 -0700203 }
Jason Macnake62a2eb2020-03-19 20:41:17 +0000204}
205
Chih-Hung Hsiehd5ad55b2020-04-17 14:32:18 -0700206#[doc(hidden)]
207#[proc_macro_derive(ProcMacroHack)]
208pub fn enum_hack(input: TokenStream) -> TokenStream {
Haibo Huangf5ffad32020-05-25 23:19:03 -0700209 let ref mut input = iter::new(input);
Chih-Hung Hsiehd5ad55b2020-04-17 14:32:18 -0700210 parse_enum_hack(input).unwrap_or_else(compile_error)
211}
212
213struct FakeCallSite {
214 derive: Ident,
215 rest: TokenStream,
216}
217
218#[doc(hidden)]
219#[proc_macro_attribute]
220pub fn fake_call_site(args: TokenStream, input: TokenStream) -> TokenStream {
Haibo Huangf5ffad32020-05-25 23:19:03 -0700221 let ref mut args = iter::new(args);
222 let ref mut input = iter::new(input);
Chih-Hung Hsiehd5ad55b2020-04-17 14:32:18 -0700223 expand_fake_call_site(args, input).unwrap_or_else(compile_error)
224}
225
226fn expand_fake_call_site(args: Iter, input: Iter) -> Result<TokenStream, Error> {
227 let span = match args.next() {
228 Some(token) => token.span(),
229 None => return Ok(input.collect()),
230 };
231
232 let input = parse_fake_call_site(input)?;
233 let mut derive = input.derive;
234 derive.set_span(span);
235 let rest = input.rest;
236
237 Ok(quote! {
238 #[derive(#derive)]
239 #rest
240 })
Jason Macnake62a2eb2020-03-19 20:41:17 +0000241}
242
243struct ExportArgs {
244 support_nested: bool,
245 internal_macro_calls: u16,
246 fake_call_site: bool,
247}
248
Jason Macnake62a2eb2020-03-19 20:41:17 +0000249fn expand_export(export: Export, args: ExportArgs) -> TokenStream {
250 let dummy = dummy_name_for_export(&export);
251
252 let attrs = export.attrs;
Chih-Hung Hsiehd5ad55b2020-04-17 14:32:18 -0700253 let ref vis = export.vis.map(|span| Ident::new("pub", span));
Jason Macnake62a2eb2020-03-19 20:41:17 +0000254 let macro_export = match vis {
255 Some(_) => quote!(#[macro_export]),
256 None => quote!(),
257 };
Chih-Hung Hsiehd5ad55b2020-04-17 14:32:18 -0700258 let crate_prefix = vis.as_ref().map(|_| quote!($crate::));
Jason Macnake62a2eb2020-03-19 20:41:17 +0000259 let enum_variant = if args.support_nested {
260 if args.internal_macro_calls == 0 {
Chih-Hung Hsiehd5ad55b2020-04-17 14:32:18 -0700261 Ident::new("Nested", Span::call_site())
Jason Macnake62a2eb2020-03-19 20:41:17 +0000262 } else {
Chih-Hung Hsiehd5ad55b2020-04-17 14:32:18 -0700263 let name = format!("Nested{}", args.internal_macro_calls);
264 Ident::new(&name, Span::call_site())
Jason Macnake62a2eb2020-03-19 20:41:17 +0000265 }
266 } else {
Chih-Hung Hsiehd5ad55b2020-04-17 14:32:18 -0700267 Ident::new("Value", Span::call_site())
Jason Macnake62a2eb2020-03-19 20:41:17 +0000268 };
269
270 let from = export.from;
Haibo Huange23d8e22020-07-26 23:48:13 -0700271 let mut actual_names = TokenStream::new();
272 let mut export_dispatch = TokenStream::new();
273 let mut export_call_site = TokenStream::new();
274 let mut macro_rules = TokenStream::new();
275 for Macro { name, export_as } in &export.macros {
276 let actual_name = actual_proc_macro_name(&name);
277 let dispatch = dispatch_macro_name(&name);
278 let call_site = call_site_macro_name(&name);
Jason Macnake62a2eb2020-03-19 20:41:17 +0000279
Haibo Huange23d8e22020-07-26 23:48:13 -0700280 if !actual_names.is_empty() {
281 actual_names.extend(quote!(,));
282 }
283 actual_names.extend(quote!(#actual_name));
Jason Macnake62a2eb2020-03-19 20:41:17 +0000284
Haibo Huange23d8e22020-07-26 23:48:13 -0700285 if !export_dispatch.is_empty() {
286 export_dispatch.extend(quote!(,));
287 }
288 export_dispatch.extend(quote!(dispatch as #dispatch));
Jason Macnake62a2eb2020-03-19 20:41:17 +0000289
Haibo Huange23d8e22020-07-26 23:48:13 -0700290 if !export_call_site.is_empty() {
291 export_call_site.extend(quote!(,));
292 }
293 export_call_site.extend(quote!(fake_call_site as #call_site));
Jason Macnake62a2eb2020-03-19 20:41:17 +0000294
Haibo Huange23d8e22020-07-26 23:48:13 -0700295 let do_derive = if !args.fake_call_site {
Jason Macnake62a2eb2020-03-19 20:41:17 +0000296 quote! {
Haibo Huange23d8e22020-07-26 23:48:13 -0700297 #[derive(#crate_prefix #actual_name)]
Jason Macnake62a2eb2020-03-19 20:41:17 +0000298 }
Haibo Huange23d8e22020-07-26 23:48:13 -0700299 } else if crate_prefix.is_some() {
300 quote! {
301 use #crate_prefix #actual_name;
302 #[#crate_prefix #call_site ($($proc_macro)*)]
303 #[derive(#actual_name)]
304 }
305 } else {
306 quote! {
307 #[#call_site ($($proc_macro)*)]
308 #[derive(#actual_name)]
309 }
310 };
Jason Macnake62a2eb2020-03-19 20:41:17 +0000311
Haibo Huange23d8e22020-07-26 23:48:13 -0700312 let proc_macro_call = if args.support_nested {
313 let extra_bangs = (0..args.internal_macro_calls)
314 .map(|_| TokenTree::Punct(Punct::new('!', Spacing::Alone)))
315 .collect::<TokenStream>();
316 quote! {
317 #crate_prefix #dispatch! { ($($proc_macro)*) #extra_bangs }
318 }
319 } else {
320 quote! {
321 proc_macro_call!()
322 }
323 };
324
325 macro_rules.extend(quote! {
326 #attrs
327 #macro_export
328 macro_rules! #export_as {
329 ($($proc_macro:tt)*) => {{
330 #do_derive
331 #[allow(dead_code)]
332 enum ProcMacroHack {
333 #enum_variant = (stringify! { $($proc_macro)* }, 0).1,
334 }
335 #proc_macro_call
336 }};
337 }
338 });
339 }
340
341 if export.macros.len() != 1 {
342 export_dispatch = quote!({#export_dispatch});
343 export_call_site = quote!({#export_call_site});
344 actual_names = quote!({#actual_names});
345 }
346
347 let export_dispatch = if args.support_nested {
348 quote! {
349 #[doc(hidden)]
350 #vis use proc_macro_nested::#export_dispatch;
351 }
352 } else {
353 quote!()
354 };
355
356 let export_call_site = if args.fake_call_site {
357 quote! {
358 #[doc(hidden)]
359 #vis use proc_macro_hack::#export_call_site;
360 }
361 } else {
362 quote!()
363 };
364
365 let expanded = quote! {
366 #[doc(hidden)]
367 #vis use #from::#actual_names;
368
369 #export_dispatch
370 #export_call_site
371
372 #macro_rules
373 };
374
375 wrap_in_enum_hack(dummy, expanded)
Jason Macnake62a2eb2020-03-19 20:41:17 +0000376}
377
378fn expand_define(define: Define) -> TokenStream {
379 let attrs = define.attrs;
380 let name = define.name;
381 let dummy = actual_proc_macro_name(&name);
382 let body = define.body;
383
384 quote! {
385 mod #dummy {
386 extern crate proc_macro;
387 pub use self::proc_macro::*;
388 }
389
390 #attrs
391 #[proc_macro_derive(#dummy)]
392 pub fn #dummy(input: #dummy::TokenStream) -> #dummy::TokenStream {
393 use std::iter::FromIterator;
394
395 let mut iter = input.into_iter();
396 iter.next().unwrap(); // `enum`
397 iter.next().unwrap(); // `ProcMacroHack`
Chih-Hung Hsiehd5ad55b2020-04-17 14:32:18 -0700398 iter.next().unwrap(); // `#`
399 iter.next().unwrap(); // `[allow(dead_code)]`
Jason Macnake62a2eb2020-03-19 20:41:17 +0000400
401 let mut braces = match iter.next().unwrap() {
402 #dummy::TokenTree::Group(group) => group.stream().into_iter(),
403 _ => unimplemented!(),
404 };
405 let variant = braces.next().unwrap(); // `Value` or `Nested`
406 let varname = variant.to_string();
407 let support_nested = varname.starts_with("Nested");
408 braces.next().unwrap(); // `=`
409
410 let mut parens = match braces.next().unwrap() {
411 #dummy::TokenTree::Group(group) => group.stream().into_iter(),
412 _ => unimplemented!(),
413 };
414 parens.next().unwrap(); // `stringify`
415 parens.next().unwrap(); // `!`
416
417 let inner = match parens.next().unwrap() {
418 #dummy::TokenTree::Group(group) => group.stream(),
419 _ => unimplemented!(),
420 };
421
422 let output: #dummy::TokenStream = #name(inner.clone());
423
424 fn count_bangs(input: #dummy::TokenStream) -> usize {
425 let mut count = 0;
426 for token in input {
427 match token {
428 #dummy::TokenTree::Punct(punct) => {
429 if punct.as_char() == '!' {
430 count += 1;
431 }
432 }
433 #dummy::TokenTree::Group(group) => {
434 count += count_bangs(group.stream());
435 }
436 _ => {}
437 }
438 }
439 count
440 }
441
442 // macro_rules! proc_macro_call {
443 // () => { #output }
444 // }
445 #dummy::TokenStream::from_iter(vec![
446 #dummy::TokenTree::Ident(
447 #dummy::Ident::new("macro_rules", #dummy::Span::call_site()),
448 ),
449 #dummy::TokenTree::Punct(
450 #dummy::Punct::new('!', #dummy::Spacing::Alone),
451 ),
452 #dummy::TokenTree::Ident(
453 #dummy::Ident::new(
454 &if support_nested {
455 let extra_bangs = if varname == "Nested" {
456 0
457 } else {
458 varname["Nested".len()..].parse().unwrap()
459 };
460 format!("proc_macro_call_{}", extra_bangs + count_bangs(inner))
461 } else {
462 String::from("proc_macro_call")
463 },
464 #dummy::Span::call_site(),
465 ),
466 ),
467 #dummy::TokenTree::Group(
468 #dummy::Group::new(#dummy::Delimiter::Brace, #dummy::TokenStream::from_iter(vec![
469 #dummy::TokenTree::Group(
470 #dummy::Group::new(#dummy::Delimiter::Parenthesis, #dummy::TokenStream::new()),
471 ),
472 #dummy::TokenTree::Punct(
473 #dummy::Punct::new('=', #dummy::Spacing::Joint),
474 ),
475 #dummy::TokenTree::Punct(
476 #dummy::Punct::new('>', #dummy::Spacing::Alone),
477 ),
478 #dummy::TokenTree::Group(
479 #dummy::Group::new(#dummy::Delimiter::Brace, output),
480 ),
481 ])),
482 ),
483 ])
484 }
485
486 fn #name #body
487 }
488}
489
490fn actual_proc_macro_name(conceptual: &Ident) -> Ident {
Chih-Hung Hsiehd5ad55b2020-04-17 14:32:18 -0700491 Ident::new(
492 &format!("proc_macro_hack_{}", conceptual),
493 conceptual.span(),
494 )
Jason Macnake62a2eb2020-03-19 20:41:17 +0000495}
496
497fn dispatch_macro_name(conceptual: &Ident) -> Ident {
Chih-Hung Hsiehd5ad55b2020-04-17 14:32:18 -0700498 Ident::new(
499 &format!("proc_macro_call_{}", conceptual),
500 conceptual.span(),
501 )
Jason Macnake62a2eb2020-03-19 20:41:17 +0000502}
503
504fn call_site_macro_name(conceptual: &Ident) -> Ident {
Chih-Hung Hsiehd5ad55b2020-04-17 14:32:18 -0700505 Ident::new(
506 &format!("proc_macro_fake_call_site_{}", conceptual),
507 conceptual.span(),
508 )
Jason Macnake62a2eb2020-03-19 20:41:17 +0000509}
510
511fn dummy_name_for_export(export: &Export) -> String {
512 let mut dummy = String::new();
Chih-Hung Hsiehd5ad55b2020-04-17 14:32:18 -0700513 let from = unraw(&export.from).to_string();
Jason Macnake62a2eb2020-03-19 20:41:17 +0000514 write!(dummy, "_{}{}", from.len(), from).unwrap();
515 for m in &export.macros {
Chih-Hung Hsiehd5ad55b2020-04-17 14:32:18 -0700516 let name = unraw(&m.name).to_string();
Jason Macnake62a2eb2020-03-19 20:41:17 +0000517 write!(dummy, "_{}{}", name.len(), name).unwrap();
518 }
519 dummy
520}
521
Chih-Hung Hsiehd5ad55b2020-04-17 14:32:18 -0700522fn unraw(ident: &Ident) -> Ident {
523 let string = ident.to_string();
524 if string.starts_with("r#") {
525 Ident::new(&string[2..], ident.span())
526 } else {
527 ident.clone()
528 }
529}
530
Jason Macnake62a2eb2020-03-19 20:41:17 +0000531fn wrap_in_enum_hack(dummy: String, inner: TokenStream) -> TokenStream {
532 let dummy = Ident::new(&dummy, Span::call_site());
533 quote! {
534 #[derive(proc_macro_hack::ProcMacroHack)]
535 enum #dummy {
536 Value = (stringify! { #inner }, 0).1,
537 }
538 }
539}