use super::*; mod structs; pub(super) fn mod_contents(p: &mut Parser, stop_on_r_curly: bool) { attributes::inner_attributes(p); while !p.at(EOF) && !(stop_on_r_curly && p.at(R_CURLY)) { item(p); } } pub(super) const ITEM_FIRST: TokenSet = token_set![ EXTERN_KW, MOD_KW, USE_KW, STRUCT_KW, ENUM_KW, FN_KW, PUB_KW, POUND ]; fn item(p: &mut Parser) { let item = p.start(); attributes::outer_attributes(p); visibility(p); let la = p.nth(1); let item_kind = match p.current() { EXTERN_KW if la == CRATE_KW => { extern_crate_item(p); EXTERN_CRATE_ITEM } MOD_KW => { mod_item(p); MOD_ITEM } USE_KW => { use_item(p); USE_ITEM } STRUCT_KW => { structs::struct_item(p); STRUCT_ITEM } ENUM_KW => { structs::enum_item(p); ENUM_ITEM } FN_KW => { fn_item(p); FN_ITEM } L_CURLY => { item.abandon(p); error_block(p, "expected item"); return; } err_token => { item.abandon(p); let message = if err_token == SEMI { //TODO: if the item is incomplete, this message is misleading "expected item, found `;`\n\ consider removing this semicolon" } else { "expected item" }; p.err_and_bump(message); return; } }; item.complete(p, item_kind); } fn type_param_list(p: &mut Parser) { if !p.at(L_ANGLE) { return; } let m = p.start(); p.bump(); while !p.at(EOF) && !p.at(R_ANGLE) { match p.current() { LIFETIME => lifetime_param(p), IDENT => type_param(p), _ => p.err_and_bump("expected type parameter"), } if !p.at(R_ANGLE) && !p.expect(COMMA) { break; } } p.expect(R_ANGLE); m.complete(p, TYPE_PARAM_LIST); fn lifetime_param(p: &mut Parser) { assert!(p.at(LIFETIME)); let m = p.start(); p.bump(); if p.eat(COLON) { while p.at(LIFETIME) { p.bump(); if !p.eat(PLUS) { break; } } } m.complete(p, LIFETIME_PARAM); } fn type_param(p: &mut Parser) { assert!(p.at(IDENT)); let m = p.start(); p.bump(); m.complete(p, TYPE_PARAM); //TODO: bounds } } fn where_clause(_: &mut Parser) {} fn extern_crate_item(p: &mut Parser) { assert!(p.at(EXTERN_KW)); p.bump(); assert!(p.at(CRATE_KW)); p.bump(); p.expect(IDENT) && alias(p) && p.expect(SEMI); } fn mod_item(p: &mut Parser) { assert!(p.at(MOD_KW)); p.bump(); if p.expect(IDENT) && !p.eat(SEMI) { if p.expect(L_CURLY) { mod_contents(p, true); p.expect(R_CURLY); } } } pub(super) fn is_use_tree_start(kind: SyntaxKind) -> bool { kind == STAR || kind == L_CURLY } fn use_item(p: &mut Parser) { assert!(p.at(USE_KW)); p.bump(); use_tree(p); p.expect(SEMI); fn use_tree(p: &mut Parser) { let la = p.nth(1); let m = p.start(); match (p.current(), la) { (STAR, _) => p.bump(), (COLONCOLON, STAR) => { p.bump(); p.bump(); } (L_CURLY, _) | (COLONCOLON, L_CURLY) => { if p.at(COLONCOLON) { p.bump(); } nested_trees(p); } _ if paths::is_path_start(p) => { paths::use_path(p); match p.current() { AS_KW => { alias(p); } COLONCOLON => { p.bump(); match p.current() { STAR => { p.bump(); } L_CURLY => nested_trees(p), _ => { // is this unreachable? p.error().message("expected `{` or `*`").emit(); } } } _ => (), } } _ => { m.abandon(p); p.err_and_bump("expected one of `*`, `::`, `{`, `self`, `super`, `indent`"); return; } } m.complete(p, USE_TREE); } fn nested_trees(p: &mut Parser) { assert!(p.at(L_CURLY)); p.bump(); while !p.at(EOF) && !p.at(R_CURLY) { use_tree(p); if !p.at(R_CURLY) { p.expect(COMMA); } } p.expect(R_CURLY); } } fn fn_item(p: &mut Parser) { assert!(p.at(FN_KW)); p.bump(); p.expect(IDENT); if p.at(L_PAREN) { fn_value_parameters(p); } else { p.error().message("expected function arguments").emit(); } if p.at(L_CURLY) { p.expect(L_CURLY); p.expect(R_CURLY); } fn fn_value_parameters(p: &mut Parser) { assert!(p.at(L_PAREN)); p.bump(); p.expect(R_PAREN); } }