2019-03-03 09:40:03 +00:00
|
|
|
use crate::ParseError;
|
2019-04-07 16:12:07 +00:00
|
|
|
use crate::subtree_parser::Parser;
|
2019-04-24 15:01:32 +00:00
|
|
|
use smallvec::{SmallVec, smallvec};
|
2019-03-02 19:20:26 +00:00
|
|
|
|
2019-04-25 15:12:57 +00:00
|
|
|
#[derive(Debug, Clone)]
|
2019-01-31 12:26:01 +00:00
|
|
|
pub(crate) struct TtCursor<'a> {
|
|
|
|
subtree: &'a tt::Subtree,
|
|
|
|
pos: usize,
|
|
|
|
}
|
|
|
|
|
2019-04-23 18:59:38 +00:00
|
|
|
pub(crate) struct TtCursorMemento {
|
|
|
|
pos: usize,
|
|
|
|
}
|
|
|
|
|
2019-01-31 12:26:01 +00:00
|
|
|
impl<'a> TtCursor<'a> {
|
|
|
|
pub(crate) fn new(subtree: &'a tt::Subtree) -> TtCursor<'a> {
|
|
|
|
TtCursor { subtree, pos: 0 }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub(crate) fn is_eof(&self) -> bool {
|
|
|
|
self.pos == self.subtree.token_trees.len()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub(crate) fn current(&self) -> Option<&'a tt::TokenTree> {
|
|
|
|
self.subtree.token_trees.get(self.pos)
|
|
|
|
}
|
|
|
|
|
2019-03-03 09:40:03 +00:00
|
|
|
pub(crate) fn at_punct(&self) -> Option<&'a tt::Punct> {
|
2019-01-31 12:26:01 +00:00
|
|
|
match self.current() {
|
2019-03-03 09:40:03 +00:00
|
|
|
Some(tt::TokenTree::Leaf(tt::Leaf::Punct(it))) => Some(it),
|
|
|
|
_ => None,
|
2019-01-31 12:26:01 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub(crate) fn at_char(&self, char: char) -> bool {
|
|
|
|
match self.at_punct() {
|
2019-03-03 09:40:03 +00:00
|
|
|
Some(tt::Punct { char: c, .. }) if *c == char => true,
|
2019-01-31 12:26:01 +00:00
|
|
|
_ => false,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-03 09:40:03 +00:00
|
|
|
pub(crate) fn at_ident(&mut self) -> Option<&'a tt::Ident> {
|
2019-01-31 12:26:01 +00:00
|
|
|
match self.current() {
|
2019-03-03 09:40:03 +00:00
|
|
|
Some(tt::TokenTree::Leaf(tt::Leaf::Ident(i))) => Some(i),
|
|
|
|
_ => None,
|
2019-01-31 12:26:01 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-19 13:21:47 +00:00
|
|
|
pub(crate) fn at_literal(&mut self) -> Option<&'a tt::Literal> {
|
|
|
|
match self.current() {
|
|
|
|
Some(tt::TokenTree::Leaf(tt::Leaf::Literal(i))) => Some(i),
|
|
|
|
_ => None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-31 12:26:01 +00:00
|
|
|
pub(crate) fn bump(&mut self) {
|
|
|
|
self.pos += 1;
|
|
|
|
}
|
|
|
|
pub(crate) fn rev_bump(&mut self) {
|
|
|
|
self.pos -= 1;
|
|
|
|
}
|
|
|
|
|
2019-03-03 09:40:03 +00:00
|
|
|
pub(crate) fn eat(&mut self) -> Option<&'a tt::TokenTree> {
|
|
|
|
self.current().map(|it| {
|
|
|
|
self.bump();
|
|
|
|
it
|
|
|
|
})
|
2019-01-31 12:26:01 +00:00
|
|
|
}
|
|
|
|
|
2019-03-03 09:40:03 +00:00
|
|
|
pub(crate) fn eat_subtree(&mut self) -> Result<&'a tt::Subtree, ParseError> {
|
2019-03-02 19:20:26 +00:00
|
|
|
match self.current() {
|
|
|
|
Some(tt::TokenTree::Subtree(sub)) => {
|
2019-01-31 12:26:01 +00:00
|
|
|
self.bump();
|
2019-03-02 19:20:26 +00:00
|
|
|
Ok(sub)
|
2019-01-31 12:26:01 +00:00
|
|
|
}
|
2019-03-03 11:45:30 +00:00
|
|
|
_ => Err(ParseError::Expected(String::from("subtree"))),
|
2019-01-31 12:26:01 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-03 09:40:03 +00:00
|
|
|
pub(crate) fn eat_punct(&mut self) -> Option<&'a tt::Punct> {
|
2019-03-02 19:20:26 +00:00
|
|
|
self.at_punct().map(|it| {
|
2019-01-31 12:26:01 +00:00
|
|
|
self.bump();
|
2019-03-02 19:20:26 +00:00
|
|
|
it
|
|
|
|
})
|
2019-01-31 12:26:01 +00:00
|
|
|
}
|
|
|
|
|
2019-03-03 09:40:03 +00:00
|
|
|
pub(crate) fn eat_ident(&mut self) -> Option<&'a tt::Ident> {
|
2019-03-02 19:20:26 +00:00
|
|
|
self.at_ident().map(|i| {
|
2019-01-31 12:26:01 +00:00
|
|
|
self.bump();
|
2019-03-02 19:20:26 +00:00
|
|
|
i
|
|
|
|
})
|
2019-01-31 12:26:01 +00:00
|
|
|
}
|
|
|
|
|
2019-04-19 13:21:47 +00:00
|
|
|
pub(crate) fn eat_literal(&mut self) -> Option<&'a tt::Literal> {
|
|
|
|
self.at_literal().map(|i| {
|
|
|
|
self.bump();
|
|
|
|
i
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-04-06 12:14:28 +00:00
|
|
|
pub(crate) fn eat_path(&mut self) -> Option<tt::TokenTree> {
|
2019-04-07 16:12:07 +00:00
|
|
|
let parser = Parser::new(&mut self.pos, self.subtree);
|
|
|
|
parser.parse_path()
|
2019-04-06 04:12:32 +00:00
|
|
|
}
|
|
|
|
|
2019-04-13 10:38:31 +00:00
|
|
|
pub(crate) fn eat_expr(&mut self) -> Option<tt::TokenTree> {
|
|
|
|
let parser = Parser::new(&mut self.pos, self.subtree);
|
|
|
|
parser.parse_expr()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub(crate) fn eat_ty(&mut self) -> Option<tt::TokenTree> {
|
|
|
|
let parser = Parser::new(&mut self.pos, self.subtree);
|
|
|
|
parser.parse_ty()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub(crate) fn eat_pat(&mut self) -> Option<tt::TokenTree> {
|
|
|
|
let parser = Parser::new(&mut self.pos, self.subtree);
|
|
|
|
parser.parse_pat()
|
|
|
|
}
|
|
|
|
|
2019-04-17 04:34:43 +00:00
|
|
|
pub(crate) fn eat_stmt(&mut self) -> Option<tt::TokenTree> {
|
|
|
|
let parser = Parser::new(&mut self.pos, self.subtree);
|
|
|
|
parser.parse_stmt()
|
|
|
|
}
|
|
|
|
|
2019-04-19 10:30:43 +00:00
|
|
|
pub(crate) fn eat_block(&mut self) -> Option<tt::TokenTree> {
|
|
|
|
let parser = Parser::new(&mut self.pos, self.subtree);
|
|
|
|
parser.parse_block()
|
|
|
|
}
|
|
|
|
|
2019-04-19 11:33:29 +00:00
|
|
|
pub(crate) fn eat_meta(&mut self) -> Option<tt::TokenTree> {
|
|
|
|
let parser = Parser::new(&mut self.pos, self.subtree);
|
|
|
|
parser.parse_meta()
|
|
|
|
}
|
|
|
|
|
2019-04-18 02:21:36 +00:00
|
|
|
pub(crate) fn eat_item(&mut self) -> Option<tt::TokenTree> {
|
|
|
|
let parser = Parser::new(&mut self.pos, self.subtree);
|
|
|
|
parser.parse_item()
|
|
|
|
}
|
|
|
|
|
2019-04-19 13:15:19 +00:00
|
|
|
pub(crate) fn eat_lifetime(&mut self) -> Option<tt::TokenTree> {
|
2019-04-19 13:43:41 +00:00
|
|
|
// check if it start from "`"
|
|
|
|
if let Some(ident) = self.at_ident() {
|
|
|
|
if ident.text.chars().next()? != '\'' {
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-19 13:15:19 +00:00
|
|
|
self.eat_ident().cloned().map(|ident| tt::Leaf::from(ident).into())
|
|
|
|
}
|
|
|
|
|
2019-05-02 13:24:51 +00:00
|
|
|
pub(crate) fn try_eat_vis(&mut self) -> Option<tt::TokenTree> {
|
|
|
|
// `vis` matcher is optional
|
|
|
|
let old_pos = self.pos;
|
2019-04-19 13:38:26 +00:00
|
|
|
let parser = Parser::new(&mut self.pos, self.subtree);
|
2019-05-02 13:24:51 +00:00
|
|
|
|
|
|
|
let res = parser.parse_vis();
|
|
|
|
if res.is_none() {
|
|
|
|
self.pos = old_pos;
|
|
|
|
}
|
|
|
|
res
|
2019-05-02 02:08:11 +00:00
|
|
|
}
|
2019-04-19 13:38:26 +00:00
|
|
|
|
2019-03-03 09:40:03 +00:00
|
|
|
pub(crate) fn expect_char(&mut self, char: char) -> Result<(), ParseError> {
|
2019-01-31 12:26:01 +00:00
|
|
|
if self.at_char(char) {
|
|
|
|
self.bump();
|
2019-03-03 09:40:03 +00:00
|
|
|
Ok(())
|
|
|
|
} else {
|
2019-03-03 11:45:30 +00:00
|
|
|
Err(ParseError::Expected(format!("`{}`", char)))
|
2019-01-31 12:26:01 +00:00
|
|
|
}
|
|
|
|
}
|
2019-04-23 18:59:38 +00:00
|
|
|
|
2019-04-24 15:01:32 +00:00
|
|
|
fn eat_punct3(&mut self, p: &tt::Punct) -> Option<SmallVec<[tt::Punct; 3]>> {
|
|
|
|
let sec = self.eat_punct()?.clone();
|
|
|
|
let third = self.eat_punct()?.clone();
|
|
|
|
Some(smallvec![p.clone(), sec, third])
|
|
|
|
}
|
|
|
|
|
|
|
|
fn eat_punct2(&mut self, p: &tt::Punct) -> Option<SmallVec<[tt::Punct; 3]>> {
|
|
|
|
let sec = self.eat_punct()?.clone();
|
|
|
|
Some(smallvec![p.clone(), sec])
|
|
|
|
}
|
|
|
|
|
|
|
|
fn eat_multi_char_punct<'b, I>(
|
|
|
|
&mut self,
|
|
|
|
p: &tt::Punct,
|
|
|
|
iter: &mut TokenPeek<'b, I>,
|
|
|
|
) -> Option<SmallVec<[tt::Punct; 3]>>
|
|
|
|
where
|
|
|
|
I: Iterator<Item = &'b tt::TokenTree>,
|
|
|
|
{
|
|
|
|
if let Some((m, _)) = iter.current_punct3(p) {
|
|
|
|
if let r @ Some(_) = match m {
|
|
|
|
('<', '<', '=') | ('>', '>', '=') | ('.', '.', '.') | ('.', '.', '=') => {
|
|
|
|
self.eat_punct3(p)
|
|
|
|
}
|
|
|
|
_ => None,
|
|
|
|
} {
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if let Some((m, _)) = iter.current_punct2(p) {
|
|
|
|
if let r @ Some(_) = match m {
|
|
|
|
('<', '=')
|
|
|
|
| ('>', '=')
|
|
|
|
| ('+', '=')
|
|
|
|
| ('-', '=')
|
|
|
|
| ('|', '=')
|
|
|
|
| ('&', '=')
|
|
|
|
| ('^', '=')
|
|
|
|
| ('/', '=')
|
|
|
|
| ('*', '=')
|
|
|
|
| ('%', '=')
|
|
|
|
| ('&', '&')
|
|
|
|
| ('|', '|')
|
|
|
|
| ('<', '<')
|
|
|
|
| ('>', '>')
|
|
|
|
| ('-', '>')
|
|
|
|
| ('!', '=')
|
|
|
|
| ('=', '>')
|
|
|
|
| ('=', '=')
|
|
|
|
| ('.', '.')
|
|
|
|
| (':', ':') => self.eat_punct2(p),
|
|
|
|
|
|
|
|
_ => None,
|
|
|
|
} {
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
None
|
|
|
|
}
|
|
|
|
|
|
|
|
pub(crate) fn eat_seperator(&mut self) -> Option<crate::Separator> {
|
|
|
|
match self.eat()? {
|
|
|
|
tt::TokenTree::Leaf(tt::Leaf::Literal(lit)) => {
|
|
|
|
Some(crate::Separator::Literal(lit.clone()))
|
|
|
|
}
|
|
|
|
tt::TokenTree::Leaf(tt::Leaf::Ident(ident)) => {
|
|
|
|
Some(crate::Separator::Ident(ident.clone()))
|
|
|
|
}
|
|
|
|
tt::TokenTree::Leaf(tt::Leaf::Punct(punct)) => {
|
|
|
|
match punct.char {
|
|
|
|
'*' | '+' | '?' => return None,
|
|
|
|
_ => {}
|
|
|
|
};
|
|
|
|
|
|
|
|
// FIXME: The parser is only handle some compositeable punct,
|
|
|
|
// But at this phase, some punct still is jointed.
|
|
|
|
// So we by pass that check here.
|
|
|
|
let mut peekable = TokenPeek::new(self.subtree.token_trees[self.pos..].iter());
|
|
|
|
let puncts = self.eat_multi_char_punct(punct, &mut peekable);
|
|
|
|
let puncts = puncts.unwrap_or_else(|| smallvec![punct.clone()]);
|
|
|
|
|
|
|
|
Some(crate::Separator::Puncts(puncts))
|
|
|
|
}
|
|
|
|
_ => None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-23 18:59:38 +00:00
|
|
|
#[must_use]
|
|
|
|
pub(crate) fn save(&self) -> TtCursorMemento {
|
|
|
|
TtCursorMemento { pos: self.pos }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub(crate) fn rollback(&mut self, memento: TtCursorMemento) {
|
|
|
|
self.pos = memento.pos;
|
|
|
|
}
|
2019-01-31 12:26:01 +00:00
|
|
|
}
|
2019-05-02 02:02:17 +00:00
|
|
|
|
|
|
|
pub(crate) struct TokenPeek<'a, I>
|
|
|
|
where
|
|
|
|
I: Iterator<Item = &'a tt::TokenTree>,
|
|
|
|
{
|
|
|
|
iter: itertools::MultiPeek<I>,
|
|
|
|
}
|
|
|
|
|
|
|
|
// helper function
|
|
|
|
fn to_punct(tt: &tt::TokenTree) -> Option<&tt::Punct> {
|
|
|
|
if let tt::TokenTree::Leaf(tt::Leaf::Punct(pp)) = tt {
|
|
|
|
return Some(pp);
|
|
|
|
}
|
|
|
|
None
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a, I> TokenPeek<'a, I>
|
|
|
|
where
|
|
|
|
I: Iterator<Item = &'a tt::TokenTree>,
|
|
|
|
{
|
|
|
|
pub fn new(iter: I) -> Self {
|
|
|
|
TokenPeek { iter: itertools::multipeek(iter) }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn current_punct2(&mut self, p: &tt::Punct) -> Option<((char, char), bool)> {
|
|
|
|
if p.spacing != tt::Spacing::Joint {
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
|
|
|
|
self.iter.reset_peek();
|
|
|
|
let p1 = to_punct(self.iter.peek()?)?;
|
|
|
|
Some(((p.char, p1.char), p1.spacing == tt::Spacing::Joint))
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn current_punct3(&mut self, p: &tt::Punct) -> Option<((char, char, char), bool)> {
|
|
|
|
self.current_punct2(p).and_then(|((p0, p1), last_joint)| {
|
|
|
|
if !last_joint {
|
|
|
|
None
|
|
|
|
} else {
|
|
|
|
let p2 = to_punct(*self.iter.peek()?)?;
|
|
|
|
Some(((p0, p1, p2.char), p2.spacing == tt::Spacing::Joint))
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|