Skip to content

Commit 1bcc6dc

Browse files
committed
Auto merge of #46895 - ricochet1k:macro-lifetimes, r=jseyfried
Allow lifetimes in macros This is a resurrection of PR #41927 which was a resurrection of #33135, which is intended to fix #34303. In short, this allows macros_rules! to use :lifetime as a matcher to match 'lifetimes. Still to do: - [x] Feature gate
2 parents f3ca88c + 8b4bdc2 commit 1bcc6dc

15 files changed

+231
-21
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
# `macro_lifetime_matcher`
2+
3+
The tracking issue for this feature is: [#46895]
4+
5+
With this feature gate enabled, the [list of fragment specifiers][frags] gains one more entry:
6+
7+
* `lifetime`: a lifetime. Examples: 'static, 'a.
8+
9+
A `lifetime` variable may be followed by anything.
10+
11+
[#46895]: https://github.com/rust-lang/rust/issues/46895
12+
[frags]: ../book/first-edition/macros.html#syntactic-requirements
13+
14+
------------------------

src/libsyntax/ext/quote.rs

+6
Original file line numberDiff line numberDiff line change
@@ -190,6 +190,12 @@ pub mod rt {
190190
}
191191
}
192192

193+
impl ToTokens for ast::Lifetime {
194+
fn to_tokens(&self, _cx: &ExtCtxt) -> Vec<TokenTree> {
195+
vec![TokenTree::Token(DUMMY_SP, token::Lifetime(self.ident))]
196+
}
197+
}
198+
193199
macro_rules! impl_to_tokens_slice {
194200
($t: ty, $sep: expr) => {
195201
impl ToTokens for [$t] {

src/libsyntax/ext/tt/macro_parser.rs

+1
Original file line numberDiff line numberDiff line change
@@ -603,6 +603,7 @@ fn parse_nt<'a>(p: &mut Parser<'a>, sp: Span, name: &str) -> Nonterminal {
603603
"path" => token::NtPath(panictry!(p.parse_path_common(PathStyle::Type, false))),
604604
"meta" => token::NtMeta(panictry!(p.parse_meta_item())),
605605
"vis" => token::NtVis(panictry!(p.parse_visibility(true))),
606+
"lifetime" => token::NtLifetime(p.expect_lifetime()),
606607
// this is not supposed to happen, since it has been checked
607608
// when compiling the macro.
608609
_ => p.span_bug(sp, "invalid fragment specifier")

src/libsyntax/ext/tt/macro_rules.rs

+21-8
Original file line numberDiff line numberDiff line change
@@ -768,10 +768,11 @@ fn token_can_be_followed_by_any(tok: &quoted::TokenTree) -> bool {
768768
/// ANYTHING without fear of future compatibility hazards).
769769
fn frag_can_be_followed_by_any(frag: &str) -> bool {
770770
match frag {
771-
"item" | // always terminated by `}` or `;`
772-
"block" | // exactly one token tree
773-
"ident" | // exactly one token tree
774-
"meta" | // exactly one token tree
771+
"item" | // always terminated by `}` or `;`
772+
"block" | // exactly one token tree
773+
"ident" | // exactly one token tree
774+
"meta" | // exactly one token tree
775+
"lifetime" | // exactly one token tree
775776
"tt" => // exactly one token tree
776777
true,
777778

@@ -832,8 +833,8 @@ fn is_in_follow(tok: &quoted::TokenTree, frag: &str) -> Result<bool, (String, &'
832833
TokenTree::MetaVarDecl(_, _, frag) if frag.name == "block" => Ok(true),
833834
_ => Ok(false),
834835
},
835-
"ident" => {
836-
// being a single token, idents are harmless
836+
"ident" | "lifetime" => {
837+
// being a single token, idents and lifetimes are harmless
837838
Ok(true)
838839
},
839840
"meta" | "tt" => {
@@ -887,9 +888,21 @@ fn is_legal_fragment_specifier(sess: &ParseSess,
887888
match frag_name {
888889
"item" | "block" | "stmt" | "expr" | "pat" |
889890
"path" | "ty" | "ident" | "meta" | "tt" | "" => true,
891+
"lifetime" => {
892+
if !features.borrow().macro_lifetime_matcher &&
893+
!attr::contains_name(attrs, "allow_internal_unstable") {
894+
let explain = feature_gate::EXPLAIN_LIFETIME_MATCHER;
895+
emit_feature_err(sess,
896+
"macro_lifetime_matcher",
897+
frag_span,
898+
GateIssue::Language,
899+
explain);
900+
}
901+
true
902+
},
890903
"vis" => {
891-
if !features.borrow().macro_vis_matcher
892-
&& !attr::contains_name(attrs, "allow_internal_unstable") {
904+
if !features.borrow().macro_vis_matcher &&
905+
!attr::contains_name(attrs, "allow_internal_unstable") {
893906
let explain = feature_gate::EXPLAIN_VIS_MATCHER;
894907
emit_feature_err(sess,
895908
"macro_vis_matcher",

src/libsyntax/feature_gate.rs

+6
Original file line numberDiff line numberDiff line change
@@ -447,6 +447,9 @@ declare_features! (
447447

448448
// Termination trait in main (RFC 1937)
449449
(active, termination_trait, "1.24.0", Some(43301)),
450+
451+
// Allows use of the :lifetime macro fragment specifier
452+
(active, macro_lifetime_matcher, "1.24.0", Some(46895)),
450453
);
451454

452455
declare_features! (
@@ -1226,6 +1229,9 @@ pub const EXPLAIN_DERIVE_UNDERSCORE: &'static str =
12261229
pub const EXPLAIN_VIS_MATCHER: &'static str =
12271230
":vis fragment specifier is experimental and subject to change";
12281231

1232+
pub const EXPLAIN_LIFETIME_MATCHER: &'static str =
1233+
":lifetime fragment specifier is experimental and subject to change";
1234+
12291235
pub const EXPLAIN_PLACEMENT_IN: &'static str =
12301236
"placement-in expression syntax is experimental and subject to change.";
12311237

src/libsyntax/fold.rs

+1
Original file line numberDiff line numberDiff line change
@@ -642,6 +642,7 @@ pub fn noop_fold_interpolated<T: Folder>(nt: token::Nonterminal, fld: &mut T)
642642
token::NtWhereClause(fld.fold_where_clause(where_clause)),
643643
token::NtArg(arg) => token::NtArg(fld.fold_arg(arg)),
644644
token::NtVis(vis) => token::NtVis(fld.fold_vis(vis)),
645+
token::NtLifetime(lifetime) => token::NtLifetime(fld.fold_lifetime(lifetime)),
645646
}
646647
}
647648

src/libsyntax/parse/parser.rs

+10-8
Original file line numberDiff line numberDiff line change
@@ -1295,6 +1295,10 @@ impl<'a> Parser<'a> {
12951295
fn get_label(&mut self) -> ast::Ident {
12961296
match self.token {
12971297
token::Lifetime(ref ident) => *ident,
1298+
token::Interpolated(ref nt) => match nt.0 {
1299+
token::NtLifetime(lifetime) => lifetime.ident,
1300+
_ => self.bug("not a lifetime"),
1301+
},
12981302
_ => self.bug("not a lifetime"),
12991303
}
13001304
}
@@ -2031,14 +2035,12 @@ impl<'a> Parser<'a> {
20312035
}
20322036

20332037
/// Parse single lifetime 'a or panic.
2034-
fn expect_lifetime(&mut self) -> Lifetime {
2035-
match self.token {
2036-
token::Lifetime(ident) => {
2037-
let ident_span = self.span;
2038-
self.bump();
2039-
Lifetime { ident: ident, span: ident_span, id: ast::DUMMY_NODE_ID }
2040-
}
2041-
_ => self.span_bug(self.span, "not a lifetime")
2038+
pub fn expect_lifetime(&mut self) -> Lifetime {
2039+
if let Some(lifetime) = self.token.lifetime(self.span) {
2040+
self.bump();
2041+
lifetime
2042+
} else {
2043+
self.span_bug(self.span, "not a lifetime")
20422044
}
20432045
}
20442046

src/libsyntax/parse/token.rs

+23-5
Original file line numberDiff line numberDiff line change
@@ -251,7 +251,7 @@ impl Token {
251251
Lt | BinOp(Shl) | // associated path
252252
ModSep => true, // global path
253253
Interpolated(ref nt) => match nt.0 {
254-
NtIdent(..) | NtTy(..) | NtPath(..) => true,
254+
NtIdent(..) | NtTy(..) | NtPath(..) | NtLifetime(..) => true,
255255
_ => false,
256256
},
257257
_ => false,
@@ -314,14 +314,26 @@ impl Token {
314314
false
315315
}
316316

317-
/// Returns `true` if the token is a lifetime.
318-
pub fn is_lifetime(&self) -> bool {
317+
/// Returns a lifetime with the span and a dummy id if it is a lifetime,
318+
/// or the original lifetime if it is an interpolated lifetime, ignoring
319+
/// the span.
320+
pub fn lifetime(&self, span: Span) -> Option<ast::Lifetime> {
319321
match *self {
320-
Lifetime(..) => true,
321-
_ => false,
322+
Lifetime(ident) =>
323+
Some(ast::Lifetime { ident: ident, span: span, id: ast::DUMMY_NODE_ID }),
324+
Interpolated(ref nt) => match nt.0 {
325+
NtLifetime(lifetime) => Some(lifetime),
326+
_ => None,
327+
},
328+
_ => None,
322329
}
323330
}
324331

332+
/// Returns `true` if the token is a lifetime.
333+
pub fn is_lifetime(&self) -> bool {
334+
self.lifetime(syntax_pos::DUMMY_SP).is_some()
335+
}
336+
325337
/// Returns `true` if the token is either the `mut` or `const` keyword.
326338
pub fn is_mutability(&self) -> bool {
327339
self.is_keyword(keywords::Mut) ||
@@ -486,6 +498,10 @@ impl Token {
486498
let token = Token::Ident(ident.node);
487499
tokens = Some(TokenTree::Token(ident.span, token).into());
488500
}
501+
Nonterminal::NtLifetime(lifetime) => {
502+
let token = Token::Lifetime(lifetime.ident);
503+
tokens = Some(TokenTree::Token(lifetime.span, token).into());
504+
}
489505
Nonterminal::NtTT(ref tt) => {
490506
tokens = Some(tt.clone().into());
491507
}
@@ -524,6 +540,7 @@ pub enum Nonterminal {
524540
NtGenerics(ast::Generics),
525541
NtWhereClause(ast::WhereClause),
526542
NtArg(ast::Arg),
543+
NtLifetime(ast::Lifetime),
527544
}
528545

529546
impl fmt::Debug for Nonterminal {
@@ -546,6 +563,7 @@ impl fmt::Debug for Nonterminal {
546563
NtWhereClause(..) => f.pad("NtWhereClause(..)"),
547564
NtArg(..) => f.pad("NtArg(..)"),
548565
NtVis(..) => f.pad("NtVis(..)"),
566+
NtLifetime(..) => f.pad("NtLifetime(..)"),
549567
}
550568
}
551569
}

src/libsyntax/print/pprust.rs

+1
Original file line numberDiff line numberDiff line change
@@ -279,6 +279,7 @@ pub fn token_to_string(tok: &Token) -> String {
279279
token::NtWhereClause(ref e) => where_clause_to_string(e),
280280
token::NtArg(ref e) => arg_to_string(e),
281281
token::NtVis(ref e) => vis_to_string(e),
282+
token::NtLifetime(ref e) => lifetime_to_string(e),
282283
}
283284
}
284285
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
// Copyright 2012 The Rust Project Developers. See the COPYRIGHT
2+
// file at the top-level directory of this distribution and at
3+
// http://rust-lang.org/COPYRIGHT.
4+
//
5+
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6+
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7+
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8+
// option. This file may not be copied, modified, or distributed
9+
// except according to those terms.
10+
11+
#![feature(macro_lifetime_matcher)]
12+
13+
macro_rules! foo {
14+
($l:lifetime, $l2:lifetime) => {
15+
fn f<$l: $l2, $l2>(arg: &$l str, arg2: &$l2 str) -> &$l str {
16+
arg
17+
}
18+
}
19+
}
20+
21+
pub fn main() {
22+
foo!('a, 'b);
23+
let x: &'static str = f("hi", "there");
24+
assert_eq!("hi", x);
25+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,44 @@
1+
// Copyright 2012 The Rust Project Developers. See the COPYRIGHT
2+
// file at the top-level directory of this distribution and at
3+
// http://rust-lang.org/COPYRIGHT.
4+
//
5+
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6+
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7+
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8+
// option. This file may not be copied, modified, or distributed
9+
// except according to those terms.
10+
11+
#![allow(unreachable_code)]
12+
#![feature(macro_lifetime_matcher)]
13+
14+
macro_rules! x {
15+
($a:lifetime) => {
16+
$a: loop {
17+
break $a;
18+
panic!("failed");
19+
}
20+
}
21+
}
22+
macro_rules! br {
23+
($a:lifetime) => {
24+
break $a;
25+
}
26+
}
27+
macro_rules! br2 {
28+
($b:lifetime) => {
29+
'b: loop {
30+
break $b; // this $b should refer to the outer loop.
31+
}
32+
}
33+
}
34+
fn main() {
35+
x!('a);
36+
'c: loop {
37+
br!('c);
38+
panic!("failed");
39+
}
40+
'b: loop {
41+
br2!('b);
42+
panic!("failed");
43+
}
44+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
// Copyright 2012 The Rust Project Developers. See the COPYRIGHT
2+
// file at the top-level directory of this distribution and at
3+
// http://rust-lang.org/COPYRIGHT.
4+
//
5+
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6+
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7+
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8+
// option. This file may not be copied, modified, or distributed
9+
// except according to those terms.
10+
11+
#![feature(macro_lifetime_matcher)]
12+
13+
macro_rules! foo {
14+
($l:lifetime) => {
15+
fn f(arg: &$l str) -> &$l str {
16+
arg
17+
}
18+
}
19+
}
20+
21+
pub fn main() {
22+
foo!('static);
23+
let x: &'static str = f("hi");
24+
assert_eq!("hi", x);
25+
}

src/test/run-pass/macro-lifetime.rs

+25
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
// Copyright 2012 The Rust Project Developers. See the COPYRIGHT
2+
// file at the top-level directory of this distribution and at
3+
// http://rust-lang.org/COPYRIGHT.
4+
//
5+
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6+
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7+
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8+
// option. This file may not be copied, modified, or distributed
9+
// except according to those terms.
10+
11+
#![feature(macro_lifetime_matcher)]
12+
13+
macro_rules! foo {
14+
($l:lifetime) => {
15+
fn f<$l>(arg: &$l str) -> &$l str {
16+
arg
17+
}
18+
}
19+
}
20+
21+
pub fn main() {
22+
foo!('a);
23+
let x: &'static str = f("hi");
24+
assert_eq!("hi", x);
25+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
// Copyright 2017 The Rust Project Developers. See the COPYRIGHT
2+
// file at the top-level directory of this distribution and at
3+
// http://rust-lang.org/COPYRIGHT.
4+
//
5+
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6+
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7+
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8+
// option. This file may not be copied, modified, or distributed
9+
// except according to those terms.
10+
11+
// Test that the :lifetime macro fragment cannot be used when macro_lifetime_matcher
12+
// feature gate is not used.
13+
14+
macro_rules! m { ($lt:lifetime) => {} }
15+
//~^ ERROR :lifetime fragment specifier is experimental and subject to change
16+
17+
fn main() {
18+
m!('a);
19+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
error: :lifetime fragment specifier is experimental and subject to change (see issue #46895)
2+
--> $DIR/feature-gate-macro-lifetime-matcher.rs:14:19
3+
|
4+
14 | macro_rules! m { ($lt:lifetime) => {} }
5+
| ^^^^^^^^^^^^
6+
|
7+
= help: add #![feature(macro_lifetime_matcher)] to the crate attributes to enable
8+
9+
error: aborting due to previous error
10+

0 commit comments

Comments
 (0)