mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-14 14:13:58 +00:00
Merge #1111
1111: Add multi-byte token support in token tree to ast item list r=matklad a=edwin0cheng As discusion in https://github.com/rust-analyzer/rust-analyzer/pull/1105 , this PR add implement all multi-byte tokens in `ra_mbe` crate. Co-authored-by: Edwin Cheng <edwin0cheng@gmail.com>
This commit is contained in:
commit
be9a44e9ba
3 changed files with 139 additions and 20 deletions
1
Cargo.lock
generated
1
Cargo.lock
generated
|
@ -1040,6 +1040,7 @@ dependencies = [
|
||||||
name = "ra_mbe"
|
name = "ra_mbe"
|
||||||
version = "0.1.0"
|
version = "0.1.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
|
"itertools 0.8.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"ra_parser 0.1.0",
|
"ra_parser 0.1.0",
|
||||||
"ra_syntax 0.1.0",
|
"ra_syntax 0.1.0",
|
||||||
"ra_tt 0.1.0",
|
"ra_tt 0.1.0",
|
||||||
|
|
|
@ -8,5 +8,5 @@ authors = ["rust-analyzer developers"]
|
||||||
ra_syntax = { path = "../ra_syntax" }
|
ra_syntax = { path = "../ra_syntax" }
|
||||||
ra_parser = { path = "../ra_parser" }
|
ra_parser = { path = "../ra_parser" }
|
||||||
tt = { path = "../ra_tt", package = "ra_tt" }
|
tt = { path = "../ra_tt", package = "ra_tt" }
|
||||||
|
itertools = "0.8.0"
|
||||||
rustc-hash = "1.0.0"
|
rustc-hash = "1.0.0"
|
||||||
|
|
|
@ -113,6 +113,51 @@ struct TtToken {
|
||||||
text: SmolStr,
|
text: SmolStr,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Some helper functions
|
||||||
|
fn to_punct(tt: &tt::TokenTree) -> Option<&tt::Punct> {
|
||||||
|
if let tt::TokenTree::Leaf(tt::Leaf::Punct(pp)) = tt {
|
||||||
|
return Some(pp);
|
||||||
|
}
|
||||||
|
None
|
||||||
|
}
|
||||||
|
|
||||||
|
struct TokenPeek<'a, I>
|
||||||
|
where
|
||||||
|
I: Iterator<Item = &'a tt::TokenTree>,
|
||||||
|
{
|
||||||
|
iter: itertools::MultiPeek<I>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a, I> TokenPeek<'a, I>
|
||||||
|
where
|
||||||
|
I: Iterator<Item = &'a tt::TokenTree>,
|
||||||
|
{
|
||||||
|
fn next(&mut self) -> Option<&tt::TokenTree> {
|
||||||
|
self.iter.next()
|
||||||
|
}
|
||||||
|
|
||||||
|
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))
|
||||||
|
}
|
||||||
|
|
||||||
|
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))
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl TtTokenSource {
|
impl TtTokenSource {
|
||||||
fn new(tt: &tt::Subtree) -> TtTokenSource {
|
fn new(tt: &tt::Subtree) -> TtTokenSource {
|
||||||
let mut res = TtTokenSource { tokens: Vec::new() };
|
let mut res = TtTokenSource { tokens: Vec::new() };
|
||||||
|
@ -121,16 +166,27 @@ impl TtTokenSource {
|
||||||
}
|
}
|
||||||
fn convert_subtree(&mut self, sub: &tt::Subtree) {
|
fn convert_subtree(&mut self, sub: &tt::Subtree) {
|
||||||
self.push_delim(sub.delimiter, false);
|
self.push_delim(sub.delimiter, false);
|
||||||
sub.token_trees.iter().for_each(|tt| self.convert_tt(tt));
|
let mut peek = TokenPeek { iter: itertools::multipeek(sub.token_trees.iter()) };
|
||||||
|
while let Some(tt) = peek.iter.next() {
|
||||||
|
self.convert_tt(tt, &mut peek);
|
||||||
|
}
|
||||||
self.push_delim(sub.delimiter, true)
|
self.push_delim(sub.delimiter, true)
|
||||||
}
|
}
|
||||||
fn convert_tt(&mut self, tt: &tt::TokenTree) {
|
|
||||||
|
fn convert_tt<'a, I>(&mut self, tt: &tt::TokenTree, iter: &mut TokenPeek<'a, I>)
|
||||||
|
where
|
||||||
|
I: Iterator<Item = &'a tt::TokenTree>,
|
||||||
|
{
|
||||||
match tt {
|
match tt {
|
||||||
tt::TokenTree::Leaf(token) => self.convert_token(token),
|
tt::TokenTree::Leaf(token) => self.convert_token(token, iter),
|
||||||
tt::TokenTree::Subtree(sub) => self.convert_subtree(sub),
|
tt::TokenTree::Subtree(sub) => self.convert_subtree(sub),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
fn convert_token(&mut self, token: &tt::Leaf) {
|
|
||||||
|
fn convert_token<'a, I>(&mut self, token: &tt::Leaf, iter: &mut TokenPeek<'a, I>)
|
||||||
|
where
|
||||||
|
I: Iterator<Item = &'a tt::TokenTree>,
|
||||||
|
{
|
||||||
let tok = match token {
|
let tok = match token {
|
||||||
tt::Leaf::Literal(l) => TtToken {
|
tt::Leaf::Literal(l) => TtToken {
|
||||||
kind: SyntaxKind::INT_NUMBER, // FIXME
|
kind: SyntaxKind::INT_NUMBER, // FIXME
|
||||||
|
@ -138,21 +194,25 @@ impl TtTokenSource {
|
||||||
text: l.text.clone(),
|
text: l.text.clone(),
|
||||||
},
|
},
|
||||||
tt::Leaf::Punct(p) => {
|
tt::Leaf::Punct(p) => {
|
||||||
let kind = match p.char {
|
if let Some(tt) = Self::convert_multi_char_punct(p, iter) {
|
||||||
// lexer may produce combpund tokens for these ones
|
tt
|
||||||
'.' => DOT,
|
} else {
|
||||||
':' => COLON,
|
let kind = match p.char {
|
||||||
'=' => EQ,
|
// lexer may produce combpund tokens for these ones
|
||||||
'!' => EXCL,
|
'.' => DOT,
|
||||||
'-' => MINUS,
|
':' => COLON,
|
||||||
c => SyntaxKind::from_char(c).unwrap(),
|
'=' => EQ,
|
||||||
};
|
'!' => EXCL,
|
||||||
let text = {
|
'-' => MINUS,
|
||||||
let mut buf = [0u8; 4];
|
c => SyntaxKind::from_char(c).unwrap(),
|
||||||
let s: &str = p.char.encode_utf8(&mut buf);
|
};
|
||||||
SmolStr::new(s)
|
let text = {
|
||||||
};
|
let mut buf = [0u8; 4];
|
||||||
TtToken { kind, is_joint_to_next: p.spacing == tt::Spacing::Joint, text }
|
let s: &str = p.char.encode_utf8(&mut buf);
|
||||||
|
SmolStr::new(s)
|
||||||
|
};
|
||||||
|
TtToken { kind, is_joint_to_next: p.spacing == tt::Spacing::Joint, text }
|
||||||
|
}
|
||||||
}
|
}
|
||||||
tt::Leaf::Ident(ident) => {
|
tt::Leaf::Ident(ident) => {
|
||||||
let kind = SyntaxKind::from_keyword(ident.text.as_str()).unwrap_or(IDENT);
|
let kind = SyntaxKind::from_keyword(ident.text.as_str()).unwrap_or(IDENT);
|
||||||
|
@ -161,6 +221,64 @@ impl TtTokenSource {
|
||||||
};
|
};
|
||||||
self.tokens.push(tok)
|
self.tokens.push(tok)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn convert_multi_char_punct<'a, I>(
|
||||||
|
p: &tt::Punct,
|
||||||
|
iter: &mut TokenPeek<'a, I>,
|
||||||
|
) -> Option<TtToken>
|
||||||
|
where
|
||||||
|
I: Iterator<Item = &'a tt::TokenTree>,
|
||||||
|
{
|
||||||
|
if let Some((m, is_joint_to_next)) = iter.current_punct3(p) {
|
||||||
|
if let Some((kind, text)) = match m {
|
||||||
|
('<', '<', '=') => Some((SHLEQ, "<<=")),
|
||||||
|
('>', '>', '=') => Some((SHREQ, ">>=")),
|
||||||
|
('.', '.', '.') => Some((DOTDOTDOT, "...")),
|
||||||
|
('.', '.', '=') => Some((DOTDOTEQ, "..=")),
|
||||||
|
_ => None,
|
||||||
|
} {
|
||||||
|
iter.next();
|
||||||
|
iter.next();
|
||||||
|
return Some(TtToken { kind, is_joint_to_next, text: text.into() });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some((m, is_joint_to_next)) = iter.current_punct2(p) {
|
||||||
|
if let Some((kind, text)) = match m {
|
||||||
|
('<', '<') => Some((SHL, "<<")),
|
||||||
|
('>', '>') => Some((SHR, ">>")),
|
||||||
|
|
||||||
|
('|', '|') => Some((PIPEPIPE, "||")),
|
||||||
|
('&', '&') => Some((AMPAMP, "&&")),
|
||||||
|
('%', '=') => Some((PERCENTEQ, "%=")),
|
||||||
|
('*', '=') => Some((STAREQ, "*=")),
|
||||||
|
('/', '=') => Some((SLASHEQ, "/=")),
|
||||||
|
('^', '=') => Some((CARETEQ, "^=")),
|
||||||
|
|
||||||
|
('&', '=') => Some((AMPEQ, "&=")),
|
||||||
|
('|', '=') => Some((PIPEEQ, "|=")),
|
||||||
|
('-', '=') => Some((MINUSEQ, "-=")),
|
||||||
|
('+', '=') => Some((PLUSEQ, "+=")),
|
||||||
|
('>', '=') => Some((GTEQ, ">=")),
|
||||||
|
('<', '=') => Some((LTEQ, "<=")),
|
||||||
|
|
||||||
|
('-', '>') => Some((THIN_ARROW, "->")),
|
||||||
|
('!', '=') => Some((NEQ, "!=")),
|
||||||
|
('=', '>') => Some((FAT_ARROW, "=>")),
|
||||||
|
('=', '=') => Some((EQEQ, "==")),
|
||||||
|
('.', '.') => Some((DOTDOT, "..")),
|
||||||
|
(':', ':') => Some((COLONCOLON, "::")),
|
||||||
|
|
||||||
|
_ => None,
|
||||||
|
} {
|
||||||
|
iter.next();
|
||||||
|
return Some(TtToken { kind, is_joint_to_next, text: text.into() });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
None
|
||||||
|
}
|
||||||
|
|
||||||
fn push_delim(&mut self, d: tt::Delimiter, closing: bool) {
|
fn push_delim(&mut self, d: tt::Delimiter, closing: bool) {
|
||||||
let (kinds, texts) = match d {
|
let (kinds, texts) = match d {
|
||||||
tt::Delimiter::Parenthesis => ([L_PAREN, R_PAREN], "()"),
|
tt::Delimiter::Parenthesis => ([L_PAREN, R_PAREN], "()"),
|
||||||
|
|
Loading…
Reference in a new issue