Skip to content

Commit

Permalink
glue tokens when building token stream
Browse files Browse the repository at this point in the history
  • Loading branch information
matklad committed Aug 19, 2019
1 parent 8b932df commit 914e1f4
Show file tree
Hide file tree
Showing 3 changed files with 34 additions and 10 deletions.
40 changes: 32 additions & 8 deletions src/libsyntax/parse/lexer/tokentrees.rs
Original file line number Diff line number Diff line change
Expand Up @@ -39,29 +39,29 @@ struct TokenTreesReader<'a> {
impl<'a> TokenTreesReader<'a> {
// Parse a stream of tokens into a list of `TokenTree`s, up to an `Eof`.
fn parse_all_token_trees(&mut self) -> PResult<'a, TokenStream> {
let mut tts = Vec::new();
let mut buf = TokenStreamBuilder::default();

self.real_token();
while self.token != token::Eof {
tts.push(self.parse_token_tree()?);
buf.push(self.parse_token_tree()?);
}

Ok(TokenStream::new(tts))
Ok(buf.into_token_stream())
}

// Parse a stream of tokens into a list of `TokenTree`s, up to a `CloseDelim`.
fn parse_token_trees_until_close_delim(&mut self) -> TokenStream {
let mut tts = vec![];
let mut buf = TokenStreamBuilder::default();
loop {
if let token::CloseDelim(..) = self.token.kind {
return TokenStream::new(tts);
return buf.into_token_stream();
}

match self.parse_token_tree() {
Ok(tree) => tts.push(tree),
Ok(tree) => buf.push(tree),
Err(mut e) => {
e.emit();
return TokenStream::new(tts);
return buf.into_token_stream();
}
}
}
Expand Down Expand Up @@ -223,8 +223,32 @@ impl<'a> TokenTreesReader<'a> {
_ => {
self.token = token;
return;
},
}
}
}
}
}

#[derive(Default)]
struct TokenStreamBuilder {
buf: Vec<TreeAndJoint>,
}

impl TokenStreamBuilder {
fn push(&mut self, (tree, joint): TreeAndJoint) {
if let Some((TokenTree::Token(prev_token), Joint)) = self.buf.last() {
if let TokenTree::Token(token) = &tree {
if let Some(glued) = prev_token.glue(token) {
self.buf.pop();
self.buf.push((TokenTree::Token(glued), joint));
return;
}
}
}
self.buf.push((tree, joint))
}

fn into_token_stream(self) -> TokenStream {
TokenStream::new(self.buf)
}
}
2 changes: 1 addition & 1 deletion src/libsyntax/parse/token.rs
Original file line number Diff line number Diff line change
Expand Up @@ -551,7 +551,7 @@ impl Token {
}
}

crate fn glue(self, joint: Token) -> Option<Token> {
crate fn glue(&self, joint: &Token) -> Option<Token> {
let kind = match self.kind {
Eq => match joint.kind {
Eq => EqEq,
Expand Down
2 changes: 1 addition & 1 deletion src/libsyntax/tokenstream.rs
Original file line number Diff line number Diff line change
Expand Up @@ -414,7 +414,7 @@ impl TokenStreamBuilder {
let last_tree_if_joint = self.0.last().and_then(TokenStream::last_tree_if_joint);
if let Some(TokenTree::Token(last_token)) = last_tree_if_joint {
if let Some((TokenTree::Token(token), is_joint)) = stream.first_tree_and_joint() {
if let Some(glued_tok) = last_token.glue(token) {
if let Some(glued_tok) = last_token.glue(&token) {
let last_stream = self.0.pop().unwrap();
self.push_all_but_last_tree(&last_stream);
let glued_tt = TokenTree::Token(glued_tok);
Expand Down

0 comments on commit 914e1f4

Please sign in to comment.