Unnamed repository; edit this file 'description' to name the repository.
Parse for<'a> closure syntax
Lukas Wirth 2022-04-10
parent e691ae0 · commit d8341c5
-rw-r--r--crates/parser/src/grammar/expressions/atom.rs80
-rw-r--r--crates/parser/test_data/parser/err/0024_many_type_parens.rast126
-rw-r--r--crates/parser/test_data/parser/inline/ok/0106_lambda_expr.rast42
-rw-r--r--crates/parser/test_data/parser/inline/ok/0106_lambda_expr.rs2
-rw-r--r--crates/parser/test_data/parser/inline/ok/0157_const_block.rast24
-rw-r--r--crates/parser/test_data/parser/inline/ok/0157_const_block.rs1
-rw-r--r--crates/parser/test_data/parser/inline/ok/0162_unsafe_block.rast24
-rw-r--r--crates/parser/test_data/parser/inline/ok/0162_unsafe_block.rs1
-rw-r--r--crates/parser/test_data/parser/inline/ok/0199_effect_blocks.rast95
-rw-r--r--crates/parser/test_data/parser/inline/ok/0199_effect_blocks.rs4
-rw-r--r--crates/syntax/rust.ungram2
-rw-r--r--crates/syntax/src/ast/generated/nodes.rs2
12 files changed, 245 insertions, 158 deletions
diff --git a/crates/parser/src/grammar/expressions/atom.rs b/crates/parser/src/grammar/expressions/atom.rs
index d137a02ee8..10e5d897e0 100644
--- a/crates/parser/src/grammar/expressions/atom.rs
+++ b/crates/parser/src/grammar/expressions/atom.rs
@@ -71,16 +71,8 @@ pub(super) fn atom_expr(p: &mut Parser, r: Restrictions) -> Option<(CompletedMar
let done = match p.current() {
T!['('] => tuple_expr(p),
T!['['] => array_expr(p),
- T![|] => closure_expr(p),
- T![static] | T![async] | T![move] if la == T![|] => closure_expr(p),
- T![static] | T![async] if la == T![move] && p.nth(2) == T![|] => closure_expr(p),
- T![static] if la == T![async] && p.nth(2) == T![|] => closure_expr(p),
- T![static] if la == T![async] && p.nth(2) == T![move] && p.nth(3) == T![|] => {
- closure_expr(p)
- }
T![if] => if_expr(p),
T![let] => let_expr(p),
-
T![_] => {
// test destructuring_assignment_wildcard_pat
// fn foo() {
@@ -91,12 +83,16 @@ pub(super) fn atom_expr(p: &mut Parser, r: Restrictions) -> Option<(CompletedMar
p.bump(T![_]);
m.complete(p, UNDERSCORE_EXPR)
}
-
T![loop] => loop_expr(p, None),
T![box] => box_expr(p, None),
- T![for] => for_expr(p, None),
T![while] => while_expr(p, None),
T![try] => try_block_expr(p, None),
+ T![match] => match_expr(p),
+ T![return] => return_expr(p),
+ T![yield] => yield_expr(p),
+ T![continue] => continue_expr(p),
+ T![break] => break_expr(p, r),
+
LIFETIME_IDENT if la == T![:] => {
let m = p.start();
label(p);
@@ -121,27 +117,21 @@ pub(super) fn atom_expr(p: &mut Parser, r: Restrictions) -> Option<(CompletedMar
}
}
}
- T![async] if la == T!['{'] || (la == T![move] && p.nth(2) == T!['{']) => {
- let m = p.start();
- p.bump(T![async]);
- p.eat(T![move]);
- stmt_list(p);
- m.complete(p, BLOCK_EXPR)
- }
- T![match] => match_expr(p),
- // test unsafe_block
+ // test effect_blocks
// fn f() { unsafe { } }
- T![unsafe] if la == T!['{'] => {
+ // fn f() { const { } }
+ // fn f() { async { } }
+ // fn f() { async move { } }
+ T![const] | T![unsafe] | T![async] if la == T!['{'] => {
let m = p.start();
- p.bump(T![unsafe]);
+ p.bump_any();
stmt_list(p);
m.complete(p, BLOCK_EXPR)
}
- // test const_block
- // fn f() { const { } }
- T![const] if la == T!['{'] => {
+ T![async] if la == T![move] && p.nth(2) == T!['{'] => {
let m = p.start();
- p.bump(T![const]);
+ p.bump(T![async]);
+ p.eat(T![move]);
stmt_list(p);
m.complete(p, BLOCK_EXPR)
}
@@ -156,10 +146,11 @@ pub(super) fn atom_expr(p: &mut Parser, r: Restrictions) -> Option<(CompletedMar
stmt_list(p);
m.complete(p, BLOCK_EXPR)
}
- T![return] => return_expr(p),
- T![yield] => yield_expr(p),
- T![continue] => continue_expr(p),
- T![break] => break_expr(p, r),
+
+ T![static] | T![async] | T![move] | T![|] => closure_expr(p),
+ T![for] if la == T![<] => closure_expr(p),
+ T![for] => for_expr(p, None),
+
_ => {
p.err_recover("expected expression", EXPR_RECOVERY_SET);
return None;
@@ -254,25 +245,30 @@ fn array_expr(p: &mut Parser) -> CompletedMarker {
// static move || {};
// static async || {};
// static async move || {};
+// for<'a> || {};
+// for<'a> move || {};
// }
fn closure_expr(p: &mut Parser) -> CompletedMarker {
- assert!(
- p.at(T![|])
- || (p.at(T![move]) && p.nth(1) == T![|])
- || (p.at(T![async]) && p.nth(1) == T![|])
- || (p.at(T![async]) && p.nth(1) == T![move] && p.nth(2) == T![|])
- || (p.at(T![static]) && p.nth(1) == T![|])
- || (p.at(T![static]) && p.nth(1) == T![move] && p.nth(2) == T![|])
- || (p.at(T![static]) && p.nth(1) == T![async] && p.nth(2) == T![|])
- || (p.at(T![static])
- && p.nth(1) == T![async]
- && p.nth(2) == T![move]
- && p.nth(3) == T![|])
- );
+ assert!(match p.current() {
+ T![static] | T![async] | T![move] | T![|] => true,
+ T![for] => p.nth(1) == T![<],
+ _ => false,
+ });
+
let m = p.start();
+
+ if p.at(T![for]) {
+ types::for_binder(p);
+ }
+
p.eat(T![static]);
p.eat(T![async]);
p.eat(T![move]);
+
+ if !p.at(T![|]) {
+ p.error("expected `|`");
+ return m.complete(p, CLOSURE_EXPR);
+ }
params::param_list_closure(p);
if opt_ret_type(p) {
// test lambda_ret_block
diff --git a/crates/parser/test_data/parser/err/0024_many_type_parens.rast b/crates/parser/test_data/parser/err/0024_many_type_parens.rast
index 82e6a11249..d374f86610 100644
--- a/crates/parser/test_data/parser/err/0024_many_type_parens.rast
+++ b/crates/parser/test_data/parser/err/0024_many_type_parens.rast
@@ -182,45 +182,44 @@ SOURCE_FILE
WHITESPACE " "
TUPLE_EXPR
L_PAREN "("
- FOR_EXPR
+ CLOSURE_EXPR
FOR_KW "for"
- PATH_PAT
- PATH
- PATH_SEGMENT
- L_ANGLE "<"
- ERROR
- LIFETIME_IDENT "'a"
- R_ANGLE ">"
- WHITESPACE " "
+ GENERIC_PARAM_LIST
+ L_ANGLE "<"
+ LIFETIME_PARAM
+ LIFETIME
+ LIFETIME_IDENT "'a"
+ R_ANGLE ">"
+ WHITESPACE " "
+ BIN_EXPR
BIN_EXPR
BIN_EXPR
BIN_EXPR
- BIN_EXPR
- PATH_EXPR
- PATH
- PATH_SEGMENT
- NAME_REF
- IDENT "Trait"
- L_ANGLE "<"
- ERROR
- LIFETIME_IDENT "'a"
- R_ANGLE ">"
- ERROR
- R_PAREN ")"
- WHITESPACE " "
- PLUS "+"
- WHITESPACE " "
- PAREN_EXPR
- L_PAREN "("
PATH_EXPR
PATH
PATH_SEGMENT
NAME_REF
- IDENT "Copy"
+ IDENT "Trait"
+ L_ANGLE "<"
+ ERROR
+ LIFETIME_IDENT "'a"
+ R_ANGLE ">"
+ ERROR
R_PAREN ")"
- R_ANGLE ">"
- ERROR
- SEMICOLON ";"
+ WHITESPACE " "
+ PLUS "+"
+ WHITESPACE " "
+ PAREN_EXPR
+ L_PAREN "("
+ PATH_EXPR
+ PATH
+ PATH_SEGMENT
+ NAME_REF
+ IDENT "Copy"
+ R_PAREN ")"
+ R_ANGLE ">"
+ ERROR
+ SEMICOLON ";"
WHITESPACE "\n "
LET_EXPR
LET_KW "let"
@@ -240,49 +239,48 @@ SOURCE_FILE
L_ANGLE "<"
TUPLE_EXPR
L_PAREN "("
- FOR_EXPR
+ CLOSURE_EXPR
FOR_KW "for"
- PATH_PAT
- PATH
- PATH_SEGMENT
- L_ANGLE "<"
- ERROR
- LIFETIME_IDENT "'a"
- R_ANGLE ">"
- WHITESPACE " "
+ GENERIC_PARAM_LIST
+ L_ANGLE "<"
+ LIFETIME_PARAM
+ LIFETIME
+ LIFETIME_IDENT "'a"
+ R_ANGLE ">"
+ WHITESPACE " "
+ BIN_EXPR
BIN_EXPR
BIN_EXPR
BIN_EXPR
- BIN_EXPR
- PATH_EXPR
- PATH
- PATH_SEGMENT
- NAME_REF
- IDENT "Trait"
- L_ANGLE "<"
- ERROR
- LIFETIME_IDENT "'a"
- R_ANGLE ">"
- ERROR
- R_PAREN ")"
- WHITESPACE " "
- PLUS "+"
- WHITESPACE " "
- PAREN_EXPR
- L_PAREN "("
PATH_EXPR
PATH
PATH_SEGMENT
NAME_REF
- IDENT "Copy"
+ IDENT "Trait"
+ L_ANGLE "<"
+ ERROR
+ LIFETIME_IDENT "'a"
+ R_ANGLE ">"
+ ERROR
R_PAREN ")"
WHITESPACE " "
PLUS "+"
WHITESPACE " "
PAREN_EXPR
L_PAREN "("
- ERROR
- QUESTION "?"
+ PATH_EXPR
+ PATH
+ PATH_SEGMENT
+ NAME_REF
+ IDENT "Copy"
+ R_PAREN ")"
+ WHITESPACE " "
+ PLUS "+"
+ WHITESPACE " "
+ PAREN_EXPR
+ L_PAREN "("
+ ERROR
+ QUESTION "?"
PATH_EXPR
PATH
PATH_SEGMENT
@@ -307,23 +305,21 @@ error 141: expected SEMICOLON
error 146: expected SEMICOLON
error 146: expected expression
error 148: expected expression
-error 155: expected type
-error 158: expected IN_KW
+error 158: expected `|`
+error 158: expected COMMA
error 165: expected expression
error 168: expected expression
error 179: expected expression
-error 180: expected a block
error 180: expected COMMA
error 190: expected EQ
error 190: expected expression
error 191: expected COMMA
-error 201: expected type
-error 204: expected IN_KW
+error 204: expected `|`
+error 204: expected COMMA
error 211: expected expression
error 214: expected expression
error 228: expected expression
error 229: expected R_PAREN
-error 229: expected a block
error 229: expected COMMA
error 236: expected expression
error 237: expected COMMA
diff --git a/crates/parser/test_data/parser/inline/ok/0106_lambda_expr.rast b/crates/parser/test_data/parser/inline/ok/0106_lambda_expr.rast
index bc54b01869..c25ad8430d 100644
--- a/crates/parser/test_data/parser/inline/ok/0106_lambda_expr.rast
+++ b/crates/parser/test_data/parser/inline/ok/0106_lambda_expr.rast
@@ -199,6 +199,48 @@ SOURCE_FILE
L_CURLY "{"
R_CURLY "}"
SEMICOLON ";"
+ WHITESPACE "\n "
+ EXPR_STMT
+ CLOSURE_EXPR
+ FOR_KW "for"
+ GENERIC_PARAM_LIST
+ L_ANGLE "<"
+ LIFETIME_PARAM
+ LIFETIME
+ LIFETIME_IDENT "'a"
+ R_ANGLE ">"
+ WHITESPACE " "
+ PARAM_LIST
+ PIPE "|"
+ PIPE "|"
+ WHITESPACE " "
+ BLOCK_EXPR
+ STMT_LIST
+ L_CURLY "{"
+ R_CURLY "}"
+ SEMICOLON ";"
+ WHITESPACE "\n "
+ EXPR_STMT
+ CLOSURE_EXPR
+ FOR_KW "for"
+ GENERIC_PARAM_LIST
+ L_ANGLE "<"
+ LIFETIME_PARAM
+ LIFETIME
+ LIFETIME_IDENT "'a"
+ R_ANGLE ">"
+ WHITESPACE " "
+ MOVE_KW "move"
+ WHITESPACE " "
+ PARAM_LIST
+ PIPE "|"
+ PIPE "|"
+ WHITESPACE " "
+ BLOCK_EXPR
+ STMT_LIST
+ L_CURLY "{"
+ R_CURLY "}"
+ SEMICOLON ";"
WHITESPACE "\n"
R_CURLY "}"
WHITESPACE "\n"
diff --git a/crates/parser/test_data/parser/inline/ok/0106_lambda_expr.rs b/crates/parser/test_data/parser/inline/ok/0106_lambda_expr.rs
index d01f921664..75516d2584 100644
--- a/crates/parser/test_data/parser/inline/ok/0106_lambda_expr.rs
+++ b/crates/parser/test_data/parser/inline/ok/0106_lambda_expr.rs
@@ -10,4 +10,6 @@ fn foo() {
static move || {};
static async || {};
static async move || {};
+ for<'a> || {};
+ for<'a> move || {};
}
diff --git a/crates/parser/test_data/parser/inline/ok/0157_const_block.rast b/crates/parser/test_data/parser/inline/ok/0157_const_block.rast
deleted file mode 100644
index d2025b895d..0000000000
--- a/crates/parser/test_data/parser/inline/ok/0157_const_block.rast
+++ /dev/null
@@ -1,24 +0,0 @@
-SOURCE_FILE
- FN
- FN_KW "fn"
- WHITESPACE " "
- NAME
- IDENT "f"
- PARAM_LIST
- L_PAREN "("
- R_PAREN ")"
- WHITESPACE " "
- BLOCK_EXPR
- STMT_LIST
- L_CURLY "{"
- WHITESPACE " "
- BLOCK_EXPR
- CONST_KW "const"
- WHITESPACE " "
- STMT_LIST
- L_CURLY "{"
- WHITESPACE " "
- R_CURLY "}"
- WHITESPACE " "
- R_CURLY "}"
- WHITESPACE "\n"
diff --git a/crates/parser/test_data/parser/inline/ok/0157_const_block.rs b/crates/parser/test_data/parser/inline/ok/0157_const_block.rs
deleted file mode 100644
index a2e3565a32..0000000000
--- a/crates/parser/test_data/parser/inline/ok/0157_const_block.rs
+++ /dev/null
@@ -1 +0,0 @@
-fn f() { const { } }
diff --git a/crates/parser/test_data/parser/inline/ok/0162_unsafe_block.rast b/crates/parser/test_data/parser/inline/ok/0162_unsafe_block.rast
deleted file mode 100644
index 2149676ded..0000000000
--- a/crates/parser/test_data/parser/inline/ok/0162_unsafe_block.rast
+++ /dev/null
@@ -1,24 +0,0 @@
-SOURCE_FILE
- FN
- FN_KW "fn"
- WHITESPACE " "
- NAME
- IDENT "f"
- PARAM_LIST
- L_PAREN "("
- R_PAREN ")"
- WHITESPACE " "
- BLOCK_EXPR
- STMT_LIST
- L_CURLY "{"
- WHITESPACE " "
- BLOCK_EXPR
- UNSAFE_KW "unsafe"
- WHITESPACE " "
- STMT_LIST
- L_CURLY "{"
- WHITESPACE " "
- R_CURLY "}"
- WHITESPACE " "
- R_CURLY "}"
- WHITESPACE "\n"
diff --git a/crates/parser/test_data/parser/inline/ok/0162_unsafe_block.rs b/crates/parser/test_data/parser/inline/ok/0162_unsafe_block.rs
deleted file mode 100644
index dfb9b2a1c1..0000000000
--- a/crates/parser/test_data/parser/inline/ok/0162_unsafe_block.rs
+++ /dev/null
@@ -1 +0,0 @@
-fn f() { unsafe { } }
diff --git a/crates/parser/test_data/parser/inline/ok/0199_effect_blocks.rast b/crates/parser/test_data/parser/inline/ok/0199_effect_blocks.rast
new file mode 100644
index 0000000000..f14080c90e
--- /dev/null
+++ b/crates/parser/test_data/parser/inline/ok/0199_effect_blocks.rast
@@ -0,0 +1,95 @@
+SOURCE_FILE
+ FN
+ FN_KW "fn"
+ WHITESPACE " "
+ NAME
+ IDENT "f"
+ PARAM_LIST
+ L_PAREN "("
+ R_PAREN ")"
+ WHITESPACE " "
+ BLOCK_EXPR
+ STMT_LIST
+ L_CURLY "{"
+ WHITESPACE " "
+ BLOCK_EXPR
+ UNSAFE_KW "unsafe"
+ WHITESPACE " "
+ STMT_LIST
+ L_CURLY "{"
+ WHITESPACE " "
+ R_CURLY "}"
+ WHITESPACE " "
+ R_CURLY "}"
+ WHITESPACE "\n"
+ FN
+ FN_KW "fn"
+ WHITESPACE " "
+ NAME
+ IDENT "f"
+ PARAM_LIST
+ L_PAREN "("
+ R_PAREN ")"
+ WHITESPACE " "
+ BLOCK_EXPR
+ STMT_LIST
+ L_CURLY "{"
+ WHITESPACE " "
+ BLOCK_EXPR
+ CONST_KW "const"
+ WHITESPACE " "
+ STMT_LIST
+ L_CURLY "{"
+ WHITESPACE " "
+ R_CURLY "}"
+ WHITESPACE " "
+ R_CURLY "}"
+ WHITESPACE "\n"
+ FN
+ FN_KW "fn"
+ WHITESPACE " "
+ NAME
+ IDENT "f"
+ PARAM_LIST
+ L_PAREN "("
+ R_PAREN ")"
+ WHITESPACE " "
+ BLOCK_EXPR
+ STMT_LIST
+ L_CURLY "{"
+ WHITESPACE " "
+ BLOCK_EXPR
+ ASYNC_KW "async"
+ WHITESPACE " "
+ STMT_LIST
+ L_CURLY "{"
+ WHITESPACE " "
+ R_CURLY "}"
+ WHITESPACE " "
+ R_CURLY "}"
+ WHITESPACE "\n"
+ FN
+ FN_KW "fn"
+ WHITESPACE " "
+ NAME
+ IDENT "f"
+ PARAM_LIST
+ L_PAREN "("
+ R_PAREN ")"
+ WHITESPACE " "
+ BLOCK_EXPR
+ STMT_LIST
+ L_CURLY "{"
+ WHITESPACE " "
+ BLOCK_EXPR
+ ASYNC_KW "async"
+ WHITESPACE " "
+ MOVE_KW "move"
+ WHITESPACE " "
+ STMT_LIST
+ L_CURLY "{"
+ WHITESPACE " "
+ R_CURLY "}"
+ WHITESPACE " "
+ R_CURLY "}"
+ WHITESPACE "\n"
diff --git a/crates/parser/test_data/parser/inline/ok/0199_effect_blocks.rs b/crates/parser/test_data/parser/inline/ok/0199_effect_blocks.rs
new file mode 100644
index 0000000000..c57d24b2f7
--- /dev/null
+++ b/crates/parser/test_data/parser/inline/ok/0199_effect_blocks.rs
@@ -0,0 +1,4 @@
+fn f() { unsafe { } }
+fn f() { const { } }
+fn f() { async { } }
+fn f() { async move { } }
diff --git a/crates/syntax/rust.ungram b/crates/syntax/rust.ungram
index 4e8d4b478b..19e642968c 100644
--- a/crates/syntax/rust.ungram
+++ b/crates/syntax/rust.ungram
@@ -449,7 +449,7 @@ FieldExpr =
Attr* Expr '.' NameRef
ClosureExpr =
- Attr* 'static'? 'async'? 'move'? ParamList RetType?
+ Attr* ('for' GenericParamList)? 'static'? 'async'? 'move'? ParamList RetType?
body:Expr
IfExpr =
diff --git a/crates/syntax/src/ast/generated/nodes.rs b/crates/syntax/src/ast/generated/nodes.rs
index f51282fff4..92d793e0fd 100644
--- a/crates/syntax/src/ast/generated/nodes.rs
+++ b/crates/syntax/src/ast/generated/nodes.rs
@@ -837,6 +837,8 @@ pub struct ClosureExpr {
}
impl ast::HasAttrs for ClosureExpr {}
impl ClosureExpr {
+ pub fn for_token(&self) -> Option<SyntaxToken> { support::token(&self.syntax, T![for]) }
+ pub fn generic_param_list(&self) -> Option<GenericParamList> { support::child(&self.syntax) }
pub fn static_token(&self) -> Option<SyntaxToken> { support::token(&self.syntax, T![static]) }
pub fn async_token(&self) -> Option<SyntaxToken> { support::token(&self.syntax, T![async]) }
pub fn move_token(&self) -> Option<SyntaxToken> { support::token(&self.syntax, T![move]) }