- 
                Notifications
    You must be signed in to change notification settings 
- Fork 13.9k
Description
Hi.
I founds some bugs, did a research, and created a repository with reproducible examples, README, and instructions. Here is the content of README, and the whole repo is attached as zip archive. Code is highly documented, so please take a look. Code snippets and rust-playground are mostly useless, because we'are dealing with proc_macro multi-crate builds.
Issues with macros and :vis meta-variable
I did a lot of experiments, and I think I found three issues at once:
- 
Empty :vis meta-variable According to the documentation, :visentity inmacro_rules!should match empty visibility modifier:
 vis:a possibly empty Visibility qualifierWhile this is the case when visibility modifier is followed by some more stuff, it fails on its own: macro_rules! q1 { ($visibility:vis) => {} } q1!(); // error: unexpected end of macro invocation Followed by stuff: macro_rules! q2 { ($visibility:vis $item:ident) => {} } q2!(foobar); // okEven though, adding ident-only branch still matches the first one (with :vis):macro_rules! q3 { ($visibility:vis $item:ident) => {}; ($item:ident) => { compile_error!("Ident-only branch chosen instead of branch with empty vis"); }; } q3!(foobar); // okLive rust code of these examples can be found in ./mac/src/q.rs.
- 
Forwarding empty :vis meta-variable When rustc forwards empty :vismeta-variable, it creates an emptyproc_macro::TokenTree::Groupwhich has no delimiter (proc_macro::Delimiter::Noneaka Ø) nor inner stream content (emptyproc_macro::TokenStream). WhileNonedelimiters is a documented feature (quoted) "important to preserve operator priorities", they "may not survive roundtrip of a token stream through a string".That leads to a workaround, but at the cost of losing context and spans of tokens: let stream: proc_macro2::TokenStream = /* ... */; let stream = stream.to_string() .parse::<proc_macro2::TokenStream>() .unwrap(); Live rust code using this workaround can be found in ./usage/src/main.rs(at the bottom) and./mac/src/lib.rs(turn theUSE_ROUNDTRIPswitch on).Also, quote!macro does not produce empty groups like that, so it is higly inconsistent behavior.Unlike Nonedelimiter, completely empty group is not documented as a useful feature.
- 
Parsing empty group with syn crate Syn crate is somewhat inconsistent when it comes to parsing empty TokenTree Group. Steps to reproduce: - Enable mac_with_ident!(bar);line in./usage/src/with_ident.rsby commenting out/removing#[cfg!(none)]attribute.
- Turn off the USE_ROUNDTRIPswitch in proc macro.
- Try to build.
 As can be seen from the error message TAG_B: Error("unexpected token"),syn::parse2has no problems parsingsyn::Visibilityout of the ParseStream with single empty group, but it fails to "complete" parsing becauseParseBufferis not considered empty (it sees some "tokens" left in the buffer).
- Enable 
Proposal:
- 
Rustc macro matching Fix rustc macro matching subsystem to accept calls without arguments (e.g. q!()) as matching arms with single:vismeta-variable (e.g.macro_rules! q { ($v:vis) => {} })
- 
Rust macro expansion Fix rustc macro expansion subsystem to stop forwarding empty vistokens as an empty groups.
- 
Syn parser robustness Regardless of fixed in rustc, syn/proc_macro2 crates should handle those empty groups gracefully AND uniformly. 
 Probably, skip those blank groups altogether. It's no good that they handle a "blank group" and a "blank group which is followed by stuff" cases differently.
Meta
rustc --version --verbose:
rustc 1.42.0 (b8cedc004 2020-03-09)
binary: rustc
commit-hash: b8cedc00407a4c56a3bda1ed605c6fc166655447
commit-date: 2020-03-09
host: x86_64-unknown-linux-gnu
release: 1.42.0
LLVM version: 9.0
OS: Arch Linux
Kernel: x86_64 Linux 5.6.4-arch1-1