forked from rust-lang/rust
-
Notifications
You must be signed in to change notification settings - Fork 1
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Rollup merge of rust-lang#64690 - petrochenkov:mixed, r=dtolnay
proc_macro API: Expose `macro_rules` hygiene Proc macros do not have direct access to our oldest and most stable hygiene kind - `macro_rules` hygiene. To emulate it macro authors have to go through two steps - first generate a temporary `macro_rules` item (using a derive, at least until rust-lang#64035 is merged), then generate a macro call to that item. Popular crates like [proc_macro_hack](https://crates.io/crates/proc-macro-hack) use this trick to generate hygienic identifiers from proc macros. I'd say that these workarounds with nested macro definitions have more chances to hit some corner cases in our hygiene system, in which we don't have full confidence. So, let's provide a direct access to `macro_rules` hygiene instead. This PR does that by adding a new method `Span::mixed_site` (bikeshedding is welcome) in addition to existing `Span::call_site` (stable) and `Span::def_site` (unstable). Identifiers with this span resolve at def-site in for local variables, labels and `$crate`, and resolve at call-site for everything else, i.e. exactly like identifiers produced by `macro_rules`. This API addition opens the way to stabilizing proc macros in expression positions (rust-lang#54727), for which use of call-site hygiene or workarounds with temporary items would be quite unfortunate. (`macro_rules` expanded in expression position, on the other hand, are stable since 1.0 and widely used.) r? @dtolnay @alexcrichton
- Loading branch information
Showing
10 changed files
with
179 additions
and
35 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,42 @@ | ||
// force-host | ||
// no-prefer-dynamic | ||
|
||
#![feature(proc_macro_hygiene)] | ||
#![feature(proc_macro_mixed_site)] | ||
#![feature(proc_macro_quote)] | ||
|
||
#![crate_type = "proc-macro"] | ||
|
||
extern crate proc_macro; | ||
use proc_macro::*; | ||
|
||
#[proc_macro] | ||
pub fn proc_macro_rules(input: TokenStream) -> TokenStream { | ||
if input.is_empty() { | ||
let id = |s| TokenTree::from(Ident::new(s, Span::mixed_site())); | ||
let item_def = id("ItemDef"); | ||
let local_def = id("local_def"); | ||
let item_use = id("ItemUse"); | ||
let local_use = id("local_use"); | ||
let mut single_quote = Punct::new('\'', Spacing::Joint); | ||
single_quote.set_span(Span::mixed_site()); | ||
let label_use: TokenStream = [ | ||
TokenTree::from(single_quote), | ||
id("label_use"), | ||
].iter().cloned().collect(); | ||
quote!( | ||
struct $item_def; | ||
let $local_def = 0; | ||
|
||
$item_use; // OK | ||
$local_use; // ERROR | ||
break $label_use; // ERROR | ||
) | ||
} else { | ||
let mut dollar_crate = input.into_iter().next().unwrap(); | ||
dollar_crate.set_span(Span::mixed_site()); | ||
quote!( | ||
type A = $dollar_crate::ItemUse; | ||
) | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,26 @@ | ||
// Proc macros using `mixed_site` spans exhibit usual properties of `macro_rules` hygiene. | ||
|
||
// aux-build:mixed-site-span.rs | ||
|
||
#![feature(proc_macro_hygiene)] | ||
|
||
#[macro_use] | ||
extern crate mixed_site_span; | ||
|
||
struct ItemUse; | ||
|
||
fn main() { | ||
'label_use: loop { | ||
let local_use = 1; | ||
proc_macro_rules!(); | ||
//~^ ERROR use of undeclared label `'label_use` | ||
//~| ERROR cannot find value `local_use` in this scope | ||
ItemDef; // OK | ||
local_def; //~ ERROR cannot find value `local_def` in this scope | ||
} | ||
} | ||
|
||
macro_rules! pass_dollar_crate { | ||
() => (proc_macro_rules!($crate);) //~ ERROR cannot find type `ItemUse` in crate `$crate` | ||
} | ||
pass_dollar_crate!(); |
Oops, something went wrong.