1733: Parse arbitrarily complex `box` patterns. r=matklad a=ecstatic-morse

This fully resolves the pattern part of #1412 by enabling the parsing of complex `box` patterns such as:

```rust
let box Struct { box i, j: box Inner(box &x) } = todo!();
```

This introduces a new `ast::BoxPat` (in the mold of `ast::RefPat`) that gets translated to `hir::Pat::Missing`.

Co-authored-by: Dylan MacKenzie <ecstaticmorse@gmail.com>
This commit is contained in:
bors[bot] 2019-08-25 09:47:44 +00:00 committed by GitHub
commit 866b41ddd8
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
14 changed files with 305 additions and 108 deletions

View file

@ -1020,6 +1020,7 @@ where
} }
// FIXME: implement // FIXME: implement
ast::Pat::BoxPat(_) => Pat::Missing,
ast::Pat::LiteralPat(_) => Pat::Missing, ast::Pat::LiteralPat(_) => Pat::Missing,
ast::Pat::SlicePat(_) | ast::Pat::RangePat(_) => Pat::Missing, ast::Pat::SlicePat(_) | ast::Pat::RangePat(_) => Pat::Missing,
}; };

View file

@ -414,8 +414,6 @@ pub(crate) fn match_arm_list(p: &mut Parser) {
// X | Y if Z => (), // X | Y if Z => (),
// | X | Y if Z => (), // | X | Y if Z => (),
// | X => (), // | X => (),
// box X => (),
// Some(box X) => (),
// }; // };
// } // }
fn match_arm(p: &mut Parser) -> BlockLike { fn match_arm(p: &mut Parser) -> BlockLike {

View file

@ -56,37 +56,33 @@ const PAT_RECOVERY_SET: TokenSet =
token_set![LET_KW, IF_KW, WHILE_KW, LOOP_KW, MATCH_KW, R_PAREN, COMMA]; token_set![LET_KW, IF_KW, WHILE_KW, LOOP_KW, MATCH_KW, R_PAREN, COMMA];
fn atom_pat(p: &mut Parser, recovery_set: TokenSet) -> Option<CompletedMarker> { fn atom_pat(p: &mut Parser, recovery_set: TokenSet) -> Option<CompletedMarker> {
let la0 = p.nth(0); // Checks the token after an IDENT to see if a pattern is a path (Struct { .. }) or macro
let la1 = p.nth(1); // (T![x]).
if la0 == T![ref] let is_path_or_macro_pat =
|| la0 == T![mut] |la1| la1 == T![::] || la1 == T!['('] || la1 == T!['{'] || la1 == T![!];
|| la0 == T![box]
|| (la0 == IDENT && !(la1 == T![::] || la1 == T!['('] || la1 == T!['{'] || la1 == T![!]))
{
return Some(bind_pat(p, true));
}
if paths::is_use_path_start(p) {
return Some(path_pat(p));
}
if is_literal_pat_start(p) { let m = match p.nth(0) {
return Some(literal_pat(p)); T![box] => box_pat(p),
} T![ref] | T![mut] | IDENT if !is_path_or_macro_pat(p.nth(1)) => bind_pat(p, true),
_ if paths::is_use_path_start(p) => path_pat(p),
_ if is_literal_pat_start(p) => literal_pat(p),
let m = match la0 {
T![_] => placeholder_pat(p), T![_] => placeholder_pat(p),
T![&] => ref_pat(p), T![&] => ref_pat(p),
T!['('] => tuple_pat(p), T!['('] => tuple_pat(p),
T!['['] => slice_pat(p), T!['['] => slice_pat(p),
_ => { _ => {
p.err_recover("expected pattern", recovery_set); p.err_recover("expected pattern", recovery_set);
return None; return None;
} }
}; };
Some(m) Some(m)
} }
fn is_literal_pat_start(p: &mut Parser) -> bool { fn is_literal_pat_start(p: &Parser) -> bool {
p.at(T![-]) && (p.nth(1) == INT_NUMBER || p.nth(1) == FLOAT_NUMBER) p.at(T![-]) && (p.nth(1) == INT_NUMBER || p.nth(1) == FLOAT_NUMBER)
|| p.at_ts(expressions::LITERAL_FIRST) || p.at_ts(expressions::LITERAL_FIRST)
} }
@ -165,6 +161,9 @@ fn record_field_pat_list(p: &mut Parser) {
T![..] => p.bump(), T![..] => p.bump(),
IDENT if p.nth(1) == T![:] => record_field_pat(p), IDENT if p.nth(1) == T![:] => record_field_pat(p),
T!['{'] => error_block(p, "expected ident"), T!['{'] => error_block(p, "expected ident"),
T![box] => {
box_pat(p);
}
_ => { _ => {
bind_pat(p, false); bind_pat(p, false);
} }
@ -261,11 +260,9 @@ fn pat_list(p: &mut Parser, ket: SyntaxKind) {
// let ref mut d = (); // let ref mut d = ();
// let e @ _ = (); // let e @ _ = ();
// let ref mut f @ g @ _ = (); // let ref mut f @ g @ _ = ();
// let box i = Box::new(1i32);
// } // }
fn bind_pat(p: &mut Parser, with_at: bool) -> CompletedMarker { fn bind_pat(p: &mut Parser, with_at: bool) -> CompletedMarker {
let m = p.start(); let m = p.start();
p.eat(T![box]);
p.eat(T![ref]); p.eat(T![ref]);
p.eat(T![mut]); p.eat(T![mut]);
name(p); name(p);
@ -274,3 +271,17 @@ fn bind_pat(p: &mut Parser, with_at: bool) -> CompletedMarker {
} }
m.complete(p, BIND_PAT) m.complete(p, BIND_PAT)
} }
// test box_pat
// fn main() {
// let box i = ();
// let box Outer { box i, j: box Inner(box &x) } = ();
// let box ref mut i = ();
// }
fn box_pat(p: &mut Parser) -> CompletedMarker {
assert!(p.at(T![box]));
let m = p.start();
p.bump();
pattern(p);
m.complete(p, BOX_PAT)
}

View file

@ -149,6 +149,7 @@ pub enum SyntaxKind {
IMPL_TRAIT_TYPE, IMPL_TRAIT_TYPE,
DYN_TRAIT_TYPE, DYN_TRAIT_TYPE,
REF_PAT, REF_PAT,
BOX_PAT,
BIND_PAT, BIND_PAT,
PLACEHOLDER_PAT, PLACEHOLDER_PAT,
PATH_PAT, PATH_PAT,

View file

@ -307,6 +307,33 @@ impl BlockExpr {
} }
} }
#[derive(Debug, Clone, PartialEq, Eq, Hash)] #[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct BoxPat {
pub(crate) syntax: SyntaxNode,
}
impl AstNode for BoxPat {
fn can_cast(kind: SyntaxKind) -> bool {
match kind {
BOX_PAT => true,
_ => false,
}
}
fn cast(syntax: SyntaxNode) -> Option<Self> {
if Self::can_cast(syntax.kind()) {
Some(Self { syntax })
} else {
None
}
}
fn syntax(&self) -> &SyntaxNode {
&self.syntax
}
}
impl BoxPat {
pub fn pat(&self) -> Option<Pat> {
AstChildren::new(&self.syntax).next()
}
}
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct BreakExpr { pub struct BreakExpr {
pub(crate) syntax: SyntaxNode, pub(crate) syntax: SyntaxNode,
} }
@ -2063,6 +2090,7 @@ impl ParenType {
#[derive(Debug, Clone, PartialEq, Eq, Hash)] #[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub enum Pat { pub enum Pat {
RefPat(RefPat), RefPat(RefPat),
BoxPat(BoxPat),
BindPat(BindPat), BindPat(BindPat),
PlaceholderPat(PlaceholderPat), PlaceholderPat(PlaceholderPat),
PathPat(PathPat), PathPat(PathPat),
@ -2078,6 +2106,11 @@ impl From<RefPat> for Pat {
Pat::RefPat(node) Pat::RefPat(node)
} }
} }
impl From<BoxPat> for Pat {
fn from(node: BoxPat) -> Pat {
Pat::BoxPat(node)
}
}
impl From<BindPat> for Pat { impl From<BindPat> for Pat {
fn from(node: BindPat) -> Pat { fn from(node: BindPat) -> Pat {
Pat::BindPat(node) Pat::BindPat(node)
@ -2126,14 +2159,15 @@ impl From<LiteralPat> for Pat {
impl AstNode for Pat { impl AstNode for Pat {
fn can_cast(kind: SyntaxKind) -> bool { fn can_cast(kind: SyntaxKind) -> bool {
match kind { match kind {
REF_PAT | BIND_PAT | PLACEHOLDER_PAT | PATH_PAT | RECORD_PAT | TUPLE_STRUCT_PAT REF_PAT | BOX_PAT | BIND_PAT | PLACEHOLDER_PAT | PATH_PAT | RECORD_PAT
| TUPLE_PAT | SLICE_PAT | RANGE_PAT | LITERAL_PAT => true, | TUPLE_STRUCT_PAT | TUPLE_PAT | SLICE_PAT | RANGE_PAT | LITERAL_PAT => true,
_ => false, _ => false,
} }
} }
fn cast(syntax: SyntaxNode) -> Option<Self> { fn cast(syntax: SyntaxNode) -> Option<Self> {
let res = match syntax.kind() { let res = match syntax.kind() {
REF_PAT => Pat::RefPat(RefPat { syntax }), REF_PAT => Pat::RefPat(RefPat { syntax }),
BOX_PAT => Pat::BoxPat(BoxPat { syntax }),
BIND_PAT => Pat::BindPat(BindPat { syntax }), BIND_PAT => Pat::BindPat(BindPat { syntax }),
PLACEHOLDER_PAT => Pat::PlaceholderPat(PlaceholderPat { syntax }), PLACEHOLDER_PAT => Pat::PlaceholderPat(PlaceholderPat { syntax }),
PATH_PAT => Pat::PathPat(PathPat { syntax }), PATH_PAT => Pat::PathPat(PathPat { syntax }),
@ -2150,6 +2184,7 @@ impl AstNode for Pat {
fn syntax(&self) -> &SyntaxNode { fn syntax(&self) -> &SyntaxNode {
match self { match self {
Pat::RefPat(it) => &it.syntax, Pat::RefPat(it) => &it.syntax,
Pat::BoxPat(it) => &it.syntax,
Pat::BindPat(it) => &it.syntax, Pat::BindPat(it) => &it.syntax,
Pat::PlaceholderPat(it) => &it.syntax, Pat::PlaceholderPat(it) => &it.syntax,
Pat::PathPat(it) => &it.syntax, Pat::PathPat(it) => &it.syntax,

View file

@ -158,6 +158,7 @@ Grammar(
"DYN_TRAIT_TYPE", "DYN_TRAIT_TYPE",
"REF_PAT", "REF_PAT",
"BOX_PAT",
"BIND_PAT", "BIND_PAT",
"PLACEHOLDER_PAT", "PLACEHOLDER_PAT",
"PATH_PAT", "PATH_PAT",
@ -523,6 +524,7 @@ Grammar(
), ),
"RefPat": ( options: [ "Pat" ]), "RefPat": ( options: [ "Pat" ]),
"BoxPat": ( options: [ "Pat" ]),
"BindPat": ( "BindPat": (
options: [ "Pat" ], options: [ "Pat" ],
traits: ["NameOwner"] traits: ["NameOwner"]
@ -552,6 +554,7 @@ Grammar(
"Pat": ( "Pat": (
enum: [ enum: [
"RefPat", "RefPat",
"BoxPat",
"BindPat", "BindPat",
"PlaceholderPat", "PlaceholderPat",
"PathPat", "PathPat",

View file

@ -0,0 +1,6 @@
fn main() {
let ref box i = ();
let mut box i = ();
let ref mut box i = ();
}

View file

@ -0,0 +1,95 @@
SOURCE_FILE@[0; 91)
FN_DEF@[0; 89)
FN_KW@[0; 2) "fn"
WHITESPACE@[2; 3) " "
NAME@[3; 7)
IDENT@[3; 7) "main"
PARAM_LIST@[7; 9)
L_PAREN@[7; 8) "("
R_PAREN@[8; 9) ")"
WHITESPACE@[9; 10) " "
BLOCK@[10; 89)
L_CURLY@[10; 11) "{"
WHITESPACE@[11; 16) "\n "
LET_STMT@[16; 27)
LET_KW@[16; 19) "let"
WHITESPACE@[19; 20) " "
BIND_PAT@[20; 27)
REF_KW@[20; 23) "ref"
WHITESPACE@[23; 24) " "
ERROR@[24; 27)
BOX_KW@[24; 27) "box"
WHITESPACE@[27; 28) " "
EXPR_STMT@[28; 35)
BIN_EXPR@[28; 34)
PATH_EXPR@[28; 29)
PATH@[28; 29)
PATH_SEGMENT@[28; 29)
NAME_REF@[28; 29)
IDENT@[28; 29) "i"
WHITESPACE@[29; 30) " "
EQ@[30; 31) "="
WHITESPACE@[31; 32) " "
TUPLE_EXPR@[32; 34)
L_PAREN@[32; 33) "("
R_PAREN@[33; 34) ")"
SEMI@[34; 35) ";"
WHITESPACE@[35; 40) "\n "
LET_STMT@[40; 51)
LET_KW@[40; 43) "let"
WHITESPACE@[43; 44) " "
BIND_PAT@[44; 51)
MUT_KW@[44; 47) "mut"
WHITESPACE@[47; 48) " "
ERROR@[48; 51)
BOX_KW@[48; 51) "box"
WHITESPACE@[51; 52) " "
EXPR_STMT@[52; 59)
BIN_EXPR@[52; 58)
PATH_EXPR@[52; 53)
PATH@[52; 53)
PATH_SEGMENT@[52; 53)
NAME_REF@[52; 53)
IDENT@[52; 53) "i"
WHITESPACE@[53; 54) " "
EQ@[54; 55) "="
WHITESPACE@[55; 56) " "
TUPLE_EXPR@[56; 58)
L_PAREN@[56; 57) "("
R_PAREN@[57; 58) ")"
SEMI@[58; 59) ";"
WHITESPACE@[59; 64) "\n "
LET_STMT@[64; 79)
LET_KW@[64; 67) "let"
WHITESPACE@[67; 68) " "
BIND_PAT@[68; 79)
REF_KW@[68; 71) "ref"
WHITESPACE@[71; 72) " "
MUT_KW@[72; 75) "mut"
WHITESPACE@[75; 76) " "
ERROR@[76; 79)
BOX_KW@[76; 79) "box"
WHITESPACE@[79; 80) " "
EXPR_STMT@[80; 87)
BIN_EXPR@[80; 86)
PATH_EXPR@[80; 81)
PATH@[80; 81)
PATH_SEGMENT@[80; 81)
NAME_REF@[80; 81)
IDENT@[80; 81) "i"
WHITESPACE@[81; 82) " "
EQ@[82; 83) "="
WHITESPACE@[83; 84) " "
TUPLE_EXPR@[84; 86)
L_PAREN@[84; 85) "("
R_PAREN@[85; 86) ")"
SEMI@[86; 87) ";"
WHITESPACE@[87; 88) "\n"
R_CURLY@[88; 89) "}"
WHITESPACE@[89; 91) "\n\n"
error 24: expected a name
error 27: expected SEMI
error 48: expected a name
error 51: expected SEMI
error 76: expected a name
error 79: expected SEMI

View file

@ -5,7 +5,5 @@ fn foo() {
X | Y if Z => (), X | Y if Z => (),
| X | Y if Z => (), | X | Y if Z => (),
| X => (), | X => (),
box X => (),
Some(box X) => (),
}; };
} }

View file

@ -1,5 +1,5 @@
SOURCE_FILE@[0; 215) SOURCE_FILE@[0; 167)
FN_DEF@[0; 214) FN_DEF@[0; 166)
FN_KW@[0; 2) "fn" FN_KW@[0; 2) "fn"
WHITESPACE@[2; 3) " " WHITESPACE@[2; 3) " "
NAME@[3; 6) NAME@[3; 6)
@ -8,18 +8,18 @@ SOURCE_FILE@[0; 215)
L_PAREN@[6; 7) "(" L_PAREN@[6; 7) "("
R_PAREN@[7; 8) ")" R_PAREN@[7; 8) ")"
WHITESPACE@[8; 9) " " WHITESPACE@[8; 9) " "
BLOCK@[9; 214) BLOCK@[9; 166)
L_CURLY@[9; 10) "{" L_CURLY@[9; 10) "{"
WHITESPACE@[10; 15) "\n " WHITESPACE@[10; 15) "\n "
EXPR_STMT@[15; 212) EXPR_STMT@[15; 164)
MATCH_EXPR@[15; 211) MATCH_EXPR@[15; 163)
MATCH_KW@[15; 20) "match" MATCH_KW@[15; 20) "match"
WHITESPACE@[20; 21) " " WHITESPACE@[20; 21) " "
TUPLE_EXPR@[21; 23) TUPLE_EXPR@[21; 23)
L_PAREN@[21; 22) "(" L_PAREN@[21; 22) "("
R_PAREN@[22; 23) ")" R_PAREN@[22; 23) ")"
WHITESPACE@[23; 24) " " WHITESPACE@[23; 24) " "
MATCH_ARM_LIST@[24; 211) MATCH_ARM_LIST@[24; 163)
L_CURLY@[24; 25) "{" L_CURLY@[24; 25) "{"
WHITESPACE@[25; 34) "\n " WHITESPACE@[25; 34) "\n "
MATCH_ARM@[34; 41) MATCH_ARM@[34; 41)
@ -141,44 +141,9 @@ SOURCE_FILE@[0; 215)
L_PAREN@[154; 155) "(" L_PAREN@[154; 155) "("
R_PAREN@[155; 156) ")" R_PAREN@[155; 156) ")"
COMMA@[156; 157) "," COMMA@[156; 157) ","
WHITESPACE@[157; 166) "\n " WHITESPACE@[157; 162) "\n "
MATCH_ARM@[166; 177) R_CURLY@[162; 163) "}"
BIND_PAT@[166; 171) SEMI@[163; 164) ";"
BOX_KW@[166; 169) "box" WHITESPACE@[164; 165) "\n"
WHITESPACE@[169; 170) " " R_CURLY@[165; 166) "}"
NAME@[170; 171) WHITESPACE@[166; 167) "\n"
IDENT@[170; 171) "X"
WHITESPACE@[171; 172) " "
FAT_ARROW@[172; 174) "=>"
WHITESPACE@[174; 175) " "
TUPLE_EXPR@[175; 177)
L_PAREN@[175; 176) "("
R_PAREN@[176; 177) ")"
COMMA@[177; 178) ","
WHITESPACE@[178; 187) "\n "
MATCH_ARM@[187; 204)
TUPLE_STRUCT_PAT@[187; 198)
PATH@[187; 191)
PATH_SEGMENT@[187; 191)
NAME_REF@[187; 191)
IDENT@[187; 191) "Some"
L_PAREN@[191; 192) "("
BIND_PAT@[192; 197)
BOX_KW@[192; 195) "box"
WHITESPACE@[195; 196) " "
NAME@[196; 197)
IDENT@[196; 197) "X"
R_PAREN@[197; 198) ")"
WHITESPACE@[198; 199) " "
FAT_ARROW@[199; 201) "=>"
WHITESPACE@[201; 202) " "
TUPLE_EXPR@[202; 204)
L_PAREN@[202; 203) "("
R_PAREN@[203; 204) ")"
COMMA@[204; 205) ","
WHITESPACE@[205; 210) "\n "
R_CURLY@[210; 211) "}"
SEMI@[211; 212) ";"
WHITESPACE@[212; 213) "\n"
R_CURLY@[213; 214) "}"
WHITESPACE@[214; 215) "\n"

View file

@ -5,5 +5,4 @@ fn main() {
let ref mut d = (); let ref mut d = ();
let e @ _ = (); let e @ _ = ();
let ref mut f @ g @ _ = (); let ref mut f @ g @ _ = ();
let box i = Box::new(1i32);
} }

View file

@ -1,5 +1,5 @@
SOURCE_FILE@[0; 178) SOURCE_FILE@[0; 146)
FN_DEF@[0; 177) FN_DEF@[0; 145)
FN_KW@[0; 2) "fn" FN_KW@[0; 2) "fn"
WHITESPACE@[2; 3) " " WHITESPACE@[2; 3) " "
NAME@[3; 7) NAME@[3; 7)
@ -8,7 +8,7 @@ SOURCE_FILE@[0; 178)
L_PAREN@[7; 8) "(" L_PAREN@[7; 8) "("
R_PAREN@[8; 9) ")" R_PAREN@[8; 9) ")"
WHITESPACE@[9; 10) " " WHITESPACE@[9; 10) " "
BLOCK@[10; 177) BLOCK@[10; 145)
L_CURLY@[10; 11) "{" L_CURLY@[10; 11) "{"
WHITESPACE@[11; 16) "\n " WHITESPACE@[11; 16) "\n "
LET_STMT@[16; 27) LET_STMT@[16; 27)
@ -122,35 +122,6 @@ SOURCE_FILE@[0; 178)
L_PAREN@[140; 141) "(" L_PAREN@[140; 141) "("
R_PAREN@[141; 142) ")" R_PAREN@[141; 142) ")"
SEMI@[142; 143) ";" SEMI@[142; 143) ";"
WHITESPACE@[143; 148) "\n " WHITESPACE@[143; 144) "\n"
LET_STMT@[148; 175) R_CURLY@[144; 145) "}"
LET_KW@[148; 151) "let" WHITESPACE@[145; 146) "\n"
WHITESPACE@[151; 152) " "
BIND_PAT@[152; 157)
BOX_KW@[152; 155) "box"
WHITESPACE@[155; 156) " "
NAME@[156; 157)
IDENT@[156; 157) "i"
WHITESPACE@[157; 158) " "
EQ@[158; 159) "="
WHITESPACE@[159; 160) " "
CALL_EXPR@[160; 174)
PATH_EXPR@[160; 168)
PATH@[160; 168)
PATH@[160; 163)
PATH_SEGMENT@[160; 163)
NAME_REF@[160; 163)
IDENT@[160; 163) "Box"
COLONCOLON@[163; 165) "::"
PATH_SEGMENT@[165; 168)
NAME_REF@[165; 168)
IDENT@[165; 168) "new"
ARG_LIST@[168; 174)
L_PAREN@[168; 169) "("
LITERAL@[169; 173)
INT_NUMBER@[169; 173) "1i32"
R_PAREN@[173; 174) ")"
SEMI@[174; 175) ";"
WHITESPACE@[175; 176) "\n"
R_CURLY@[176; 177) "}"
WHITESPACE@[177; 178) "\n"

View file

@ -0,0 +1,5 @@
fn main() {
let box i = ();
let box Outer { box i, j: box Inner(box &x) } = ();
let box ref mut i = ();
}

View file

@ -0,0 +1,109 @@
SOURCE_FILE@[0; 118)
FN_DEF@[0; 117)
FN_KW@[0; 2) "fn"
WHITESPACE@[2; 3) " "
NAME@[3; 7)
IDENT@[3; 7) "main"
PARAM_LIST@[7; 9)
L_PAREN@[7; 8) "("
R_PAREN@[8; 9) ")"
WHITESPACE@[9; 10) " "
BLOCK@[10; 117)
L_CURLY@[10; 11) "{"
WHITESPACE@[11; 16) "\n "
LET_STMT@[16; 31)
LET_KW@[16; 19) "let"
WHITESPACE@[19; 20) " "
BOX_PAT@[20; 25)
BOX_KW@[20; 23) "box"
WHITESPACE@[23; 24) " "
BIND_PAT@[24; 25)
NAME@[24; 25)
IDENT@[24; 25) "i"
WHITESPACE@[25; 26) " "
EQ@[26; 27) "="
WHITESPACE@[27; 28) " "
TUPLE_EXPR@[28; 30)
L_PAREN@[28; 29) "("
R_PAREN@[29; 30) ")"
SEMI@[30; 31) ";"
WHITESPACE@[31; 36) "\n "
LET_STMT@[36; 87)
LET_KW@[36; 39) "let"
WHITESPACE@[39; 40) " "
BOX_PAT@[40; 81)
BOX_KW@[40; 43) "box"
WHITESPACE@[43; 44) " "
RECORD_PAT@[44; 81)
PATH@[44; 49)
PATH_SEGMENT@[44; 49)
NAME_REF@[44; 49)
IDENT@[44; 49) "Outer"
WHITESPACE@[49; 50) " "
RECORD_FIELD_PAT_LIST@[50; 81)
L_CURLY@[50; 51) "{"
WHITESPACE@[51; 52) " "
BOX_PAT@[52; 57)
BOX_KW@[52; 55) "box"
WHITESPACE@[55; 56) " "
BIND_PAT@[56; 57)
NAME@[56; 57)
IDENT@[56; 57) "i"
COMMA@[57; 58) ","
WHITESPACE@[58; 59) " "
RECORD_FIELD_PAT@[59; 79)
NAME@[59; 60)
IDENT@[59; 60) "j"
COLON@[60; 61) ":"
WHITESPACE@[61; 62) " "
BOX_PAT@[62; 79)
BOX_KW@[62; 65) "box"
WHITESPACE@[65; 66) " "
TUPLE_STRUCT_PAT@[66; 79)
PATH@[66; 71)
PATH_SEGMENT@[66; 71)
NAME_REF@[66; 71)
IDENT@[66; 71) "Inner"
L_PAREN@[71; 72) "("
BOX_PAT@[72; 78)
BOX_KW@[72; 75) "box"
WHITESPACE@[75; 76) " "
REF_PAT@[76; 78)
AMP@[76; 77) "&"
BIND_PAT@[77; 78)
NAME@[77; 78)
IDENT@[77; 78) "x"
R_PAREN@[78; 79) ")"
WHITESPACE@[79; 80) " "
R_CURLY@[80; 81) "}"
WHITESPACE@[81; 82) " "
EQ@[82; 83) "="
WHITESPACE@[83; 84) " "
TUPLE_EXPR@[84; 86)
L_PAREN@[84; 85) "("
R_PAREN@[85; 86) ")"
SEMI@[86; 87) ";"
WHITESPACE@[87; 92) "\n "
LET_STMT@[92; 115)
LET_KW@[92; 95) "let"
WHITESPACE@[95; 96) " "
BOX_PAT@[96; 109)
BOX_KW@[96; 99) "box"
WHITESPACE@[99; 100) " "
BIND_PAT@[100; 109)
REF_KW@[100; 103) "ref"
WHITESPACE@[103; 104) " "
MUT_KW@[104; 107) "mut"
WHITESPACE@[107; 108) " "
NAME@[108; 109)
IDENT@[108; 109) "i"
WHITESPACE@[109; 110) " "
EQ@[110; 111) "="
WHITESPACE@[111; 112) " "
TUPLE_EXPR@[112; 114)
L_PAREN@[112; 113) "("
R_PAREN@[113; 114) ")"
SEMI@[114; 115) ";"
WHITESPACE@[115; 116) "\n"
R_CURLY@[116; 117) "}"
WHITESPACE@[117; 118) "\n"