@@ -42,11 +42,86 @@ pub enum BinOpToken {
42
42
Shr ,
43
43
}
44
44
45
+ // This type must not implement `Hash` due to the unusual `PartialEq` impl below.
46
+ #[ derive( Copy , Clone , Debug , Encodable , Decodable , HashStable_Generic ) ]
47
+ pub enum InvisibleOrigin {
48
+ // From the expansion of a metavariable in a declarative macro.
49
+ MetaVar ( MetaVarKind ) ,
50
+
51
+ // Converted from `proc_macro::Delimiter` in
52
+ // `proc_macro::Delimiter::to_internal`, i.e. returned by a proc macro.
53
+ ProcMacro ,
54
+
55
+ // Converted from `TokenKind::Interpolated` in
56
+ // `TokenStream::flatten_token`. Treated similarly to `ProcMacro`.
57
+ FlattenToken ,
58
+ }
59
+
60
+ impl PartialEq for InvisibleOrigin {
61
+ #[ inline]
62
+ fn eq ( & self , _other : & InvisibleOrigin ) -> bool {
63
+ // When we had AST-based nonterminals we couldn't compare them, and the
64
+ // old `Nonterminal` type had an `eq` that always returned false,
65
+ // resulting in this restriction:
66
+ // https://doc.rust-lang.org/nightly/reference/macros-by-example.html#forwarding-a-matched-fragment
67
+ // This `eq` emulates that behaviour. We could consider lifting this
68
+ // restriction now but there are still cases involving invisible
69
+ // delimiters that make it harder than it first appears.
70
+ false
71
+ }
72
+ }
73
+
74
+ /// Annoyingly similar to `NonterminalKind`, but the slight differences are important.
75
+ #[ derive( Debug , Copy , Clone , PartialEq , Eq , Encodable , Decodable , Hash , HashStable_Generic ) ]
76
+ pub enum MetaVarKind {
77
+ Item ,
78
+ Block ,
79
+ Stmt ,
80
+ Pat ( NtPatKind ) ,
81
+ Expr {
82
+ kind : NtExprKind ,
83
+ // This field is needed for `Token::can_begin_literal_maybe_minus`.
84
+ can_begin_literal_maybe_minus : bool ,
85
+ // This field is needed for `Token::can_begin_string_literal`.
86
+ can_begin_string_literal : bool ,
87
+ } ,
88
+ Ty ,
89
+ Ident ,
90
+ Lifetime ,
91
+ Literal ,
92
+ Meta ,
93
+ Path ,
94
+ Vis ,
95
+ TT ,
96
+ }
97
+
98
+ impl fmt:: Display for MetaVarKind {
99
+ fn fmt ( & self , f : & mut fmt:: Formatter < ' _ > ) -> fmt:: Result {
100
+ let sym = match self {
101
+ MetaVarKind :: Item => sym:: item,
102
+ MetaVarKind :: Block => sym:: block,
103
+ MetaVarKind :: Stmt => sym:: stmt,
104
+ MetaVarKind :: Pat ( PatParam { inferred : true } | PatWithOr ) => sym:: pat,
105
+ MetaVarKind :: Pat ( PatParam { inferred : false } ) => sym:: pat_param,
106
+ MetaVarKind :: Expr { kind : Expr2021 { inferred : true } | Expr , .. } => sym:: expr,
107
+ MetaVarKind :: Expr { kind : Expr2021 { inferred : false } , .. } => sym:: expr_2021,
108
+ MetaVarKind :: Ty => sym:: ty,
109
+ MetaVarKind :: Ident => sym:: ident,
110
+ MetaVarKind :: Lifetime => sym:: lifetime,
111
+ MetaVarKind :: Literal => sym:: literal,
112
+ MetaVarKind :: Meta => sym:: meta,
113
+ MetaVarKind :: Path => sym:: path,
114
+ MetaVarKind :: Vis => sym:: vis,
115
+ MetaVarKind :: TT => sym:: tt,
116
+ } ;
117
+ write ! ( f, "{sym}" )
118
+ }
119
+ }
120
+
45
121
/// Describes how a sequence of token trees is delimited.
46
122
/// Cannot use `proc_macro::Delimiter` directly because this
47
123
/// structure should implement some additional traits.
48
- #[ derive( Copy , Clone , Debug , PartialEq , Eq ) ]
49
- #[ derive( Encodable , Decodable , Hash , HashStable_Generic ) ]
124
+ #[ derive( Copy , Clone , Debug , PartialEq , Encodable , Decodable , HashStable_Generic ) ]
50
125
pub enum Delimiter {
51
126
/// `( ... )`
52
127
Parenthesis ,
@@ -59,7 +134,34 @@ pub enum Delimiter {
59
134
/// "macro variable" `$var`. It is important to preserve operator priorities in cases like
60
135
/// `$var * 3` where `$var` is `1 + 2`.
61
136
/// Invisible delimiters might not survive roundtrip of a token stream through a string.
62
- Invisible ,
137
+ Invisible ( InvisibleOrigin ) ,
138
+ }
139
+
140
+ impl Delimiter {
141
+ // Should the parser skip these delimiters? Only happens for certain kinds
142
+ // of invisible delimiters. Ideally this function will eventually disappear
143
+ // and no invisible delimiters will be skipped.
144
+ #[ inline]
145
+ pub fn skip ( & self ) -> bool {
146
+ match self {
147
+ Delimiter :: Parenthesis | Delimiter :: Bracket | Delimiter :: Brace => false ,
148
+ Delimiter :: Invisible ( InvisibleOrigin :: MetaVar ( _) ) => false ,
149
+ Delimiter :: Invisible ( InvisibleOrigin :: FlattenToken | InvisibleOrigin :: ProcMacro ) => {
150
+ true
151
+ }
152
+ }
153
+ }
154
+
155
+ // This exists because `InvisibleOrigin`s should be compared. It is only used for assertions.
156
+ pub fn eq_ignoring_invisible_origin ( & self , other : & Delimiter ) -> bool {
157
+ match ( self , other) {
158
+ ( Delimiter :: Parenthesis , Delimiter :: Parenthesis ) => true ,
159
+ ( Delimiter :: Brace , Delimiter :: Brace ) => true ,
160
+ ( Delimiter :: Bracket , Delimiter :: Bracket ) => true ,
161
+ ( Delimiter :: Invisible ( _) , Delimiter :: Invisible ( _) ) => true ,
162
+ _ => false ,
163
+ }
164
+ }
63
165
}
64
166
65
167
// Note that the suffix is *not* considered when deciding the `LitKind` in this
@@ -496,10 +598,11 @@ impl Token {
496
598
/// **NB**: Take care when modifying this function, since it will change
497
599
/// the stable set of tokens that are allowed to match an expr nonterminal.
498
600
pub fn can_begin_expr ( & self ) -> bool {
601
+ use Delimiter :: * ;
499
602
match self . uninterpolate ( ) . kind {
500
603
Ident ( name, is_raw) =>
501
604
ident_can_begin_expr ( name, self . span , is_raw) , // value name or keyword
502
- OpenDelim ( .. ) | // tuple, array or block
605
+ OpenDelim ( Parenthesis | Brace | Bracket ) | // tuple, array or block
503
606
Literal ( ..) | // literal
504
607
Not | // operator not
505
608
BinOp ( Minus ) | // unary minus
@@ -510,7 +613,7 @@ impl Token {
510
613
// DotDotDot is no longer supported, but we need some way to display the error
511
614
DotDot | DotDotDot | DotDotEq | // range notation
512
615
Lt | BinOp ( Shl ) | // associated path
513
- PathSep | // global path
616
+ PathSep | // global path
514
617
Lifetime ( ..) | // labeled loop
515
618
Pound => true , // expression attributes
516
619
Interpolated ( ref nt) =>
@@ -520,6 +623,12 @@ impl Token {
520
623
NtLiteral ( ..) |
521
624
NtPath ( ..)
522
625
) ,
626
+ OpenDelim ( Delimiter :: Invisible ( InvisibleOrigin :: MetaVar (
627
+ MetaVarKind :: Block |
628
+ MetaVarKind :: Expr { .. } |
629
+ MetaVarKind :: Literal |
630
+ MetaVarKind :: Path
631
+ ) ) ) => true ,
523
632
_ => false ,
524
633
}
525
634
}
@@ -553,6 +662,14 @@ impl Token {
553
662
| NtPath ( ..)
554
663
| NtTy ( ..)
555
664
) ,
665
+ OpenDelim ( Delimiter :: Invisible ( InvisibleOrigin :: MetaVar (
666
+ MetaVarKind :: Expr { .. } |
667
+ MetaVarKind :: Literal |
668
+ MetaVarKind :: Meta |
669
+ MetaVarKind :: Pat ( _) |
670
+ MetaVarKind :: Path |
671
+ MetaVarKind :: Ty
672
+ ) ) ) => true ,
556
673
_ => false ,
557
674
}
558
675
}
@@ -573,6 +690,10 @@ impl Token {
573
690
Lt | BinOp ( Shl ) | // associated path
574
691
PathSep => true , // global path
575
692
Interpolated ( ref nt) => matches ! ( & * * nt, NtTy ( ..) | NtPath ( ..) ) ,
693
+ OpenDelim ( Delimiter :: Invisible ( InvisibleOrigin :: MetaVar (
694
+ MetaVarKind :: Ty |
695
+ MetaVarKind :: Path
696
+ ) ) ) => true ,
576
697
// For anonymous structs or unions, which only appear in specific positions
577
698
// (type of struct fields or union fields), we don't consider them as regular types
578
699
_ => false ,
@@ -585,6 +706,9 @@ impl Token {
585
706
OpenDelim ( Delimiter :: Brace ) | Literal ( ..) | BinOp ( Minus ) => true ,
586
707
Ident ( name, IdentIsRaw :: No ) if name. is_bool_lit ( ) => true ,
587
708
Interpolated ( ref nt) => matches ! ( & * * nt, NtExpr ( ..) | NtBlock ( ..) | NtLiteral ( ..) ) ,
709
+ OpenDelim ( Delimiter :: Invisible ( InvisibleOrigin :: MetaVar (
710
+ MetaVarKind :: Expr { .. } | MetaVarKind :: Block | MetaVarKind :: Literal ,
711
+ ) ) ) => true ,
588
712
_ => false ,
589
713
}
590
714
}
@@ -641,6 +765,13 @@ impl Token {
641
765
} ,
642
766
_ => false ,
643
767
} ,
768
+ OpenDelim ( Delimiter :: Invisible ( InvisibleOrigin :: MetaVar ( mv_kind) ) ) => match mv_kind {
769
+ MetaVarKind :: Literal => true ,
770
+ MetaVarKind :: Expr { can_begin_literal_maybe_minus, .. } => {
771
+ can_begin_literal_maybe_minus
772
+ }
773
+ _ => false ,
774
+ } ,
644
775
_ => false ,
645
776
}
646
777
}
@@ -656,6 +787,11 @@ impl Token {
656
787
} ,
657
788
_ => false ,
658
789
} ,
790
+ OpenDelim ( Delimiter :: Invisible ( InvisibleOrigin :: MetaVar ( mv_kind) ) ) => match mv_kind {
791
+ MetaVarKind :: Literal => true ,
792
+ MetaVarKind :: Expr { can_begin_string_literal, .. } => can_begin_string_literal,
793
+ _ => false ,
794
+ } ,
659
795
_ => false ,
660
796
}
661
797
}
@@ -896,7 +1032,7 @@ impl PartialEq<TokenKind> for Token {
896
1032
}
897
1033
}
898
1034
899
- #[ derive( Debug , Copy , Clone , PartialEq , Eq , Encodable , Decodable ) ]
1035
+ #[ derive( Debug , Copy , Clone , PartialEq , Eq , Encodable , Decodable , Hash , HashStable_Generic ) ]
900
1036
pub enum NtPatKind {
901
1037
// Matches or-patterns. Was written using `pat` in edition 2021 or later.
902
1038
PatWithOr ,
@@ -906,7 +1042,7 @@ pub enum NtPatKind {
906
1042
PatParam { inferred : bool } ,
907
1043
}
908
1044
909
- #[ derive( Debug , Copy , Clone , PartialEq , Eq , Encodable , Decodable ) ]
1045
+ #[ derive( Debug , Copy , Clone , PartialEq , Eq , Encodable , Decodable , Hash , HashStable_Generic ) ]
910
1046
pub enum NtExprKind {
911
1047
// Matches expressions using the post-edition 2024. Was written using
912
1048
// `expr` in edition 2024 or later.
@@ -933,7 +1069,7 @@ pub enum Nonterminal {
933
1069
NtVis ( P < ast:: Visibility > ) ,
934
1070
}
935
1071
936
- #[ derive( Debug , Copy , Clone , PartialEq , Encodable , Decodable ) ]
1072
+ #[ derive( Debug , Copy , Clone , PartialEq , Eq , Encodable , Decodable , Hash , HashStable_Generic ) ]
937
1073
pub enum NonterminalKind {
938
1074
Item ,
939
1075
Block ,
0 commit comments