aboutsummaryrefslogtreecommitdiff
path: root/crates/ra_syntax/src/grammar/expressions.rs
diff options
context:
space:
mode:
authorDJMcNab <[email protected]>2018-12-19 20:02:37 +0000
committerDJMcNab <[email protected]>2018-12-19 20:12:18 +0000
commitdef0a95d357f334b7329c3ee9e7da3c30563321c (patch)
tree972b6071caf13fd608a7ac3091f2158cfbadadcf /crates/ra_syntax/src/grammar/expressions.rs
parent4dce66ad319cbcbfae5f9e941b80c28cf546590c (diff)
Revert "Revert to f6f7c5"
This approach is correct, but it needs an addition to Restrictions too This reverts commit ad00d0c8a5f64142e6636e8b048204c8f8982f4a.
Diffstat (limited to 'crates/ra_syntax/src/grammar/expressions.rs')
-rw-r--r--crates/ra_syntax/src/grammar/expressions.rs39
1 files changed, 26 insertions, 13 deletions
diff --git a/crates/ra_syntax/src/grammar/expressions.rs b/crates/ra_syntax/src/grammar/expressions.rs
index 512823ddf..9d75bfb90 100644
--- a/crates/ra_syntax/src/grammar/expressions.rs
+++ b/crates/ra_syntax/src/grammar/expressions.rs
@@ -158,18 +158,19 @@ fn current_op(p: &Parser) -> (u8, Op) {
158// Parses expression with binding power of at least bp. 158// Parses expression with binding power of at least bp.
159fn expr_bp(p: &mut Parser, r: Restrictions, bp: u8) -> BlockLike { 159fn expr_bp(p: &mut Parser, r: Restrictions, bp: u8) -> BlockLike {
160 let mut lhs = match lhs(p, r) { 160 let mut lhs = match lhs(p, r) {
161 Some(lhs) => { 161 (Some(lhs), macro_blocklike) => {
162 // test stmt_bin_expr_ambiguity 162 // test stmt_bin_expr_ambiguity
163 // fn foo() { 163 // fn foo() {
164 // let _ = {1} & 2; 164 // let _ = {1} & 2;
165 // {1} &2; 165 // {1} &2;
166 // } 166 // }
167 if r.prefer_stmt && is_block(lhs.kind()) { 167 if r.prefer_stmt && (is_block(lhs.kind()) || macro_blocklike == Some(BlockLike::Block))
168 {
168 return BlockLike::Block; 169 return BlockLike::Block;
169 } 170 }
170 lhs 171 lhs
171 } 172 }
172 None => return BlockLike::NotBlock, 173 (None, _) => return BlockLike::NotBlock,
173 }; 174 };
174 175
175 loop { 176 loop {
@@ -213,7 +214,7 @@ const LHS_FIRST: TokenSet = token_set_union![
213 atom::ATOM_EXPR_FIRST, 214 atom::ATOM_EXPR_FIRST,
214]; 215];
215 216
216fn lhs(p: &mut Parser, r: Restrictions) -> Option<CompletedMarker> { 217fn lhs(p: &mut Parser, r: Restrictions) -> (Option<CompletedMarker>, Option<BlockLike>) {
217 let m; 218 let m;
218 let kind = match p.current() { 219 let kind = match p.current() {
219 // test ref_expr 220 // test ref_expr
@@ -246,18 +247,29 @@ fn lhs(p: &mut Parser, r: Restrictions) -> Option<CompletedMarker> {
246 if p.at_ts(EXPR_FIRST) { 247 if p.at_ts(EXPR_FIRST) {
247 expr_bp(p, r, 2); 248 expr_bp(p, r, 2);
248 } 249 }
249 return Some(m.complete(p, RANGE_EXPR)); 250 return (Some(m.complete(p, RANGE_EXPR)), None);
250 } 251 }
251 _ => { 252 _ => {
252 let lhs = atom::atom_expr(p, r)?; 253 let (lhs_marker, macro_block_like) = atom::atom_expr(p, r);
253 return Some(postfix_expr(p, r, lhs)); 254
255 if macro_block_like == Some(BlockLike::Block) {
256 return (lhs_marker, macro_block_like);
257 }
258 if let Some(lhs_marker) = lhs_marker {
259 return (Some(postfix_expr(p, r, lhs_marker)), macro_block_like);
260 } else {
261 return (None, None);
262 }
254 } 263 }
255 }; 264 };
256 expr_bp(p, r, 255); 265 expr_bp(p, r, 255);
257 Some(m.complete(p, kind)) 266 (Some(m.complete(p, kind)), None)
258} 267}
259 268
260fn postfix_expr(p: &mut Parser, r: Restrictions, mut lhs: CompletedMarker) -> CompletedMarker { 269fn postfix_expr(p: &mut Parser, r: Restrictions, mut lhs: CompletedMarker) -> CompletedMarker {
270 // Calls are disallowed if the type is a block and we prefer statements because the call cannot be disambiguated from a tuple
271 // E.g. `while true {break}();` is parsed as
272 // `while true {break}; ();`
261 let mut allow_calls = !r.prefer_stmt || !is_block(lhs.kind()); 273 let mut allow_calls = !r.prefer_stmt || !is_block(lhs.kind());
262 loop { 274 loop {
263 lhs = match p.current() { 275 lhs = match p.current() {
@@ -406,21 +418,22 @@ fn arg_list(p: &mut Parser) {
406// let _ = ::a::<b>; 418// let _ = ::a::<b>;
407// let _ = format!(); 419// let _ = format!();
408// } 420// }
409fn path_expr(p: &mut Parser, r: Restrictions) -> CompletedMarker { 421fn path_expr(p: &mut Parser, r: Restrictions) -> (CompletedMarker, Option<BlockLike>) {
410 assert!(paths::is_path_start(p) || p.at(L_ANGLE)); 422 assert!(paths::is_path_start(p) || p.at(L_ANGLE));
411 let m = p.start(); 423 let m = p.start();
412 paths::expr_path(p); 424 paths::expr_path(p);
413 match p.current() { 425 let res = match p.current() {
414 L_CURLY if !r.forbid_structs => { 426 L_CURLY if !r.forbid_structs => {
415 named_field_list(p); 427 named_field_list(p);
416 m.complete(p, STRUCT_LIT) 428 m.complete(p, STRUCT_LIT)
417 } 429 }
418 EXCL => { 430 EXCL => {
419 items::macro_call_after_excl(p); // TODO: Use return type (BlockLike) 431 let block_like = items::macro_call_after_excl(p); // TODO: Use return type (BlockLike)
420 m.complete(p, MACRO_CALL) 432 return (m.complete(p, MACRO_CALL), Some(block_like));
421 } 433 }
422 _ => m.complete(p, PATH_EXPR), 434 _ => m.complete(p, PATH_EXPR),
423 } 435 };
436 (res, None)
424} 437}
425 438
426// test struct_lit 439// test struct_lit