Skip to content

Commit f109128

Browse files
committed
Parse subslice patterns
cc rust-lang#1479
1 parent a4316d6 commit f109128

File tree

5 files changed

+110
-13
lines changed

5 files changed

+110
-13
lines changed

crates/ra_parser/src/grammar/patterns.rs

+33-12
Original file line numberDiff line numberDiff line change
@@ -36,9 +36,14 @@ pub(super) fn pattern_r(p: &mut Parser, recovery_set: TokenSet) {
3636
// }
3737
if p.at(T![...]) || p.at(T![..=]) || p.at(T![..]) {
3838
let m = lhs.precede(p);
39+
let dots = p.current();
3940
p.bump();
40-
atom_pat(p, recovery_set);
41-
m.complete(p, RANGE_PAT);
41+
if dots == T![..] && !p.at_ts(PATTERN_FIRST) {
42+
m.complete(p, SUBSLICE_PAT);
43+
} else {
44+
atom_pat(p, recovery_set);
45+
m.complete(p, RANGE_PAT);
46+
}
4247
}
4348
// test marco_pat
4449
// fn main() {
@@ -145,7 +150,21 @@ fn path_pat(p: &mut Parser) -> CompletedMarker {
145150
fn tuple_pat_fields(p: &mut Parser) {
146151
assert!(p.at(T!['(']));
147152
p.bump();
148-
pat_list(p, T![')']);
153+
while !p.at(EOF) && !p.at(T![')']) {
154+
match p.current() {
155+
T![..] => p.bump(),
156+
_ => {
157+
if !p.at_ts(PATTERN_FIRST) {
158+
p.error("expected a pattern");
159+
break;
160+
}
161+
pattern(p)
162+
}
163+
}
164+
if !p.at(T![')']) {
165+
p.expect(T![,]);
166+
}
167+
}
149168
p.expect(T![')']);
150169
}
151170

@@ -229,28 +248,30 @@ fn tuple_pat(p: &mut Parser) -> CompletedMarker {
229248
fn slice_pat(p: &mut Parser) -> CompletedMarker {
230249
assert!(p.at(T!['[']));
231250
let m = p.start();
251+
// test subslice_pat
252+
// fn main() {
253+
// let [a, b..] = [];
254+
// let &[ref a.., ref b] = [];
255+
// }
232256
p.bump();
233-
pat_list(p, T![']']);
234-
p.expect(T![']']);
235-
m.complete(p, SLICE_PAT)
236-
}
237-
238-
fn pat_list(p: &mut Parser, ket: SyntaxKind) {
239-
while !p.at(EOF) && !p.at(ket) {
257+
while !p.at(EOF) && !p.at(T![']']) {
240258
match p.current() {
241259
T![..] => p.bump(),
242260
_ => {
243261
if !p.at_ts(PATTERN_FIRST) {
244262
p.error("expected a pattern");
245263
break;
246264
}
247-
pattern(p)
265+
pattern(p);
266+
p.eat(T![..]);
248267
}
249268
}
250-
if !p.at(ket) {
269+
if !p.at(T![']']) {
251270
p.expect(T![,]);
252271
}
253272
}
273+
p.expect(T![']']);
274+
m.complete(p, SLICE_PAT)
254275
}
255276

256277
// test bind_pat

crates/ra_parser/src/syntax_kind/generated.rs

+2
Original file line numberDiff line numberDiff line change
@@ -165,6 +165,7 @@ pub enum SyntaxKind {
165165
SLICE_PAT,
166166
RANGE_PAT,
167167
LITERAL_PAT,
168+
SUBSLICE_PAT,
168169
TUPLE_EXPR,
169170
ARRAY_EXPR,
170171
PAREN_EXPR,
@@ -632,6 +633,7 @@ impl SyntaxKind {
632633
SLICE_PAT => &SyntaxInfo { name: "SLICE_PAT" },
633634
RANGE_PAT => &SyntaxInfo { name: "RANGE_PAT" },
634635
LITERAL_PAT => &SyntaxInfo { name: "LITERAL_PAT" },
636+
SUBSLICE_PAT => &SyntaxInfo { name: "SUBSLICE_PAT" },
635637
TUPLE_EXPR => &SyntaxInfo { name: "TUPLE_EXPR" },
636638
ARRAY_EXPR => &SyntaxInfo { name: "ARRAY_EXPR" },
637639
PAREN_EXPR => &SyntaxInfo { name: "PAREN_EXPR" },

crates/ra_syntax/src/grammar.ron

+2-1
Original file line numberDiff line numberDiff line change
@@ -168,6 +168,7 @@ Grammar(
168168
"SLICE_PAT",
169169
"RANGE_PAT",
170170
"LITERAL_PAT",
171+
"SUBSLICE_PAT",
171172

172173
// atoms
173174
"TUPLE_EXPR",
@@ -686,7 +687,7 @@ Grammar(
686687
"LifetimeArg": (),
687688

688689
"MacroItems": (
689-
traits: [ "ModuleItemOwner", "FnDefOwner" ],
690+
traits: [ "ModuleItemOwner", "FnDefOwner" ],
690691
),
691692

692693
"MacroStmts" : (
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
fn main() {
2+
let [a, b..] = [];
3+
let &[ref a.., ref b] = [];
4+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,69 @@
1+
SOURCE_FILE@[0; 69)
2+
FN_DEF@[0; 68)
3+
FN_KW@[0; 2) "fn"
4+
WHITESPACE@[2; 3) " "
5+
NAME@[3; 7)
6+
IDENT@[3; 7) "main"
7+
PARAM_LIST@[7; 9)
8+
L_PAREN@[7; 8) "("
9+
R_PAREN@[8; 9) ")"
10+
WHITESPACE@[9; 10) " "
11+
BLOCK@[10; 68)
12+
L_CURLY@[10; 11) "{"
13+
WHITESPACE@[11; 16) "\n "
14+
LET_STMT@[16; 34)
15+
LET_KW@[16; 19) "let"
16+
WHITESPACE@[19; 20) " "
17+
SLICE_PAT@[20; 28)
18+
L_BRACK@[20; 21) "["
19+
BIND_PAT@[21; 22)
20+
NAME@[21; 22)
21+
IDENT@[21; 22) "a"
22+
COMMA@[22; 23) ","
23+
WHITESPACE@[23; 24) " "
24+
SUBSLICE_PAT@[24; 27)
25+
BIND_PAT@[24; 25)
26+
NAME@[24; 25)
27+
IDENT@[24; 25) "b"
28+
DOTDOT@[25; 27) ".."
29+
R_BRACK@[27; 28) "]"
30+
WHITESPACE@[28; 29) " "
31+
EQ@[29; 30) "="
32+
WHITESPACE@[30; 31) " "
33+
ARRAY_EXPR@[31; 33)
34+
L_BRACK@[31; 32) "["
35+
R_BRACK@[32; 33) "]"
36+
SEMI@[33; 34) ";"
37+
WHITESPACE@[34; 39) "\n "
38+
LET_STMT@[39; 66)
39+
LET_KW@[39; 42) "let"
40+
WHITESPACE@[42; 43) " "
41+
REF_PAT@[43; 60)
42+
AMP@[43; 44) "&"
43+
SLICE_PAT@[44; 60)
44+
L_BRACK@[44; 45) "["
45+
SUBSLICE_PAT@[45; 52)
46+
BIND_PAT@[45; 50)
47+
REF_KW@[45; 48) "ref"
48+
WHITESPACE@[48; 49) " "
49+
NAME@[49; 50)
50+
IDENT@[49; 50) "a"
51+
DOTDOT@[50; 52) ".."
52+
COMMA@[52; 53) ","
53+
WHITESPACE@[53; 54) " "
54+
BIND_PAT@[54; 59)
55+
REF_KW@[54; 57) "ref"
56+
WHITESPACE@[57; 58) " "
57+
NAME@[58; 59)
58+
IDENT@[58; 59) "b"
59+
R_BRACK@[59; 60) "]"
60+
WHITESPACE@[60; 61) " "
61+
EQ@[61; 62) "="
62+
WHITESPACE@[62; 63) " "
63+
ARRAY_EXPR@[63; 65)
64+
L_BRACK@[63; 64) "["
65+
R_BRACK@[64; 65) "]"
66+
SEMI@[65; 66) ";"
67+
WHITESPACE@[66; 67) "\n"
68+
R_CURLY@[67; 68) "}"
69+
WHITESPACE@[68; 69) "\n"

0 commit comments

Comments
 (0)