Possibly fix #225

This commit is contained in:
DJMcNab 2018-12-18 22:59:34 +00:00
parent 29bf389034
commit 97e70bf50f
2 changed files with 40 additions and 21 deletions

View file

@ -158,18 +158,18 @@ fn current_op(p: &Parser) -> (u8, Op) {
// Parses expression with binding power of at least bp. // Parses expression with binding power of at least bp.
fn expr_bp(p: &mut Parser, r: Restrictions, bp: u8) -> BlockLike { fn expr_bp(p: &mut Parser, r: Restrictions, bp: u8) -> BlockLike {
let mut lhs = match lhs(p, r) { let mut lhs = match lhs(p, r) {
Some(lhs) => { (Some(lhs), blocklike) => {
// test stmt_bin_expr_ambiguity // test stmt_bin_expr_ambiguity
// fn foo() { // fn foo() {
// let _ = {1} & 2; // let _ = {1} & 2;
// {1} &2; // {1} &2;
// } // }
if r.prefer_stmt && is_block(lhs.kind()) { if r.prefer_stmt && (is_block(lhs.kind()) || blocklike == Some(BlockLike::Block)) {
return BlockLike::Block; return BlockLike::Block;
} }
lhs lhs
} }
None => return BlockLike::NotBlock, (None, _) => return BlockLike::NotBlock,
}; };
loop { loop {
@ -213,7 +213,7 @@ const LHS_FIRST: TokenSet = token_set_union![
atom::ATOM_EXPR_FIRST, atom::ATOM_EXPR_FIRST,
]; ];
fn lhs(p: &mut Parser, r: Restrictions) -> Option<CompletedMarker> { fn lhs(p: &mut Parser, r: Restrictions) -> (Option<CompletedMarker>, Option<BlockLike>) {
let m; let m;
let kind = match p.current() { let kind = match p.current() {
// test ref_expr // test ref_expr
@ -246,19 +246,33 @@ fn lhs(p: &mut Parser, r: Restrictions) -> Option<CompletedMarker> {
if p.at_ts(EXPR_FIRST) { if p.at_ts(EXPR_FIRST) {
expr_bp(p, r, 2); expr_bp(p, r, 2);
} }
return Some(m.complete(p, RANGE_EXPR)); return (Some(m.complete(p, RANGE_EXPR)), None);
} }
_ => { _ => {
let lhs = atom::atom_expr(p, r)?; let (lhs_marker, macro_block_like) = atom::atom_expr(p, r);
return Some(postfix_expr(p, r, lhs));
if let Some(lhs_marker) = lhs_marker {
return (
Some(postfix_expr(p, r, lhs_marker, macro_block_like)),
macro_block_like,
);
} else {
return (None, None);
}
} }
}; };
expr_bp(p, r, 255); expr_bp(p, r, 255);
Some(m.complete(p, kind)) (Some(m.complete(p, kind)), None)
} }
fn postfix_expr(p: &mut Parser, r: Restrictions, mut lhs: CompletedMarker) -> CompletedMarker { fn postfix_expr(
let mut allow_calls = !r.prefer_stmt || !is_block(lhs.kind()); p: &mut Parser,
r: Restrictions,
mut lhs: CompletedMarker,
macro_block_like: Option<BlockLike>,
) -> CompletedMarker {
let mut allow_calls =
!r.prefer_stmt || !is_block(lhs.kind()) || macro_block_like != Some(BlockLike::Block);
loop { loop {
lhs = match p.current() { lhs = match p.current() {
// test stmt_postfix_expr_ambiguity // test stmt_postfix_expr_ambiguity
@ -406,21 +420,22 @@ fn arg_list(p: &mut Parser) {
// let _ = ::a::<b>; // let _ = ::a::<b>;
// let _ = format!(); // let _ = format!();
// } // }
fn path_expr(p: &mut Parser, r: Restrictions) -> CompletedMarker { fn path_expr(p: &mut Parser, r: Restrictions) -> (CompletedMarker, Option<BlockLike>) {
assert!(paths::is_path_start(p) || p.at(L_ANGLE)); assert!(paths::is_path_start(p) || p.at(L_ANGLE));
let m = p.start(); let m = p.start();
paths::expr_path(p); paths::expr_path(p);
match p.current() { let res = match p.current() {
L_CURLY if !r.forbid_structs => { L_CURLY if !r.forbid_structs => {
named_field_list(p); named_field_list(p);
m.complete(p, STRUCT_LIT) m.complete(p, STRUCT_LIT)
} }
EXCL => { EXCL => {
items::macro_call_after_excl(p); // TODO: Use return type (BlockLike) let block_like = items::macro_call_after_excl(p); // TODO: Use return type (BlockLike)
m.complete(p, MACRO_CALL) return (m.complete(p, MACRO_CALL), Some(block_like));
} }
_ => m.complete(p, PATH_EXPR), _ => m.complete(p, PATH_EXPR),
} };
(res, None)
} }
// test struct_lit // test struct_lit

View file

@ -61,12 +61,16 @@ pub(super) const ATOM_EXPR_FIRST: TokenSet = token_set_union![
const EXPR_RECOVERY_SET: TokenSet = token_set![LET_KW]; const EXPR_RECOVERY_SET: TokenSet = token_set![LET_KW];
pub(super) fn atom_expr(p: &mut Parser, r: Restrictions) -> Option<CompletedMarker> { pub(super) fn atom_expr(
p: &mut Parser,
r: Restrictions,
) -> (Option<CompletedMarker>, Option<BlockLike>) {
if let Some(m) = literal(p) { if let Some(m) = literal(p) {
return Some(m); return (Some(m), None);
} }
if paths::is_path_start(p) || p.at(L_ANGLE) { if paths::is_path_start(p) || p.at(L_ANGLE) {
return Some(path_expr(p, r)); let path_expr = path_expr(p, r);
return (Some(path_expr.0), path_expr.1);
} }
let la = p.nth(1); let la = p.nth(1);
let done = match p.current() { let done = match p.current() {
@ -94,7 +98,7 @@ pub(super) fn atom_expr(p: &mut Parser, r: Restrictions) -> Option<CompletedMark
// } // }
p.error("expected a loop"); p.error("expected a loop");
m.complete(p, ERROR); m.complete(p, ERROR);
return None; return (None, None);
} }
} }
} }
@ -111,10 +115,10 @@ pub(super) fn atom_expr(p: &mut Parser, r: Restrictions) -> Option<CompletedMark
BREAK_KW => break_expr(p), BREAK_KW => break_expr(p),
_ => { _ => {
p.err_recover("expected expression", EXPR_RECOVERY_SET); p.err_recover("expected expression", EXPR_RECOVERY_SET);
return None; return (None, None);
} }
}; };
Some(done) (Some(done), None)
} }
// test tuple_expr // test tuple_expr