rust/src/parser/event_parser/grammar/attributes.rs

61 lines
1.3 KiB
Rust
Raw Normal View History

2018-01-07 19:46:10 +01:00
use super::*;
2018-01-20 19:49:58 +01:00
#[derive(PartialEq, Eq)]
2018-01-11 21:01:12 +01:00
enum AttrKind {
Inner, Outer
}
2018-01-07 19:46:10 +01:00
pub(super) fn inner_attributes(p: &mut Parser) {
2018-01-15 19:44:30 +01:00
repeat(p, |p| attribute(p, AttrKind::Inner))
2018-01-07 19:46:10 +01:00
}
2018-01-11 21:01:12 +01:00
pub(super) fn outer_attributes(p: &mut Parser) {
2018-01-15 19:44:30 +01:00
repeat(p, |p| attribute(p, AttrKind::Outer))
2018-01-07 19:46:10 +01:00
}
2018-01-11 21:01:12 +01:00
fn attribute(p: &mut Parser, kind: AttrKind) -> bool {
2018-01-20 19:49:58 +01:00
if p.at(POUND) {
if kind == AttrKind::Inner && p.raw_lookahead(1) != EXCL {
return false;
}
2018-01-20 21:25:34 +01:00
let attr = p.start();
2018-01-20 19:49:58 +01:00
p.bump();
if kind == AttrKind::Inner {
p.bump();
}
p.expect(L_BRACK) && meta_item(p) && p.expect(R_BRACK);
2018-01-20 21:25:34 +01:00
attr.complete(p, ATTR);
2018-01-20 19:49:58 +01:00
true
} else {
false
2018-01-08 20:40:14 +01:00
}
2018-01-07 19:46:10 +01:00
}
fn meta_item(p: &mut Parser) -> bool {
2018-01-20 19:49:58 +01:00
if p.at(IDENT) {
2018-01-20 21:25:34 +01:00
let meta_item = p.start();
2018-01-20 19:49:58 +01:00
p.bump();
2018-01-07 19:46:10 +01:00
if p.eat(EQ) {
if !expressions::literal(p) {
p.error()
.message("expected literal")
.emit();
}
} else if p.eat(L_PAREN) {
comma_list(p, R_PAREN, meta_item_inner);
p.expect(R_PAREN);
}
2018-01-20 21:25:34 +01:00
meta_item.complete(p, META_ITEM);
2018-01-20 19:49:58 +01:00
true
} else {
false
}
2018-01-07 19:46:10 +01:00
}
fn meta_item_inner(p: &mut Parser) -> bool {
meta_item(p) || expressions::literal(p)
}